Blame src/common/oscap_acquire.c

Packit 517ee8
/*
Packit 517ee8
 * Copyright 2012--2014 Red Hat Inc., Durham, North Carolina.
Packit 517ee8
 * All Rights Reserved.
Packit 517ee8
 *
Packit 517ee8
 * This library is free software; you can redistribute it and/or
Packit 517ee8
 * modify it under the terms of the GNU Lesser General Public
Packit 517ee8
 * License as published by the Free Software Foundation; either
Packit 517ee8
 * version 2.1 of the License, or (at your option) any later version.
Packit 517ee8
 *
Packit 517ee8
 * This library is distributed in the hope that it will be useful,
Packit 517ee8
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 517ee8
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 517ee8
 * Lesser General Public License for more details.
Packit 517ee8
 *
Packit 517ee8
 * You should have received a copy of the GNU Lesser General Public
Packit 517ee8
 * License along with this library; if not, write to the Free Software
Packit 517ee8
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Packit 517ee8
 *
Packit 517ee8
 */
Packit 517ee8
Packit 517ee8
#ifdef HAVE_CONFIG_H
Packit 517ee8
#include <config.h>
Packit 517ee8
#endif
Packit 517ee8
Packit 517ee8
#include <string.h>
Packit 517ee8
#include <errno.h>
Packit 517ee8
#ifdef OS_WINDOWS
Packit 517ee8
#include <io.h>
Packit 517ee8
#include <direct.h>
Packit 517ee8
/* The rand_s function requires constant _CRT_RAND_S to be defined before including <stdlib.h>.
Packit 517ee8
 * See https://docs.microsoft.com/en-us/cpp/c-runtime-library/reference/rand-s
Packit 517ee8
 */
Packit 517ee8
#define _CRT_RAND_S
Packit 517ee8
#else
Packit 517ee8
#include <unistd.h>
Packit 517ee8
#include <ftw.h>
Packit 517ee8
#endif
Packit 517ee8
#include <stdlib.h>
Packit 517ee8
#include <limits.h>
Packit 517ee8
#include <sys/stat.h>
Packit 517ee8
#include <fcntl.h>
Packit 517ee8
Packit 517ee8
#include <curl/curl.h>
Packit 517ee8
#include <curl/easy.h>
Packit 517ee8
Packit 517ee8
#include "oscap_acquire.h"
Packit 517ee8
#include "common/util.h"
Packit 517ee8
#include "common/oscap_buffer.h"
Packit 517ee8
#include "common/_error.h"
Packit 517ee8
#include "oscap_string.h"
Packit 517ee8
#include "oscap_helpers.h"
Packit Service deda86
#include "debug_priv.h"
Packit 517ee8
Packit 517ee8
#ifndef OSCAP_TEMP_DIR
Packit 517ee8
#define OSCAP_TEMP_DIR "/tmp"
Packit 517ee8
#endif
Packit 517ee8
Packit 517ee8
#define TEMP_DIR_TEMPLATE OSCAP_TEMP_DIR "/oscap.XXXXXX"
Packit 517ee8
#define TEMP_URL_TEMPLATE "downloaded.XXXXXX"
Packit 517ee8
Packit 517ee8
#ifdef OS_WINDOWS
Packit 517ee8
char *oscap_acquire_temp_dir()
Packit 517ee8
{
Packit 517ee8
	WCHAR temp_path[PATH_MAX];
Packit 517ee8
	WCHAR temp_dir[PATH_MAX];
Packit 517ee8
Packit 517ee8
	DWORD ret = GetTempPathW(PATH_MAX, temp_path);
Packit 517ee8
	if (ret > PATH_MAX || ret == 0) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_WINDOWS, "Could not retrieve the path of the directory for temporary files.");
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	unsigned int unique;
Packit 517ee8
	rand_s(&unique);
Packit 517ee8
	ret = GetTempFileNameW(temp_path, L"oscap", unique, temp_dir);
Packit 517ee8
	if (ret == 0) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_WINDOWS, "Could not get a name for new temporary directory.");
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
	char *temp_dir_str = oscap_windows_wstr_to_str(temp_dir);
Packit 517ee8
Packit 517ee8
	WCHAR *path_prefix = temp_dir;
Packit 517ee8
	do {
Packit 517ee8
		/* Use wide characters with L modifier because we work with a wide string. */
Packit 517ee8
		WCHAR *delimiter = wcschr(path_prefix, L'\\');
Packit 517ee8
		if (delimiter == NULL) {
Packit 517ee8
			break;
Packit 517ee8
		}
Packit 517ee8
		*delimiter = L'\0';
Packit 517ee8
		if (!CreateDirectoryW(temp_dir, NULL)) {
Packit 517ee8
			ret = GetLastError();
Packit 517ee8
			if (ret != ERROR_ALREADY_EXISTS) {
Packit 517ee8
				char *error_message = oscap_windows_error_message(ret);
Packit 517ee8
				oscap_seterr(OSCAP_EFAMILY_WINDOWS, "Could not create temp directory '%s': %s.", temp_dir_str, error_message);
Packit 517ee8
				free(error_message);
Packit 517ee8
				free(temp_dir_str);
Packit 517ee8
				return NULL;
Packit 517ee8
			}
Packit 517ee8
		}
Packit 517ee8
		*delimiter = L'\\';
Packit 517ee8
		path_prefix = ++delimiter;
Packit 517ee8
	} while (*path_prefix != L'\0');
Packit 517ee8
Packit 517ee8
	if (!CreateDirectoryW(temp_dir, NULL)) {
Packit 517ee8
		ret = GetLastError();
Packit 517ee8
		char *error_message = oscap_windows_error_message(ret);
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_WINDOWS, "Could not create temp directory '%s': %s.", temp_dir_str, error_message);
Packit 517ee8
		free(error_message);
Packit 517ee8
		free(temp_dir_str);
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	return temp_dir_str;
Packit 517ee8
}
Packit 517ee8
#else
Packit 517ee8
char *oscap_acquire_temp_dir()
Packit 517ee8
{
Packit 517ee8
	char *temp_dir = oscap_strdup(TEMP_DIR_TEMPLATE);
Packit 517ee8
	if (mkdtemp(temp_dir) == NULL) {
Packit 517ee8
		free(temp_dir);
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_GLIBC, "Could not create temp directory " TEMP_DIR_TEMPLATE ". %s", strerror(errno));
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
	return temp_dir;
Packit 517ee8
}
Packit 517ee8
#endif
Packit 517ee8
Packit 517ee8
#ifdef OS_WINDOWS
Packit 517ee8
static bool _recursive_delete_directory(WCHAR *directory)
Packit 517ee8
{
Packit 517ee8
	if (directory == NULL) {
Packit 517ee8
		return false;
Packit 517ee8
	}
Packit 517ee8
	WCHAR find_pattern[MAX_PATH];
Packit 517ee8
	wcsncpy(find_pattern, directory, MAX_PATH);
Packit 517ee8
	wcsncat(find_pattern, L"\\*", MAX_PATH);
Packit 517ee8
	WCHAR dir_path[MAX_PATH];
Packit 517ee8
	wcsncpy(dir_path, directory, MAX_PATH);
Packit 517ee8
	wcsncat(dir_path, L"\\", MAX_PATH);
Packit 517ee8
	WCHAR file_path[MAX_PATH];
Packit 517ee8
	wcsncpy(file_path, dir_path, MAX_PATH);
Packit 517ee8
	DWORD err;
Packit 517ee8
Packit 517ee8
	WIN32_FIND_DATAW find_data;
Packit 517ee8
	HANDLE find_handle = FindFirstFileW(find_pattern, &find_data);
Packit 517ee8
	if (find_handle == INVALID_HANDLE_VALUE) {
Packit 517ee8
		err = GetLastError();
Packit 517ee8
		char *error_message = oscap_windows_error_message(err);
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_WINDOWS, "FindFirstFileW error: %s", error_message);
Packit 517ee8
		free(error_message);
Packit 517ee8
		return false;
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	do {
Packit 517ee8
		if (wcscmp(find_data.cFileName, L".") != 0 && wcscmp(find_data.cFileName, L"..") != 0) {
Packit 517ee8
			wcsncat(file_path, find_data.cFileName, MAX_PATH);
Packit 517ee8
			if (find_data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
Packit 517ee8
				if (!_recursive_delete_directory(file_path)) {
Packit 517ee8
					FindClose(find_handle);
Packit 517ee8
					return false;
Packit 517ee8
				}
Packit 517ee8
			} else {
Packit 517ee8
				if (!DeleteFileW(file_path)) {
Packit 517ee8
					err = GetLastError();
Packit 517ee8
					char *error_message = oscap_windows_error_message(err);
Packit 517ee8
					oscap_seterr(OSCAP_EFAMILY_WINDOWS, "DeleteFileW Error: %s", error_message);
Packit 517ee8
					free(error_message);
Packit 517ee8
					FindClose(find_handle);
Packit 517ee8
					return false;
Packit 517ee8
				}
Packit 517ee8
			}
Packit 517ee8
			wcsncpy(file_path, dir_path, MAX_PATH);
Packit 517ee8
		}
Packit 517ee8
	} while (FindNextFileW(find_handle, &find_data) != 0);
Packit 517ee8
	FindClose(find_handle);
Packit 517ee8
Packit 517ee8
	if (!RemoveDirectoryW(dir_path)) {
Packit 517ee8
		err = GetLastError();
Packit 517ee8
		char *error_message = oscap_windows_error_message(err);
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_WINDOWS, "RemoveDirectoryW error: %s", error_message);
Packit 517ee8
		free(error_message);
Packit 517ee8
		return false;
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	return true;
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
void oscap_acquire_cleanup_dir(char **dir_path)
Packit 517ee8
{
Packit 517ee8
	if (*dir_path != NULL) {
Packit 517ee8
		WCHAR *dir_path_wstr = oscap_windows_str_to_wstr(*dir_path);
Packit 517ee8
		_recursive_delete_directory(dir_path_wstr);
Packit 517ee8
		free(dir_path_wstr);
Packit 517ee8
		free(*dir_path);
Packit 517ee8
		*dir_path = NULL;
Packit 517ee8
	}
Packit 517ee8
}
Packit 517ee8
#else
Packit 517ee8
static int __unlink_cb(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf)
Packit 517ee8
{
Packit 517ee8
	int rv = remove(fpath);
Packit 517ee8
Packit 517ee8
	if (rv)
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_GLIBC, "Could not remove %s. %s", fpath, strerror(errno));
Packit 517ee8
Packit 517ee8
	return rv;
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
void oscap_acquire_cleanup_dir(char **dir_path)
Packit 517ee8
{
Packit 517ee8
	if (*dir_path != NULL)
Packit 517ee8
	{
Packit 517ee8
		nftw(*dir_path, __unlink_cb, 64, FTW_DEPTH | FTW_PHYS | FTW_MOUNT);
Packit 517ee8
		free(*dir_path);
Packit 517ee8
		*dir_path = NULL;
Packit 517ee8
	}
Packit 517ee8
}
Packit 517ee8
#endif
Packit 517ee8
Packit 517ee8
int
Packit 517ee8
oscap_acquire_temp_file(const char *dir, const char *template, char **filename)
Packit 517ee8
{
Packit 517ee8
#ifdef OS_WINDOWS
Packit 517ee8
	int old_mode;
Packit 517ee8
#else
Packit 517ee8
	mode_t old_mode;
Packit 517ee8
#endif
Packit 517ee8
	int fd;
Packit 517ee8
Packit 517ee8
	if (dir == NULL || template == NULL || filename == NULL)
Packit 517ee8
		return -1;
Packit 517ee8
Packit 517ee8
	*filename = malloc(PATH_MAX * sizeof(char));
Packit 517ee8
	old_mode = umask(077); /* Override unusual umask. Ensure 0700 permissions. */
Packit 517ee8
#ifdef OS_WINDOWS
Packit 517ee8
	char *base_name = oscap_strdup(template);
Packit 517ee8
	_mktemp_s(base_name, strlen(base_name) + 1); // +1 for terminator
Packit 517ee8
	snprintf(*filename, PATH_MAX, "%s/%s", dir, base_name);
Packit 517ee8
	free(base_name);
Packit 517ee8
	fd = open(*filename, _O_RDWR | _O_CREAT, _S_IREAD | _S_IWRITE);
Packit 517ee8
#else
Packit 517ee8
	snprintf(*filename, PATH_MAX, "%s/%s", dir, template);
Packit 517ee8
	fd = mkstemp(*filename);
Packit 517ee8
#endif
Packit 517ee8
	(void) umask(old_mode);
Packit 517ee8
	if (fd < 1) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_GLIBC, "mkstemp for %s failed: %s", *filename, strerror(errno));
Packit 517ee8
		free(*filename);
Packit 517ee8
		*filename = NULL;
Packit 517ee8
	}
Packit 517ee8
	return fd;
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
bool
Packit 517ee8
oscap_acquire_url_is_supported(const char *url)
Packit 517ee8
{
Packit 517ee8
	return oscap_str_startswith(url, "http://") || oscap_str_startswith(url, "https://");
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
char *
Packit 517ee8
oscap_acquire_url_to_filename(const char *url)
Packit 517ee8
{
Packit 517ee8
	/* RFC 3986: 2.1. Percent-Encoding */
Packit 517ee8
	char *curl_filename = NULL;
Packit 517ee8
	char *filename = NULL;
Packit 517ee8
	CURL *curl;
Packit 517ee8
Packit 517ee8
	if (curl_global_init(CURL_GLOBAL_ALL) != 0) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to initialize libcurl.");
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
	curl = curl_easy_init();
Packit 517ee8
	if (curl == NULL) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to initialize libcurl.");
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	curl_filename = curl_easy_escape(curl , url , 0);
Packit 517ee8
	if (curl_filename == NULL) {
Packit 517ee8
		curl_easy_cleanup(curl);
Packit 517ee8
		curl_global_cleanup();
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to escape the given url %s", url);
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
	filename = oscap_strdup(curl_filename);
Packit 517ee8
	curl_free(curl_filename);
Packit 517ee8
	curl_easy_cleanup(curl);
Packit 517ee8
	curl_global_cleanup();
Packit 517ee8
	return filename;
Packit 517ee8
}
Packit 517ee8
Packit Service deda86
static int _curl_trace(CURL *handle, curl_infotype type, char *data, size_t size, void *userp)
Packit Service deda86
{
Packit Service deda86
	const char *title;
Packit Service deda86
Packit Service deda86
	switch (type) {
Packit Service deda86
	case CURLINFO_TEXT:
Packit Service deda86
		title = "== cURL info";
Packit Service deda86
		break;
Packit Service deda86
	case CURLINFO_HEADER_OUT:
Packit Service deda86
		title = "=> cURL header (out)";
Packit Service deda86
		break;
Packit Service deda86
	case CURLINFO_HEADER_IN:
Packit Service deda86
		title = "<= cURL header (in)";
Packit Service deda86
		break;
Packit Service deda86
	case CURLINFO_DATA_OUT:
Packit Service deda86
	case CURLINFO_SSL_DATA_OUT:
Packit Service deda86
	case CURLINFO_DATA_IN:
Packit Service deda86
	case CURLINFO_SSL_DATA_IN:
Packit Service deda86
	default:
Packit Service deda86
		return 0;
Packit Service deda86
		break;
Packit Service deda86
	}
Packit Service deda86
Packit Service deda86
	dD("%s: %s", title, data);
Packit Service deda86
Packit Service deda86
	return 0;
Packit Service deda86
}
Packit Service deda86
Packit 517ee8
char* oscap_acquire_url_download(const char *url, size_t* memory_size)
Packit 517ee8
{
Packit 517ee8
	CURL *curl;
Packit 517ee8
	curl = curl_easy_init();
Packit 517ee8
	if (curl == NULL) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to initialize libcurl.");
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
Packit Service 12d954
	CURLcode res;
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_URL, url);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_URL to '%s': %s", url, curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_to_memory_callback);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_WRITEFUNCTION to write_to_memory_callback: %s", curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_ACCEPT_ENCODING, "");
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_ACCEPT_ENCODING to an empty string: %s", curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_TRANSFER_ENCODING, true);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_TRANSFER_ENCODING to true: %s", curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, true);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_FOLLOWLOCATION to true: %s", curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_VERBOSE, true);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_VERBOSE to true: %s", curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, _curl_trace);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_DEBUGFUNCTION to _curl_trace: %s", curl_easy_strerror(res));
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	struct oscap_buffer *buffer = oscap_buffer_new();
Packit Service 12d954
	res = curl_easy_setopt(curl, CURLOPT_WRITEDATA, buffer);
Packit Service 12d954
	if (res != 0) {
Packit Service 12d954
		curl_easy_cleanup(curl);
Packit Service 12d954
		oscap_seterr(OSCAP_EFAMILY_NET, "Failed to set CURLOPT_WRITEDATA as buffer: %s", curl_easy_strerror(res));
Packit Service 12d954
		oscap_buffer_free(buffer);
Packit Service 12d954
		return NULL;
Packit Service 12d954
	}
Packit Service 12d954
Packit Service 12d954
	res = curl_easy_perform(curl);
Packit 517ee8
	curl_easy_cleanup(curl);
Packit 517ee8
Packit 517ee8
	if (res != 0) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_NET, "Download failed: %s", curl_easy_strerror(res));
Packit 517ee8
		oscap_buffer_free(buffer);
Packit 517ee8
		return NULL;
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	*memory_size = oscap_buffer_get_length(buffer);
Packit 517ee8
	char* data = oscap_buffer_bequeath(buffer); // get data and free buffer struct
Packit 517ee8
	return data;
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
size_t write_to_memory_callback(char *ptr, size_t size, size_t nmemb, void *userdata)
Packit 517ee8
{
Packit 517ee8
	size_t new_received_size = size * nmemb; // total size of newly received data
Packit 517ee8
	oscap_buffer_append_binary_data((struct oscap_buffer*)userdata, ptr, new_received_size);
Packit 517ee8
	return new_received_size;
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
char *
Packit 517ee8
oscap_acquire_pipe_to_string(int fd)
Packit 517ee8
{
Packit 517ee8
	struct oscap_string *pipe_string = oscap_string_new();
Packit 517ee8
Packit 517ee8
	char readbuf;
Packit 517ee8
	// FIXME: Read by larger chunks in the future
Packit 517ee8
	while (read(fd, &readbuf, 1) > 0) {
Packit 517ee8
Packit 517ee8
		if (readbuf == '&') {
Packit 517ee8
			// & is a special case, we have to "escape" it manually
Packit 517ee8
			// (all else will eventually get handled by libxml)
Packit 517ee8
			oscap_string_append_string(pipe_string, "&");
Packit 517ee8
		} else {
Packit 517ee8
			oscap_string_append_char(pipe_string, readbuf);
Packit 517ee8
		}
Packit 517ee8
	}
Packit 517ee8
Packit 517ee8
	close(fd);
Packit 517ee8
	return oscap_string_bequeath(pipe_string);
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
char *oscap_acquire_guess_realpath(const char *filepath)
Packit 517ee8
{
Packit 517ee8
	char resolved_name[PATH_MAX];
Packit 517ee8
Packit 517ee8
	char *rpath = oscap_realpath(filepath, resolved_name);
Packit 517ee8
	if (rpath != NULL)
Packit 517ee8
		rpath = oscap_strdup(rpath);
Packit 517ee8
	else {
Packit 517ee8
		// file does not exists, let's try to guess realpath
Packit 517ee8
		// this is not 100% correct, but it is good enough
Packit 517ee8
		char *copy = oscap_strdup(filepath);
Packit 517ee8
		if (copy == NULL) {
Packit 517ee8
			oscap_seterr(OSCAP_EFAMILY_OSCAP, "Cannot guess realpath for %s, directory: cannot allocate memory!", filepath);
Packit 517ee8
			return NULL;
Packit 517ee8
		}
Packit 517ee8
Packit 517ee8
		char *dir_name = oscap_dirname(copy);
Packit 517ee8
		char *real_dir = oscap_realpath(dir_name, resolved_name);
Packit 517ee8
		if (real_dir == NULL) {
Packit 517ee8
			oscap_seterr(OSCAP_EFAMILY_OSCAP, "Cannot guess realpath for %s, directory: %s does not exists!", filepath, dir_name);
Packit 517ee8
			free(copy);
Packit 517ee8
			free(dir_name);
Packit 517ee8
			return NULL;
Packit 517ee8
		}
Packit 517ee8
		free(dir_name);
Packit 517ee8
		char *base_name = oscap_basename((char *)filepath);
Packit 517ee8
		rpath = oscap_sprintf("%s/%s", real_dir, base_name);
Packit 517ee8
		free(base_name);
Packit 517ee8
		free(copy);
Packit 517ee8
	}
Packit 517ee8
	return rpath;
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
int oscap_acquire_mkdir_p(const char *path)
Packit 517ee8
{
Packit 517ee8
	// NOTE: This assumes a UNIX VFS path, C:\\folder\\folder would break it!
Packit 517ee8
Packit 517ee8
	if (strlen(path) > PATH_MAX) {
Packit 517ee8
		return -1;
Packit 517ee8
	}
Packit 517ee8
	else {
Packit 517ee8
		char temp[PATH_MAX + 1]; // +1 for \0
Packit 517ee8
		unsigned int i;
Packit 517ee8
Packit 517ee8
		for (i = 0; i <= strlen(path); i++) {
Packit 517ee8
			if (path[i] == '/' || path[i] == '\0') {
Packit 517ee8
				strncpy(temp, path, i);
Packit 517ee8
				temp[i] = '\0';
Packit 517ee8
Packit 517ee8
				// skip leading '/', we will never be creating the root anyway
Packit 517ee8
				if (strlen(temp) == 0)
Packit 517ee8
					continue;
Packit 517ee8
Packit 517ee8
#ifndef OS_WINDOWS
Packit 517ee8
				if (mkdir(temp, S_IRWXU) != 0 && errno != EEXIST) {
Packit 517ee8
#else
Packit 517ee8
				if (mkdir(temp) != 0 && errno != EEXIST) {
Packit 517ee8
#endif
Packit 517ee8
					oscap_seterr(OSCAP_EFAMILY_GLIBC,
Packit 517ee8
						"Error making directory '%s', while doing recursive mkdir for '%s', error was '%s'.",
Packit 517ee8
						temp, path, strerror(errno));
Packit 517ee8
					return -1;
Packit 517ee8
				}
Packit 517ee8
			}
Packit 517ee8
		}
Packit 517ee8
Packit 517ee8
		return 0;
Packit 517ee8
	}
Packit 517ee8
}
Packit 517ee8
Packit 517ee8
int oscap_acquire_ensure_parent_dir(const char *filepath)
Packit 517ee8
{
Packit 517ee8
	char *filepath_cpy = oscap_strdup(filepath);
Packit 517ee8
	char *dirpath = oscap_dirname(filepath_cpy);
Packit 517ee8
	int ret = oscap_acquire_mkdir_p(dirpath);
Packit 517ee8
	if (ret != 0) {
Packit 517ee8
		oscap_seterr(OSCAP_EFAMILY_GLIBC, "Error making directory '%s' to ensure correct path of '%s'.", dirpath, filepath);
Packit 517ee8
	}
Packit 517ee8
	free(dirpath);
Packit 517ee8
	free(filepath_cpy);
Packit 517ee8
	return ret;
Packit 517ee8
}