243 lines
7.3 KiB
C
243 lines
7.3 KiB
C
|
/*
|
||
|
* 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 "client.h"
|
||
|
#include "download.h"
|
||
|
#include "fs.h"
|
||
|
#include "ls.h"
|
||
|
#include "rdp.h"
|
||
|
|
||
|
#include <freerdp/channels/channels.h>
|
||
|
#include <freerdp/client/cliprdr.h>
|
||
|
#include <freerdp/freerdp.h>
|
||
|
#include <guacamole/client.h>
|
||
|
#include <guacamole/protocol.h>
|
||
|
#include <guacamole/socket.h>
|
||
|
#include <guacamole/stream.h>
|
||
|
#include <guacamole/string.h>
|
||
|
#include <winpr/stream.h>
|
||
|
#include <winpr/wtypes.h>
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
int guac_rdp_download_ack_handler(guac_user* user, guac_stream* stream,
|
||
|
char* message, guac_protocol_status status) {
|
||
|
|
||
|
guac_client* client = user->client;
|
||
|
guac_rdp_client* rdp_client = (guac_rdp_client*) client->data;
|
||
|
guac_rdp_download_status* download_status = (guac_rdp_download_status*) stream->data;
|
||
|
|
||
|
/* Get filesystem, return error if no filesystem */
|
||
|
guac_rdp_fs* fs = rdp_client->filesystem;
|
||
|
if (fs == NULL) {
|
||
|
guac_protocol_send_ack(user->socket, stream, "FAIL (NO FS)",
|
||
|
GUAC_PROTOCOL_STATUS_SERVER_ERROR);
|
||
|
guac_socket_flush(user->socket);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* If successful, read data */
|
||
|
if (status == GUAC_PROTOCOL_STATUS_SUCCESS) {
|
||
|
|
||
|
/* Attempt read into buffer */
|
||
|
char buffer[4096];
|
||
|
int bytes_read = guac_rdp_fs_read(fs,
|
||
|
download_status->file_id,
|
||
|
download_status->offset, buffer, sizeof(buffer));
|
||
|
|
||
|
/* If bytes read, send as blob */
|
||
|
if (bytes_read > 0) {
|
||
|
download_status->offset += bytes_read;
|
||
|
guac_protocol_send_blob(user->socket, stream,
|
||
|
buffer, bytes_read);
|
||
|
}
|
||
|
|
||
|
/* If EOF, send end */
|
||
|
else if (bytes_read == 0) {
|
||
|
guac_protocol_send_end(user->socket, stream);
|
||
|
guac_user_free_stream(user, stream);
|
||
|
free(download_status);
|
||
|
}
|
||
|
|
||
|
/* Otherwise, fail stream */
|
||
|
else {
|
||
|
guac_user_log(user, GUAC_LOG_ERROR,
|
||
|
"Error reading file for download");
|
||
|
guac_protocol_send_end(user->socket, stream);
|
||
|
guac_user_free_stream(user, stream);
|
||
|
free(download_status);
|
||
|
}
|
||
|
|
||
|
guac_socket_flush(user->socket);
|
||
|
|
||
|
}
|
||
|
|
||
|
/* Otherwise, return stream to user */
|
||
|
else
|
||
|
guac_user_free_stream(user, stream);
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
}
|
||
|
|
||
|
int guac_rdp_download_get_handler(guac_user* user, guac_object* object,
|
||
|
char* name) {
|
||
|
|
||
|
guac_client* client = user->client;
|
||
|
guac_rdp_client* rdp_client = (guac_rdp_client*) client->data;
|
||
|
|
||
|
/* Get filesystem, ignore request if no filesystem */
|
||
|
guac_rdp_fs* fs = rdp_client->filesystem;
|
||
|
if (fs == NULL)
|
||
|
return 0;
|
||
|
|
||
|
/* Attempt to open file for reading */
|
||
|
int file_id = guac_rdp_fs_open(fs, name, GENERIC_READ, 0, FILE_OPEN, 0);
|
||
|
if (file_id < 0) {
|
||
|
guac_user_log(user, GUAC_LOG_INFO, "Unable to read file \"%s\"",
|
||
|
name);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* Get opened file */
|
||
|
guac_rdp_fs_file* file = guac_rdp_fs_get_file(fs, file_id);
|
||
|
if (file == NULL) {
|
||
|
guac_client_log(fs->client, GUAC_LOG_DEBUG,
|
||
|
"%s: Successful open produced bad file_id: %i",
|
||
|
__func__, file_id);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* If directory, send contents of directory */
|
||
|
if (file->attributes & FILE_ATTRIBUTE_DIRECTORY) {
|
||
|
|
||
|
/* Create stream data */
|
||
|
guac_rdp_ls_status* ls_status = malloc(sizeof(guac_rdp_ls_status));
|
||
|
ls_status->fs = fs;
|
||
|
ls_status->file_id = file_id;
|
||
|
guac_strlcpy(ls_status->directory_name, name,
|
||
|
sizeof(ls_status->directory_name));
|
||
|
|
||
|
/* Allocate stream for body */
|
||
|
guac_stream* stream = guac_user_alloc_stream(user);
|
||
|
stream->ack_handler = guac_rdp_ls_ack_handler;
|
||
|
stream->data = ls_status;
|
||
|
|
||
|
/* Init JSON object state */
|
||
|
guac_common_json_begin_object(user, stream,
|
||
|
&ls_status->json_state);
|
||
|
|
||
|
/* Associate new stream with get request */
|
||
|
guac_protocol_send_body(user->socket, object, stream,
|
||
|
GUAC_USER_STREAM_INDEX_MIMETYPE, name);
|
||
|
|
||
|
}
|
||
|
|
||
|
/* Otherwise, send file contents */
|
||
|
else {
|
||
|
|
||
|
/* Create stream data */
|
||
|
guac_rdp_download_status* download_status = malloc(sizeof(guac_rdp_download_status));
|
||
|
download_status->file_id = file_id;
|
||
|
download_status->offset = 0;
|
||
|
|
||
|
/* Allocate stream for body */
|
||
|
guac_stream* stream = guac_user_alloc_stream(user);
|
||
|
stream->data = download_status;
|
||
|
stream->ack_handler = guac_rdp_download_ack_handler;
|
||
|
|
||
|
/* Associate new stream with get request */
|
||
|
guac_protocol_send_body(user->socket, object, stream,
|
||
|
"application/octet-stream", name);
|
||
|
|
||
|
}
|
||
|
|
||
|
guac_socket_flush(user->socket);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void* guac_rdp_download_to_user(guac_user* user, void* data) {
|
||
|
|
||
|
/* Do not bother attempting the download if the user has left */
|
||
|
if (user == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
guac_client* client = user->client;
|
||
|
guac_rdp_client* rdp_client = (guac_rdp_client*) client->data;
|
||
|
guac_rdp_fs* filesystem = rdp_client->filesystem;
|
||
|
|
||
|
/* Ignore download if filesystem has been unloaded */
|
||
|
if (filesystem == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
/* Attempt to open requested file */
|
||
|
char* path = (char*) data;
|
||
|
int file_id = guac_rdp_fs_open(filesystem, path,
|
||
|
FILE_READ_DATA, 0, FILE_OPEN, 0);
|
||
|
|
||
|
/* If file opened successfully, start stream */
|
||
|
if (file_id >= 0) {
|
||
|
|
||
|
guac_rdp_download_status* download_status;
|
||
|
const char* basename;
|
||
|
|
||
|
int i;
|
||
|
char c;
|
||
|
|
||
|
/* Associate stream with transfer status */
|
||
|
guac_stream* stream = guac_user_alloc_stream(user);
|
||
|
stream->data = download_status = malloc(sizeof(guac_rdp_download_status));
|
||
|
stream->ack_handler = guac_rdp_download_ack_handler;
|
||
|
download_status->file_id = file_id;
|
||
|
download_status->offset = 0;
|
||
|
|
||
|
/* Get basename from absolute path */
|
||
|
i=0;
|
||
|
basename = path;
|
||
|
do {
|
||
|
|
||
|
c = path[i];
|
||
|
if (c == '/' || c == '\\')
|
||
|
basename = &(path[i+1]);
|
||
|
|
||
|
i++;
|
||
|
|
||
|
} while (c != '\0');
|
||
|
|
||
|
guac_user_log(user, GUAC_LOG_DEBUG, "%s: Initiating download "
|
||
|
"of \"%s\"", __func__, path);
|
||
|
|
||
|
/* Begin stream */
|
||
|
guac_protocol_send_file(user->socket, stream,
|
||
|
"application/octet-stream", basename);
|
||
|
guac_socket_flush(user->socket);
|
||
|
|
||
|
/* Download started successfully */
|
||
|
return stream;
|
||
|
|
||
|
}
|
||
|
|
||
|
/* Download failed */
|
||
|
guac_user_log(user, GUAC_LOG_ERROR, "Unable to download \"%s\"", path);
|
||
|
return NULL;
|
||
|
|
||
|
}
|
||
|
|