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

User id and group id integration in NPF #45

Closed
wants to merge 13 commits into from

Conversation

Emmankoko
Copy link

we can set our uids or gids as numbers,(1001), names(Kojo), or a variable that holds a number of name($my_id)

pass all user 1000
pass all user Kojo
pass all user $myuser(containing Kojo or 1001)

same applies with group id.
NB: keyword for gid is usergroup
eg. pass all usergroup wheel(we cannot use group because group already exist)

we can set our uids or gids as numbers,(1001), names(Kojo), or a variable
that holds a number of name($my_id)

pass all user 1000
pass all user Kojo
pass all user $myuser(containing Kojo or 1001)

same applies with group id.
NB: keyword for gid is usergroup
eg. pass all usergroup wheel(we cannot use group because group already exist)
so we perform a comparison with the user on our rule and the user for the process
involved, if it matches, proceed to block or pass.
if it doesn't match, reveres from pass to block.

if we pass on user Christos, it has to block on user Martin.
iif we block on user Martin, it passes on user Christos.

(pass->block)just rip of pass bit fields from return flag and insert block field
(block->pass)same here. just rip block bit fields and insert pass field
uid_item
: uid
{
if (($$ = calloc(1, sizeof(struct r_uid))) == NULL)
Copy link
Contributor

Choose a reason for hiding this comment

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

write a method that takes 3 arguments instead of repeating the code.

struct passwd *pw;

if ((pw = getpwnam(user)) == NULL) {
return 1;
Copy link
Contributor

Choose a reason for hiding this comment

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

Don't use 1 as an error, what is the difference between unknown and error?

npfctl_parse_user(const char *user)
{
uint32_t uid;
if (!(strcmp(user, "unknown")))
Copy link
Contributor

Choose a reason for hiding this comment

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

no parens around strcmp

return -1;

matched = 0;
if ( rl->uid)
Copy link
Contributor

Choose a reason for hiding this comment

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

whitespace


group_id
: /* empty */ { $$ = NULL; }
| USERGROUP gids { $$ = $2; }
Copy link
Contributor

Choose a reason for hiding this comment

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

Are you calling this "USERGROUP" instead of "GROUP" because group is already used?

@Emmankoko
Copy link
Author

Emmankoko commented Mar 17, 2025 via email

functions implementing for setting the uid and gid structure.
group keyword used gids
whitespace fixed
no parens on strcmp
static int
npf_match(uint8_t op, uint32_t rid1, uint32_t rid2, uint32_t id_lp)
{
switch (op) {
Copy link
Contributor

Choose a reason for hiding this comment

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

no parens in return.

lib/libnpf/npf.c Outdated
}

int
npf_rule_setrugrp(nl_rule_t *rl, struct r_gid *gid)
Copy link
Contributor

Choose a reason for hiding this comment

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

why ugrp instead of group?

lib/libnpf/npf.c Outdated
int
npf_rule_setruser(nl_rule_t *rl, struct r_uid *uid)
{
int error = 0;
Copy link
Contributor

Choose a reason for hiding this comment

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

why use error? delete the var and return directly.

const uint64_t *uid = nvlist_get_number_array(req, "r_user", &uitems);
KASSERT(uitems == 3);

r_uid.uid[0] = (uint32_t)uid[0];
Copy link
Contributor

Choose a reason for hiding this comment

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

these should be (uid_t) not (uint32_t).

Copy link
Contributor

Choose a reason for hiding this comment

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

but... you are using so much uint32_t and uid_t/gid_t interchangeably, so why don't you just call it id in the structure, make it uint32_t and then merge all the duplicated code. You can put in a comment why (both uid_t and gid_t and uint32_t and using uint32_t saves a lot of code duplication).

@@ -59,6 +59,16 @@ typedef union {
uint32_t word32[4];
} npf_addr_t;

struct r_uid {
Copy link
Contributor

Choose a reason for hiding this comment

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

If you merge the two structs call it r_id { uint32_t id[2]; uint8_t op; };
Then you can merge most of the code.

int
npf_socket_lookup_uid(npf_cache_t *npc, int dir, uint32_t *uid)
{
struct inpcbtable *tb = NULL;
Copy link
Contributor

Choose a reason for hiding this comment

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

tb is NULL, why put it in a variable instead of passing NULL?


if (npf_iscached(npc, NPC_IP4)) {
so = npf_ip_socket(npc, tb, dir);

Copy link
Contributor

Choose a reason for hiding this comment

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

remove blank line


} else if (npf_iscached(npc, NPC_IP6)) {
so = npf_ip6_socket(npc, tb, dir);
}
Copy link
Contributor

Choose a reason for hiding this comment

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

else return -1; then you don't need to over-initialize so = NULL;

}

int
npf_socket_lookup_gid(npf_cache_t *npc, int dir, uint32_t *gid)
Copy link
Contributor

Choose a reason for hiding this comment

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

this is duplicated for uid and gid merge them.

@@ -118,6 +118,9 @@ struct npf_rule {
LIST_ENTRY(npf_rule) r_aentry;
nvlist_t * r_info;
size_t r_info_len;

rid_t *uid;
Copy link
Contributor

Choose a reason for hiding this comment

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

I don't know about storing pointers here and needing to allocate and free them seems to be a lot of work for very little benefit since the structures are small. Perhaps just store the structures and add NPF_OP_INVALID as the first element of the enum to mean 0 == unused?

Copy link
Author

Choose a reason for hiding this comment

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

I agree. too small to be allocating for it not optimal.

/* again, use a single rule getid function both uid and gids */
static rid_t
npf_rule_getrid(const nvlist_t *req, const char *name)
{
Copy link
Contributor

@zoulasc zoulasc Mar 28, 2025

Choose a reason for hiding this comment

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

Why all this machinery? get rid of npf_rule_getrid and put all of it in npf_rule_setrid and operate on the pointer directly.

}

void
npf_rule_setuid(const nvlist_t *req, npf_rule_t *rl, const char *name)
Copy link
Contributor

@zoulasc zoulasc Mar 28, 2025

Choose a reason for hiding this comment

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

and these two are overkill too. Just do it directly.

Copy link
Author

Choose a reason for hiding this comment

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

I don't think It will be possible to do it directly here means I would have to call it in npf_ctl.c where it extracts the nvpair. I would have to pass rl->uid as argument to the function and it won't work because rl is an opaque type in npf_ruleset TU and hence cannot be dereferenced out of the ruleset TU

Choose a reason for hiding this comment

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

Also, don't memcpy() structures -- just assign them.

* if socket id doesn't match any of rule ids, reverse action
*/
int
npf_rule_match_user(npf_rule_t *rl, npf_cache_t *npc, int dir)
Copy link
Contributor

Choose a reason for hiding this comment

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

these two can be merged too, by passing the function npf_socket_lookup_uid/gid as a parameter.

double fpnum;
npfvar_t * var;
addr_port_t addrport;
filt_opts_t filtopts;
opt_proto_t optproto;
rule_group_t rulegroup;
rid_t *uid;
Copy link
Contributor

Choose a reason for hiding this comment

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

those don't have to be pointers either. makes the code simpler.

@Emmankoko Emmankoko closed this Mar 29, 2025
@Emmankoko Emmankoko reopened this Mar 29, 2025
@Emmankoko Emmankoko closed this Mar 29, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants