2012-09-20 07:51:34 +04:00
|
|
|
/**
|
|
|
|
* xrdp: A Remote Desktop Protocol server.
|
|
|
|
*
|
2013-06-08 21:51:53 +04:00
|
|
|
* Copyright (C) Jay Sorg 2004-2013
|
2012-09-20 07:51:34 +04:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2005-10-06 23:27:38 +04:00
|
|
|
|
2005-11-14 01:06:49 +03:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @file env.c
|
|
|
|
* @brief User environment handling code
|
|
|
|
* @author Jay Sorg
|
2012-09-20 07:51:34 +04:00
|
|
|
*
|
2005-11-14 01:06:49 +03:00
|
|
|
*/
|
|
|
|
|
2017-03-03 07:33:23 +03:00
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
|
|
#include <config_ac.h>
|
|
|
|
#endif
|
|
|
|
|
2017-01-05 00:13:20 +03:00
|
|
|
#include <grp.h>
|
|
|
|
|
2022-09-08 17:05:51 +03:00
|
|
|
#include "env.h"
|
|
|
|
#include "config.h"
|
2014-05-10 03:02:19 +04:00
|
|
|
#include "list.h"
|
2022-09-08 17:05:51 +03:00
|
|
|
#include "log.h"
|
|
|
|
#include "os_calls.h"
|
2005-10-06 23:27:38 +04:00
|
|
|
#include "sesman.h"
|
2015-08-19 03:33:13 +03:00
|
|
|
#include "ssl_calls.h"
|
2020-12-21 15:36:00 +03:00
|
|
|
#include "string_calls.h"
|
2022-09-08 17:05:51 +03:00
|
|
|
#include "xrdp_sockets.h"
|
2006-07-01 18:25:12 +04:00
|
|
|
|
2005-10-06 23:27:38 +04:00
|
|
|
/******************************************************************************/
|
2017-03-12 19:35:00 +03:00
|
|
|
int
|
2016-10-17 09:14:06 +03:00
|
|
|
env_check_password_file(const char *filename, const char *passwd)
|
2005-10-06 23:27:38 +04:00
|
|
|
{
|
2012-09-20 07:51:34 +04:00
|
|
|
char encryptedPasswd[16];
|
2015-08-19 03:33:13 +03:00
|
|
|
char key[24];
|
2015-10-11 01:44:31 +03:00
|
|
|
char passwd_hash[20];
|
|
|
|
char passwd_hash_text[40];
|
2012-09-20 07:51:34 +04:00
|
|
|
int fd;
|
2015-10-11 01:44:31 +03:00
|
|
|
int passwd_bytes;
|
|
|
|
void *des;
|
|
|
|
void *sha1;
|
2012-09-20 07:51:34 +04:00
|
|
|
|
2016-02-14 07:41:07 +03:00
|
|
|
/* create password hash from password */
|
2015-10-11 01:44:31 +03:00
|
|
|
passwd_bytes = g_strlen(passwd);
|
|
|
|
sha1 = ssl_sha1_info_create();
|
2022-01-20 19:43:00 +03:00
|
|
|
ssl_sha1_clear(sha1);
|
2015-10-11 01:44:31 +03:00
|
|
|
ssl_sha1_transform(sha1, "xrdp_vnc", 8);
|
|
|
|
ssl_sha1_transform(sha1, passwd, passwd_bytes);
|
|
|
|
ssl_sha1_transform(sha1, passwd, passwd_bytes);
|
|
|
|
ssl_sha1_complete(sha1, passwd_hash);
|
|
|
|
ssl_sha1_info_delete(sha1);
|
|
|
|
g_snprintf(passwd_hash_text, 39, "%2.2x%2.2x%2.2x%2.2x",
|
|
|
|
(tui8)passwd_hash[0], (tui8)passwd_hash[1],
|
|
|
|
(tui8)passwd_hash[2], (tui8)passwd_hash[3]);
|
|
|
|
passwd_hash_text[39] = 0;
|
|
|
|
passwd = passwd_hash_text;
|
|
|
|
|
|
|
|
/* create file from password */
|
2015-08-19 03:33:13 +03:00
|
|
|
g_memset(encryptedPasswd, 0, sizeof(encryptedPasswd));
|
2015-10-11 01:44:31 +03:00
|
|
|
g_strncpy(encryptedPasswd, passwd, 8);
|
2015-08-19 03:33:13 +03:00
|
|
|
g_memset(key, 0, sizeof(key));
|
|
|
|
g_mirror_memcpy(key, g_fixedkey, 8);
|
2017-07-13 19:01:07 +03:00
|
|
|
des = ssl_des3_encrypt_info_create(key, 0);
|
2015-08-19 03:33:13 +03:00
|
|
|
ssl_des3_encrypt(des, 8, encryptedPasswd, encryptedPasswd);
|
|
|
|
ssl_des3_info_delete(des);
|
2015-10-11 01:44:31 +03:00
|
|
|
fd = g_file_open_ex(filename, 0, 1, 1, 1);
|
2012-09-20 07:51:34 +04:00
|
|
|
if (fd == -1)
|
|
|
|
{
|
2020-11-30 03:36:20 +03:00
|
|
|
LOG(LOG_LEVEL_WARNING,
|
|
|
|
"Cannot write VNC password hash to file %s: %s",
|
|
|
|
filename, g_get_strerror());
|
2012-09-20 07:51:34 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
g_file_write(fd, encryptedPasswd, 8);
|
|
|
|
g_file_close(fd);
|
|
|
|
return 0;
|
2005-10-06 23:27:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2016-07-22 11:48:37 +03:00
|
|
|
/* its the responsibility of the caller to free passwd_file */
|
2017-03-12 19:35:00 +03:00
|
|
|
int
|
2022-12-14 12:36:07 +03:00
|
|
|
env_set_user(int uid, char **passwd_file, int display,
|
2016-10-17 09:14:06 +03:00
|
|
|
const struct list *env_names, const struct list *env_values)
|
2005-10-06 23:27:38 +04:00
|
|
|
{
|
2012-09-20 07:51:34 +04:00
|
|
|
int error;
|
|
|
|
int pw_gid;
|
2014-05-10 03:02:19 +04:00
|
|
|
int index;
|
2016-07-22 11:48:37 +03:00
|
|
|
int len;
|
2014-05-10 03:02:19 +04:00
|
|
|
char *name;
|
|
|
|
char *value;
|
2022-12-14 12:36:07 +03:00
|
|
|
char *pw_username;
|
2016-07-22 11:48:37 +03:00
|
|
|
char *pw_shell;
|
|
|
|
char *pw_dir;
|
2012-09-20 07:51:34 +04:00
|
|
|
char text[256];
|
2018-01-23 02:42:29 +03:00
|
|
|
char hostname[256];
|
2012-09-20 07:51:34 +04:00
|
|
|
|
2022-12-14 12:36:07 +03:00
|
|
|
pw_username = 0;
|
2016-07-22 11:48:37 +03:00
|
|
|
pw_shell = 0;
|
|
|
|
pw_dir = 0;
|
|
|
|
|
2022-12-14 12:36:07 +03:00
|
|
|
error = g_getuser_info_by_uid(uid, &pw_username, &pw_gid, &pw_shell,
|
|
|
|
&pw_dir, 0);
|
2012-09-20 07:51:34 +04:00
|
|
|
|
2005-10-06 23:27:38 +04:00
|
|
|
if (error == 0)
|
|
|
|
{
|
2012-09-20 07:51:34 +04:00
|
|
|
g_rm_temp_dir();
|
2022-12-15 18:08:11 +03:00
|
|
|
g_clearenv();
|
|
|
|
#ifdef HAVE_SETUSERCONTEXT
|
|
|
|
error = g_set_allusercontext(uid);
|
|
|
|
#else
|
|
|
|
/* Set some of the things setusercontext() handles on other
|
|
|
|
* systems */
|
|
|
|
|
|
|
|
/* Primary group. Note that secondary groups should already
|
|
|
|
* have been set, if we're not using setusercontext() */
|
2012-09-20 07:51:34 +04:00
|
|
|
error = g_setgid(pw_gid);
|
|
|
|
|
|
|
|
if (error == 0)
|
2010-08-10 03:06:51 +04:00
|
|
|
{
|
2012-09-20 07:51:34 +04:00
|
|
|
error = g_setuid(uid);
|
2007-05-07 01:02:25 +04:00
|
|
|
}
|
2012-09-20 07:51:34 +04:00
|
|
|
|
2022-12-15 18:08:11 +03:00
|
|
|
if (error == 0)
|
|
|
|
{
|
|
|
|
g_setenv("PATH", "/sbin:/bin:/usr/bin:/usr/local/bin", 1);
|
|
|
|
}
|
|
|
|
#endif
|
2017-07-13 19:01:07 +03:00
|
|
|
g_mk_socket_path(0);
|
2012-09-20 07:51:34 +04:00
|
|
|
|
|
|
|
if (error == 0)
|
|
|
|
{
|
|
|
|
g_setenv("SHELL", pw_shell, 1);
|
2022-12-14 12:36:07 +03:00
|
|
|
g_setenv("USER", pw_username, 1);
|
|
|
|
g_setenv("LOGNAME", pw_username, 1);
|
2012-09-20 07:51:34 +04:00
|
|
|
g_sprintf(text, "%d", uid);
|
|
|
|
g_setenv("UID", text, 1);
|
|
|
|
g_setenv("HOME", pw_dir, 1);
|
|
|
|
g_set_current_dir(pw_dir);
|
|
|
|
g_sprintf(text, ":%d.0", display);
|
|
|
|
g_setenv("DISPLAY", text, 1);
|
2013-07-31 02:03:34 +04:00
|
|
|
g_setenv("XRDP_SESSION", "1", 1);
|
2021-04-19 18:32:04 +03:00
|
|
|
/* XRDP_SOCKET_PATH should be set even here. It's used by
|
|
|
|
* xorgxrdp and the pulseaudio plugin */
|
2017-05-09 08:22:37 +03:00
|
|
|
g_setenv("XRDP_SOCKET_PATH", XRDP_SOCKET_PATH, 1);
|
2018-08-27 04:47:05 +03:00
|
|
|
/* pulse sink socket */
|
|
|
|
g_snprintf(text, sizeof(text) - 1, CHANSRV_PORT_OUT_BASE_STR, display);
|
2018-08-27 09:19:14 +03:00
|
|
|
g_setenv("XRDP_PULSE_SINK_SOCKET", text, 1);
|
2018-08-27 04:47:05 +03:00
|
|
|
/* pulse source socket */
|
|
|
|
g_snprintf(text, sizeof(text) - 1, CHANSRV_PORT_IN_BASE_STR, display);
|
2018-08-27 09:19:14 +03:00
|
|
|
g_setenv("XRDP_PULSE_SOURCE_SOCKET", text, 1);
|
2014-05-10 03:02:19 +04:00
|
|
|
if ((env_names != 0) && (env_values != 0) &&
|
2020-11-30 03:36:20 +03:00
|
|
|
(env_names->count == env_values->count))
|
2014-05-10 03:02:19 +04:00
|
|
|
{
|
|
|
|
for (index = 0; index < env_names->count; index++)
|
|
|
|
{
|
|
|
|
name = (char *) list_get_item(env_names, index),
|
|
|
|
value = (char *) list_get_item(env_values, index),
|
|
|
|
g_setenv(name, value, 1);
|
|
|
|
}
|
|
|
|
}
|
2018-01-23 02:42:29 +03:00
|
|
|
g_gethostname(hostname, 255);
|
|
|
|
hostname[255] = 0;
|
2012-09-20 07:51:34 +04:00
|
|
|
if (passwd_file != 0)
|
|
|
|
{
|
|
|
|
if (0 == g_cfg->auth_file_path)
|
|
|
|
{
|
|
|
|
/* if no auth_file_path is set, then we go for
|
2018-03-23 09:58:35 +03:00
|
|
|
$HOME/.vnc/sesman_passwd-USERNAME@HOSTNAME:DISPLAY */
|
2016-11-07 09:06:24 +03:00
|
|
|
if (!g_directory_exist(".vnc"))
|
2014-07-21 00:14:09 +04:00
|
|
|
{
|
2016-11-07 09:06:24 +03:00
|
|
|
if (g_mkdir(".vnc") < 0)
|
|
|
|
{
|
2020-11-30 03:36:20 +03:00
|
|
|
LOG(LOG_LEVEL_ERROR,
|
|
|
|
"Error creating .vnc directory: %s",
|
|
|
|
g_get_strerror());
|
2016-11-07 09:06:24 +03:00
|
|
|
}
|
2016-07-22 11:48:37 +03:00
|
|
|
}
|
|
|
|
|
2018-03-23 09:58:35 +03:00
|
|
|
len = g_snprintf(NULL, 0, "%s/.vnc/sesman_passwd-%s@%s:%d",
|
2022-12-14 12:36:07 +03:00
|
|
|
pw_dir, pw_username, hostname, display);
|
2016-07-22 11:48:37 +03:00
|
|
|
|
|
|
|
*passwd_file = (char *) g_malloc(len + 1, 1);
|
|
|
|
if (*passwd_file != NULL)
|
|
|
|
{
|
2018-01-23 02:42:29 +03:00
|
|
|
/* Try legacy names first, remove if found */
|
|
|
|
g_sprintf(*passwd_file, "%s/.vnc/sesman_%s_passwd:%d",
|
2022-12-14 12:36:07 +03:00
|
|
|
pw_dir, pw_username, display);
|
2018-01-23 02:42:29 +03:00
|
|
|
if (g_file_exist(*passwd_file))
|
|
|
|
{
|
2020-11-30 03:36:20 +03:00
|
|
|
LOG(LOG_LEVEL_WARNING, "Removing old "
|
|
|
|
"password file %s", *passwd_file);
|
2018-01-23 02:42:29 +03:00
|
|
|
g_file_delete(*passwd_file);
|
|
|
|
}
|
2016-12-17 05:23:53 +03:00
|
|
|
g_sprintf(*passwd_file, "%s/.vnc/sesman_%s_passwd",
|
2022-12-14 12:36:07 +03:00
|
|
|
pw_dir, pw_username);
|
2016-12-17 05:23:53 +03:00
|
|
|
if (g_file_exist(*passwd_file))
|
|
|
|
{
|
2020-11-30 03:36:20 +03:00
|
|
|
LOG(LOG_LEVEL_WARNING, "Removing insecure "
|
|
|
|
"password file %s", *passwd_file);
|
2016-12-17 05:23:53 +03:00
|
|
|
g_file_delete(*passwd_file);
|
|
|
|
}
|
2018-03-23 09:58:35 +03:00
|
|
|
g_sprintf(*passwd_file, "%s/.vnc/sesman_passwd-%s@%s:%d",
|
2022-12-14 12:36:07 +03:00
|
|
|
pw_dir, pw_username, hostname, display);
|
2014-07-21 00:14:09 +04:00
|
|
|
}
|
2012-09-20 07:51:34 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* we use auth_file_path as requested */
|
2022-12-14 12:36:07 +03:00
|
|
|
len = g_snprintf(NULL, 0, g_cfg->auth_file_path, pw_username);
|
2016-07-22 11:48:37 +03:00
|
|
|
|
|
|
|
*passwd_file = (char *) g_malloc(len + 1, 1);
|
|
|
|
if (*passwd_file != NULL)
|
|
|
|
{
|
2022-12-14 12:36:07 +03:00
|
|
|
g_sprintf(*passwd_file, g_cfg->auth_file_path, pw_username);
|
2016-07-22 11:48:37 +03:00
|
|
|
}
|
2012-09-20 07:51:34 +04:00
|
|
|
}
|
|
|
|
|
2016-07-22 11:48:37 +03:00
|
|
|
if (*passwd_file != NULL)
|
|
|
|
{
|
2020-10-08 04:55:08 +03:00
|
|
|
LOG_DEVEL(LOG_LEVEL_DEBUG, "pass file: %s", *passwd_file);
|
2016-07-22 11:48:37 +03:00
|
|
|
}
|
2012-09-20 07:51:34 +04:00
|
|
|
}
|
2016-07-22 11:48:37 +03:00
|
|
|
|
2022-12-14 12:36:07 +03:00
|
|
|
g_free(pw_username);
|
2016-07-22 11:48:37 +03:00
|
|
|
g_free(pw_dir);
|
|
|
|
g_free(pw_shell);
|
2012-09-20 07:51:34 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-11-30 03:36:20 +03:00
|
|
|
LOG(LOG_LEVEL_ERROR,
|
2022-12-14 12:36:07 +03:00
|
|
|
"error getting user info for uid %d", uid);
|
2005-10-06 23:27:38 +04:00
|
|
|
}
|
2012-09-20 07:51:34 +04:00
|
|
|
|
|
|
|
return error;
|
2005-10-06 23:27:38 +04:00
|
|
|
}
|