/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you 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. */ #include "config.h" #include "file.h" #include "file-download.h" #include "file-ls.h" #include "file-upload.h" #include #include #include #include #include #include #include #include /** * Translates an absolute path for a shared folder to an absolute path which is * within the real "shared folder" path specified in the connection settings. * No checking is performed on the path provided, which is assumed to have * already been normalized and validated as absolute. * * @param folder * The folder containing the file whose path is being translated. * * @param virtual_path * The absolute path to the file on the simulated folder, relative to the * shared folder root. * * @param real_path * The buffer in which to store the absolute path to the real file on the * local filesystem. */ static void __guac_spice_folder_translate_path(guac_spice_folder* folder, const char* virtual_path, char* real_path) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: virtual_path=\"%s\", drive_path=\"%s\"", __func__, virtual_path, folder->path); /* Get drive path */ char* path = folder->path; int i; /* Start with path from settings */ for (i=0; iclient, GUAC_LOG_DEBUG, "%s: virtual_path=\"%s\", real_path=\"%s\"", __func__, virtual_path, real_path); } guac_spice_folder* guac_spice_folder_alloc(guac_client* client, const char* folder_path, int create_folder, int disable_download, int disable_upload) { guac_client_log(client, GUAC_LOG_DEBUG, "Initializing shared folder at " "\"%s\".", folder_path); /* Create folder if it does not exist */ if (create_folder) { guac_client_log(client, GUAC_LOG_DEBUG, "%s: Creating folder \"%s\" if necessary.", __func__, folder_path); /* Log error if directory creation fails */ if (mkdir(folder_path, S_IRWXU) && errno != EEXIST) { guac_client_log(client, GUAC_LOG_ERROR, "Unable to create folder \"%s\": %s", folder_path, strerror(errno)); } } guac_spice_folder* folder = malloc(sizeof(guac_spice_folder)); folder->client = client; folder->path = strdup(folder_path); folder->file_id_pool = guac_pool_alloc(0); folder->open_files = 0; folder->disable_download = disable_download; folder->disable_upload = disable_upload; /* Set up Download directory and watch it. */ if (!disable_download) { guac_client_log(client, GUAC_LOG_DEBUG, "%s: Setting up Download/ folder watch.", __func__); if (create_folder) { guac_client_log(client, GUAC_LOG_DEBUG, "%s: Creating Download/ folder.", __func__); char *download_path; download_path = guac_strdup(folder_path); guac_strlcat(download_path, "/Download", GUAC_SPICE_FOLDER_MAX_PATH); if (mkdir(folder_path, S_IRWXU) && errno != EEXIST) { guac_client_log(client, GUAC_LOG_ERROR, "%s: Unable to create folder \"%s\": %s", __func__, download_path, strerror(errno)); } } if(pthread_create(&(folder->download_thread), NULL, guac_spice_file_download_monitor, (void*) folder)) { guac_client_log(client, GUAC_LOG_ERROR, "%s: Unable to create Download folder thread monitor.", __func__); } } return folder; } void guac_spice_folder_free(guac_spice_folder* folder) { guac_pool_free(folder->file_id_pool); free(folder->path); free(folder); } guac_object* guac_spice_folder_alloc_object(guac_spice_folder *folder, guac_user* user) { /* Init folder */ guac_object* folder_object = guac_user_alloc_object(user); folder_object->get_handler = guac_spice_file_download_get_handler; /* Assign upload handler only if uploads are not disabled. */ if (!folder->disable_upload) folder_object->put_handler = guac_spice_file_upload_put_handler; folder_object->data = folder; /* Send filesystem to user */ guac_protocol_send_filesystem(user->socket, folder_object, "Shared Folder"); guac_socket_flush(user->socket); return folder_object; } int guac_spice_folder_append_filename(char* fullpath, const char* path, const char* filename) { int i; /* Disallow "." as a filename */ if (strcmp(filename, ".") == 0) return 0; /* Disallow ".." as a filename */ if (strcmp(filename, "..") == 0) return 0; /* Copy path, append trailing slash */ for (i=0; i 0 && path[i-1] != '/' && path[i-1] != '\\') fullpath[i++] = '/'; break; } /* Copy character if not end of string */ fullpath[i] = c; } /* Append filename */ for (; iclient, GUAC_LOG_DEBUG, "%s: Ignoring close for bad file_id: %i", __func__, file_id); return; } file = &(folder->files[file_id]); guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Closed \"%s\" (file_id=%i)", __func__, file->absolute_path, file_id); /* Close directory, if open */ if (file->dir != NULL) closedir(file->dir); /* Close file */ close(file->fd); /* Free paths */ free(file->absolute_path); free(file->real_path); /* Free ID back to pool */ guac_pool_free_int(folder->file_id_pool, file_id); folder->open_files--; } int guac_spice_folder_delete(guac_spice_folder* folder, int file_id) { /* Get file */ guac_spice_folder_file* file = guac_spice_folder_get_file(folder, file_id); if (file == NULL) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Delete of bad file_id: %i", __func__, file_id); return GUAC_SPICE_FOLDER_EINVAL; } /* If directory, attempt removal */ if (S_ISDIR(file->stmode)) { if (rmdir(file->real_path)) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: rmdir() failed: \"%s\"", __func__, file->real_path); return guac_spice_folder_get_errorcode(errno); } } /* Otherwise, attempt deletion */ else if (unlink(file->real_path)) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: unlink() failed: \"%s\"", __func__, file->real_path); return guac_spice_folder_get_errorcode(errno); } return 0; } void* guac_spice_folder_expose(guac_user* user, void* data) { guac_spice_folder* folder = (guac_spice_folder*) data; guac_user_log(user, GUAC_LOG_DEBUG, "%s: Exposing folder \"%s\" to user.", __func__, folder->path); /* No need to expose if there is no folder or the user has left */ if (user == NULL || folder == NULL) return NULL; /* Allocate and expose folder object for user */ return guac_spice_folder_alloc_object(folder, user); } int guac_spice_folder_get_errorcode(int err) { /* Translate errno codes to GUAC_SPICE_FOLDER codes */ switch(err) { case ENFILE: return GUAC_SPICE_FOLDER_ENFILE; case ENOENT: return GUAC_SPICE_FOLDER_ENOENT; case ENOTDIR: return GUAC_SPICE_FOLDER_ENOTDIR; case ENOSPC: return GUAC_SPICE_FOLDER_ENOSPC; case EISDIR: return GUAC_SPICE_FOLDER_EISDIR; case EACCES: return GUAC_SPICE_FOLDER_EACCES; case EEXIST: return GUAC_SPICE_FOLDER_EEXIST; case EINVAL: return GUAC_SPICE_FOLDER_EINVAL; case ENOSYS: return GUAC_SPICE_FOLDER_ENOSYS; case ENOTSUP: return GUAC_SPICE_FOLDER_ENOTSUP; default: return GUAC_SPICE_FOLDER_EINVAL; } } guac_spice_folder_file* guac_spice_folder_get_file(guac_spice_folder* folder, int file_id) { /* Validate ID */ if (file_id < 0 || file_id >= GUAC_SPICE_FOLDER_MAX_FILES) return NULL; /* Return file at given ID */ return &(folder->files[file_id]); } int guac_spice_folder_normalize_path(const char* path, char* abs_path) { int path_depth = 0; const char* path_components[GUAC_SPICE_FOLDER_MAX_PATH_DEPTH]; /* If original path is not absolute, normalization fails */ if (path[0] != '/') return 1; /* Create scratch copy of path excluding leading slash (we will be * replacing path separators with null terminators and referencing those * substrings directly as path components) */ char path_scratch[GUAC_SPICE_FOLDER_MAX_PATH - 1]; int length = guac_strlcpy(path_scratch, path + 1, sizeof(path_scratch)); /* Fail if provided path is too long */ if (length >= sizeof(path_scratch)) return 1; /* Locate all path components within path */ const char* current_path_component = &(path_scratch[0]); for (int i = 0; i <= length; i++) { /* If current character is a path separator, parse as component */ char c = path_scratch[i]; if (c == '/' || c == '\0') { /* Terminate current component */ path_scratch[i] = '\0'; /* If component refers to parent, just move up in depth */ if (strcmp(current_path_component, "..") == 0) { if (path_depth > 0) path_depth--; } /* Otherwise, if component not current directory, add to list */ else if (strcmp(current_path_component, ".") != 0 && strcmp(current_path_component, "") != 0) { /* Fail normalization if path is too deep */ if (path_depth >= GUAC_SPICE_FOLDER_MAX_PATH_DEPTH) return 1; path_components[path_depth++] = current_path_component; } /* Update start of next component */ current_path_component = &(path_scratch[i+1]); } /* end if separator */ /* We do not currently support named streams */ else if (c == ':') return 1; } /* end for each character */ /* Add leading slash for resulting absolute path */ abs_path[0] = '/'; /* Append normalized components to path, separated by slashes */ guac_strljoin(abs_path + 1, path_components, path_depth, "/", GUAC_SPICE_FOLDER_MAX_PATH - 1); return 0; } int guac_spice_folder_open(guac_spice_folder* folder, const char* path, int flags, bool overwrite, bool directory) { char real_path[GUAC_SPICE_FOLDER_MAX_PATH]; char normalized_path[GUAC_SPICE_FOLDER_MAX_PATH]; struct stat file_stat; int fd; int file_id; guac_spice_folder_file* file; guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: path=\"%s\", flags=0x%x, overwrite=0x%x, " "directory=0x%x", __func__, path, flags, overwrite, directory); /* If no files available, return too many open */ if (folder->open_files >= GUAC_SPICE_FOLDER_MAX_FILES) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Too many open files.", __func__, path); return GUAC_SPICE_FOLDER_ENFILE; } /* If path empty, return an error */ if (path[0] == '\0') return GUAC_SPICE_FOLDER_EINVAL; /* If path is relative, the file does not exist */ else if (path[0] != '\\' && path[0] != '/') { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Access denied - supplied path \"%s\" is relative.", __func__, path); return GUAC_SPICE_FOLDER_ENOENT; } /* Translate access into flags */ if (directory) flags |= O_DIRECTORY; else if (overwrite) flags |= O_TRUNC; /* Normalize path, return no-such-file if invalid */ if (guac_spice_folder_normalize_path(path, normalized_path)) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Normalization of path \"%s\" failed.", __func__, path); return GUAC_SPICE_FOLDER_ENOENT; } guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Normalized path \"%s\" to \"%s\".", __func__, path, normalized_path); /* Translate normalized path to real path */ __guac_spice_folder_translate_path(folder, normalized_path, real_path); guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Translated path \"%s\" to \"%s\".", __func__, normalized_path, real_path); /* Create directory first, if necessary */ if (directory && (flags & O_CREAT)) { /* Create directory */ if (mkdir(real_path, S_IRWXU)) { if (errno != EEXIST || (flags & O_EXCL)) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: mkdir() failed: %s", __func__, strerror(errno)); return guac_spice_folder_get_errorcode(errno); } } /* Unset O_CREAT and O_EXCL as directory must exist before open() */ flags &= ~(O_CREAT | O_EXCL); } guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: native open: real_path=\"%s\", flags=0x%x", __func__, real_path, flags); /* Open file */ fd = open(real_path, flags, S_IRUSR | S_IWUSR); /* If file open failed as we're trying to write a dir, retry as read-only */ if (fd == -1 && errno == EISDIR) { flags &= ~(O_WRONLY | O_RDWR); flags |= O_RDONLY; fd = open(real_path, flags, S_IRUSR | S_IWUSR); } if (fd == -1) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: open() failed: %s", __func__, strerror(errno)); return guac_spice_folder_get_errorcode(errno); } /* Get file ID, init file */ file_id = guac_pool_next_int(folder->file_id_pool); file = &(folder->files[file_id]); file->id = file_id; file->fd = fd; file->dir = NULL; file->dir_pattern[0] = '\0'; file->absolute_path = strdup(normalized_path); file->real_path = strdup(real_path); file->bytes_written = 0; guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Opened \"%s\" as file_id=%i", __func__, normalized_path, file_id); /* Attempt to pull file information */ if (fstat(fd, &file_stat) == 0) { /* Load size and times */ file->size = file_stat.st_size; file->ctime = file_stat.st_ctime; file->mtime = file_stat.st_mtime; file->atime = file_stat.st_atime; file->stmode = file_stat.st_mode; } /* If information cannot be retrieved, fake it */ else { /* Init information to 0, lacking any alternative */ file->size = 0; file->ctime = 0; file->mtime = 0; file->atime = 0; file->stmode = 0; } folder->open_files++; return file_id; } int guac_spice_folder_read(guac_spice_folder* folder, int file_id, uint64_t offset, void* buffer, int length) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Attempt to read from file: %s", __func__, folder->path); int bytes_read; guac_spice_folder_file* file = guac_spice_folder_get_file(folder, file_id); if (file == NULL) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Read from bad file_id: %i", __func__, file_id); return GUAC_SPICE_FOLDER_EINVAL; } /* Attempt read */ lseek(file->fd, offset, SEEK_SET); bytes_read = read(file->fd, buffer, length); /* Translate errno on error */ if (bytes_read < 0) return guac_spice_folder_get_errorcode(errno); return bytes_read; } const char* guac_spice_folder_read_dir(guac_spice_folder* folder, int file_id) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Attempt to read directory: %s", __func__, folder->path); guac_spice_folder_file* file; struct dirent* result; /* Only read if file ID is valid */ if (file_id < 0 || file_id >= GUAC_SPICE_FOLDER_MAX_FILES) return NULL; file = &(folder->files[file_id]); /* Open directory if not yet open, stop if error */ if (file->dir == NULL) { file->dir = fdopendir(file->fd); if (file->dir == NULL) return NULL; } /* Read next entry, stop if error or no more entries */ if ((result = readdir(file->dir)) == NULL) return NULL; /* Return filename */ return result->d_name; } int guac_spice_folder_write(guac_spice_folder* folder, int file_id, uint64_t offset, void* buffer, int length) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Attempt to write file: %s", __func__, folder->path); int bytes_written; guac_spice_folder_file* file = guac_spice_folder_get_file(folder, file_id); if (file == NULL) { guac_client_log(folder->client, GUAC_LOG_DEBUG, "%s: Write to bad file_id: %i", __func__, file_id); return GUAC_SPICE_FOLDER_EINVAL; } /* Attempt write */ lseek(file->fd, offset, SEEK_SET); bytes_written = write(file->fd, buffer, length); /* Translate errno on error */ if (bytes_written < 0) return guac_spice_folder_get_errorcode(errno); file->bytes_written += bytes_written; return bytes_written; } void guac_spice_client_file_transfer_handler(SpiceMainChannel* main_channel, SpiceFileTransferTask* task, guac_client* client) { guac_client_log(client, GUAC_LOG_DEBUG, "File transfer handler."); } #include "file.h"