1 /*
2 * Copyright (C) 2011-2021 Free Software Foundation, Inc.
3 * Written by Ben Walton.
4 *
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 3 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <https://www.gnu.org/licenses/>. */
17
18 #include <config.h>
19
20 #include <unistd.h>
21
22 /* for HOST_NAME_MAX */
23 #include <limits.h>
24 /* for strlen */
25 #include <string.h>
26
27 #include <errno.h>
28 #include <stdio.h>
29
30 #include "root-uid.h"
31
32 #include "macros.h"
33
34 #define TESTHOSTNAME "gnulib-hostname"
35
36 /* mingw and MSVC 9 lack geteuid, so setup a dummy value.
37 On Cygwin, geteuid() may return non-zero even for user accounts with
38 administrator privileges, so use a dummy value as well. */
39 #if !HAVE_GETEUID || defined __CYGWIN__
40 # define geteuid() ROOT_UID
41 #endif
42
43 int
44 main (int argc, _GL_UNUSED char *argv[])
/* ![[previous]](../icons/n_left.png)
![[next]](../icons/n_right.png)
![[first]](../icons/n_first.png)
![[last]](../icons/n_last.png)
![[top]](../icons/top.png)
![[bottom]](../icons/bottom.png)
![[index]](../icons/index.png)
*/
45 {
46 char origname[HOST_NAME_MAX];
47 char newname[HOST_NAME_MAX];
48 char longname[HOST_NAME_MAX + 2];
49 int rcs, i;
50
51 /* skip the tests if we don't have root privilege. this does not
52 consider things like CAP_SYS_ADMIN (linux) or PRIV_SYS_ADMIN
53 (solaris), etc. systems without a working geteuid (mingw, MSVC
54 9) will always skip this test. */
55 if (geteuid () != ROOT_UID)
56 {
57 fprintf (stderr, "Skipping test: insufficient permissions.\n");
58 return 77;
59 }
60
61 /* we want to ensure we can do a get/set/get check to ensure the
62 change is accepted. record the current name so it can be restored
63 later */
64 ASSERT (gethostname (origname, sizeof (origname)) == 0);
65
66 /* try setting a valid hostname. if it fails -1/ENOSYS, we will
67 skip the test for long names as this is an indication we're using
68 the stub function that doesn't do anything on this platform. */
69 rcs = sethostname (TESTHOSTNAME, strlen (TESTHOSTNAME));
70
71 if (rcs != 0)
72 {
73 if (rcs == -1 && errno == ENOSYS)
74 {
75 fprintf (stderr,
76 "Skipping test: sethostname is not really implemented.\n");
77 return 77;
78 }
79 else if (rcs == -1
80 && (errno == EPERM
81 || errno == EACCES)) /* Cygwin */
82 {
83 fprintf (stderr, "Skipping test: insufficient permissions.\n");
84 return 77;
85 }
86 else
87 {
88 fprintf (stderr, "error setting valid hostname.\n");
89 return 1;
90 }
91 }
92 else
93 {
94 ASSERT (gethostname (newname, sizeof (newname)) == 0);
95
96 /* On Windows, a hostname change becomes effective only after
97 a reboot. */
98 #if !(defined _WIN32 || defined __CYGWIN__)
99
100 /* if we don't get back what we put in, there is no need to
101 restore the original name as we will assume it was not
102 properly changed. */
103 if (strcmp (newname, TESTHOSTNAME) != 0)
104 {
105 fprintf (stderr, "set/get comparison failed.\n");
106 return 1;
107 }
108 #endif
109 }
110
111 /* glibc does allow setting a zero length name, so the lower bound
112 needs no test. validate that we are constrained by
113 HOST_NAME_MAX */
114 for (i = 0; i < (HOST_NAME_MAX + 1); i++)
115 longname[i] = 'a';
116
117 longname[i] = '\0';
118
119 rcs = sethostname (longname, (HOST_NAME_MAX + 1));
120
121 if (rcs != -1)
122 {
123 /* attempt to restore the original name. */
124 ASSERT (sethostname (origname, strlen (origname)) == 0);
125 fprintf (stderr, "setting a too long hostname succeeded.\n");
126 return 1;
127 }
128
129 /* restore the original name. */
130 ASSERT (sethostname (origname, strlen (origname)) == 0);
131
132 return 0;
133 }