1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
/*
* Copyright (C) 2009-2010 Red Hat, Inc.
* Copyright (C) 2013 Canonical Limited
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the licence, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
* Authors: Ryan Lortie <desrt@desrt.ca>
* Matthias Clasen <mclasen@redhat.com>
*/
#include "config.h"
#include "user-classify.h"
#include <string.h>
#include <unistd.h>
static const char *default_excludes[] = {
"bin",
"root",
"daemon",
"adm",
"lp",
"sync",
"shutdown",
"halt",
"mail",
"news",
"uucp",
"nobody",
"postgres",
"pvm",
"rpm",
"nfsnobody",
"pcap",
"mysql",
"ftp",
"games",
"man",
"at",
"gdm",
"gnome-initial-setup",
"git"
};
static gboolean
user_classify_is_blacklisted (const char *username)
{
static GHashTable *exclusions;
if (exclusions == NULL) {
guint i;
exclusions = g_hash_table_new (g_str_hash, g_str_equal);
for (i = 0; i < G_N_ELEMENTS (default_excludes); i++) {
g_hash_table_add (exclusions, (gpointer) default_excludes[i]);
}
}
if (g_hash_table_contains (exclusions, username)) {
return TRUE;
}
return FALSE;
}
static gboolean
is_invalid_shell (const char *shell)
{
g_autofree gchar *basename = NULL;
int ret = FALSE;
#ifdef HAVE_GETUSERSHELL
/* getusershell returns a whitelist of valid shells.
* assume the shell is invalid unless there is a match */
ret = TRUE;
char *valid_shell;
setusershell ();
while ((valid_shell = getusershell ()) != NULL) {
if (g_strcmp0 (shell, valid_shell) != 0)
continue;
ret = FALSE;
break;
}
endusershell ();
#endif
/* always check for false and nologin since they are sometimes included by getusershell */
basename = g_path_get_basename (shell);
if (shell[0] == '\0') {
return TRUE;
} else if (g_strcmp0 (basename, "nologin") == 0) {
return TRUE;
} else if (g_strcmp0 (basename, "false") == 0) {
return TRUE;
}
return ret;
}
gboolean
user_classify_is_human (uid_t uid,
const gchar *username,
const gchar *shell)
{
if (user_classify_is_blacklisted (username))
return FALSE;
if (shell != NULL && is_invalid_shell (shell))
return FALSE;
return uid >= MINIMUM_UID;
}
|