Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Config file support #326

Open
wants to merge 6 commits into
base: main
Choose a base branch
from
Open

Config file support #326

wants to merge 6 commits into from

Conversation

dacav
Copy link
Contributor

@dacav dacav commented Dec 13, 2024

Fixes #265

@dacav dacav requested a review from LDVG December 13, 2024 15:08
@dacav dacav self-assigned this Dec 13, 2024
@dacav dacav marked this pull request as draft December 13, 2024 15:08
@dacav
Copy link
Contributor Author

dacav commented Dec 13, 2024

  • Manpages update

@dacav dacav force-pushed the dacav/config_file branch 2 times, most recently from 611e678 to 257c433 Compare December 16, 2024 23:23
cfg.c Outdated Show resolved Hide resolved
cfg.c Outdated Show resolved Hide resolved
README Outdated Show resolved Hide resolved
man/pam_u2f.8.txt Outdated Show resolved Hide resolved
@dacav dacav force-pushed the dacav/config_file branch 2 times, most recently from 81fb36f to e9d2621 Compare December 17, 2024 08:24
@dacav dacav marked this pull request as ready for review December 17, 2024 08:36
@dacav dacav force-pushed the dacav/config_file branch 2 times, most recently from 3b245e0 to 5365c0d Compare December 17, 2024 08:50
Copy link
Contributor

@LDVG LDVG left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks like a pretty good start to me. I've left mostly nitpicking comments in-line. 🙂

Makefile.am Outdated Show resolved Hide resolved
README Outdated Show resolved Hide resolved
README Outdated Show resolved Hide resolved
README Outdated Show resolved Hide resolved
README Outdated Show resolved Hide resolved
man/pam_u2f.8.txt Outdated Show resolved Hide resolved
man/pam_u2f.8.txt Outdated Show resolved Hide resolved
tests/cfg.c Outdated Show resolved Hide resolved
tests/cfg.c Outdated Show resolved Hide resolved
tests/cfg.c Show resolved Hide resolved
@dacav dacav marked this pull request as draft December 17, 2024 14:45
@dacav dacav force-pushed the dacav/config_file branch from 5365c0d to 98fdb4d Compare December 18, 2024 20:57
@dacav
Copy link
Contributor Author

dacav commented Dec 18, 2024

Current situation:

What is left to do for the current PR (soon follow-up):

  • A proper MAX_PATH (currently using hard-wired 4096 - Linux).
  • A temporary --with-security-confdir solution, mimicking --with-pam-dir (future refinement)
  • Trailing comments and spaces around '='
  • Only file owner can write (hopefully that's root)
  • Only root can own the file (requires fakeroot-ish behaviour)
  • Comprehensive config unit test
  • Inject the security-confdir into manpages (currently "/etc/security" is a fixed string

@dacav dacav force-pushed the dacav/config_file branch 6 times, most recently from 0adeda2 to a6e93f3 Compare December 23, 2024 22:50
@dacav dacav marked this pull request as ready for review January 2, 2025 08:51
@dacav
Copy link
Contributor Author

dacav commented Jan 2, 2025

Oh no. I figured out this even more work!

Autoconf BUG (?)

  1. ./configure --with-sconf-dir=/a
  2. make && sudo make install
  3. ./configure --with-sconf-dir=/b
  4. make # WILL NOT RECOMPILE (well, only manpage)
  5. sudo make install # not what expected: old binary is installed again

This is a little pathological (maybe low prio?) but actually a defect.

Vulnerable for untidy sysadmin:

Currently my code is checking that the configuration file (1) belongs to root, (2) can only be written by root, (3) is not a symlink. But I'm not checking the status of the above directories... Therefore, assuming that pam_u2f.so is configured to use /my/hopefully/notalink/pam_u2f.conf as a configuration file, in the (also pathological?) situation the unprivileged user might be able to hijack the configuration:

  user@localhost:/my$ ls -lR .
  .:
  total 4
  drwxrwxrwx 4 root root 4096 Jan  2 11:23 hopefully [1]

  ./hopefully:
  total 8
  drwxr-xr-x 2 root root 4096 Jan  2 11:15 notalink
  drwxr-xr-x 2 root root 4096 Jan  2 11:23 oops

  ./hopefully/notalink:
  total 4
  -rw-r--r-- 1 root root 13 Jan  2 11:05 pam_u2f.conf [2]

  ./hopefully/oops:
  total 4
  -rw-r--r-- 1 root root 13 Jan  2 11:23 pam_u2f.conf [2]

[1] world-writable directory
[2] a root-owned-not-writable copy exists

Attack:

  user@localhost:/my$ cd hopefully
  user@localhost:/my/hopefully$ mv notalink/ goodbye
  user@localhost:/my/hopefully$ ln -s oops notalink
  user@localhost:/my/hopefully$ cd ..
  user@localhost:/my$ ls -lR
  .:
  total 4
  drwxrwxrwx 4 root root 4096 Jan  2 11:25 hopefully

  ./hopefully:
  total 8
  drwxr-xr-x 2 root root 4096 Jan  2 11:15 goodbye
  lrwxrwxrwx 1 user user    4 Jan  2 11:25 notalink -> oops
  drwxr-xr-x 2 root root 4096 Jan  2 11:23 oops

  ./hopefully/goodbye:
  total 4
  -rw-r--r-- 1 root root 13 Jan  2 11:05 pam_u2f.conf

  ./hopefully/oops:
  total 4
  -rw-r--r-- 1 root root 15 Jan  2 11:23 pam_u2f.conf

@dacav dacav force-pushed the dacav/config_file branch 2 times, most recently from 36e1255 to 58e0177 Compare January 8, 2025 12:45
@dacav dacav marked this pull request as draft January 8, 2025 12:59
@dacav dacav force-pushed the dacav/config_file branch 3 times, most recently from 9d2c5d1 to f90d66a Compare January 13, 2025 08:43
@dacav dacav marked this pull request as ready for review January 13, 2025 08:44
@dacav dacav force-pushed the dacav/config_file branch 2 times, most recently from 3a0ebb6 to 6faab0d Compare January 15, 2025 07:57
@dacav
Copy link
Contributor Author

dacav commented Jan 15, 2025

Rebased on current main.

Copy link
Contributor

@LDVG LDVG left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

last iteration of nit picking, this looks good to go otherwise. nice work! :)

//
// On success returns PAM_SUCCESS
// On failure returns PAM_SERVICE_ERR and sets errno to indicate the error.
static int open_safely(int *outfd, size_t *outsize, const char *path) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not a huge fan of using errno this way. It appears we only need to be able to discern ENOENT, success, and a generic error for everything else. What about returning -ENOENT, a non-negative integer (fd), and -EINVAL respectively instead of the PAM_* value?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The idea behind this is to use PAM_* error codes consistently everywhere for error handling.
It is feasible to do as you suggest, but that would be the only function that does error checking differently.
If that is in your opinion more beautiful than writing errno, then I can comply with your request. Please let me know :-)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe we have several other functions that return 0 or 1 or some other error code, I'd rather use "internal" return values for these internal functions, we only really need to concern ourselves with PAM_ return values for functions that set retval in pam_sm_authenticate()

cfg.c Outdated
Comment on lines 108 to 112
len = strlen(path);
if (!len || path[0] != '/' || path[len - 1] == '/') {
errno = EINVAL;
return PAM_SERVICE_ERR;
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wouldn't *path != '/' suffice? S_ISREG should catch trying to open up a directory.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct.
A left-over from previous iteration (path security) where I used strtok_r to split string on /.
On such situation this was a corner case to check.

cfg.c Outdated
if (fd == -1)
return PAM_SERVICE_ERR;

if (fstat(fd, &st))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd prefer to be explicit here with fstat(fd, &st) != 0.

cfg.c Outdated
Comment on lines 265 to 267
if (strncmp(argv[i], "conf=", strlen("conf=")) == 0)
continue;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since cfg_load_arg() does not handle conf=, should we just drop this short circuiting?

cfg.h Outdated
#include <stdio.h>

#define CFG_DEFAULT_PATH (SCONFDIR "/pam_u2f.conf")
#define CFG_MAX_FILE_SIZE 4096 // Arbitrary
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd drop the comment

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When I see a constant, my first thought is "why not double or half that amount"?
Anyway... OK. 😁 no strong opinion.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No strong opinions from my side either, I just don't get that much more information from this comment :)

README Outdated
A configuration file can be used to set the default
<<moduleArguments,module arguments>>.

The file has a `name = value` format, with comments starting with the `#`
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This section reads like a list, but is not a list, should we rephrase or add bullet points?

(same comment for man pages. as an additional follow up, we might want to consider a pam_u2f.conf.5)

@dacav dacav force-pushed the dacav/config_file branch from 6faab0d to 1a9a44a Compare January 23, 2025 12:49
@dacav
Copy link
Contributor Author

dacav commented Jan 23, 2025

Applied requested changes, and rebased on current main branch.

@dacav dacav force-pushed the dacav/config_file branch 3 times, most recently from 47176b2 to c1e634b Compare January 28, 2025 13:10
dacav added 6 commits January 28, 2025 17:36
Having it into another module will prevent the code from being messy
later.

The parsing procedure is taken verbatim: no semantic change, no
behavioural change.
The configuration file defines the default behaviour of pam_u2f.
Individual module invocations under /etc/pam.d can override
settings.

The file-system location of the config file is by default
$sysconfdir/security/pam_u2f.conf, where $sysconfdir is supplied at
build time.

A new module configuration, "conf=", allows to override it at
runtime. Only absolute paths are accepted.
- split-input format: add trailing blob for config file

  The corpus needs some update.

- wrappers (-Wl,--wrap) integrate fuzzing of the configuration file.

  The configuration file, mutated by the fuzzer, is made
  available to the cfg.c implementation.

  The mock-up works under the assumption that only the cfg.c
  module works by opening "/" with open(3), and follows up with an
  alternation of openat(3) and fstat(3) calls.
The SCONFDIR variable is expanded by the build system within
the pam_u2f.8 manpage.
@dacav dacav force-pushed the dacav/config_file branch from c1e634b to 2332f47 Compare January 28, 2025 16:37
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

[RFE] Allow to set pam_u2f arguments in configuration file
2 participants