mirror of
https://github.com/akuker/RASCSI.git
synced 2024-06-08 05:29:40 +00:00
b319d72601
commit 0ab4918c5a59f978f48cf26f431ff809e9ddae33 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Mon Dec 20 16:52:03 2021 +0100 Scan depth determines availability of folder filter commit 16590cc4e4420a348fae610d749082c9d718be0a Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Mon Dec 20 15:47:08 2021 +0100 Updated operation count check commit 82f7c99755f535a7a5c30fe66e377705c5306faa Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Mon Dec 20 15:41:32 2021 +0100 Only restrict shutdown parameters, not everything if not root commit 9bd50d37b11c48b2130e4f6e66d12def88ddc38f Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Mon Dec 20 15:28:08 2021 +0100 Shutdown functionality is only available if started with root permissions commit aa5f3331abf4c178e8ce738c14fd584bd41d1b94 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Mon Dec 20 10:26:14 2021 +0100 Squashed commit of the following: commit4ae273ccbd
Author: akuker <34318535+akuker@users.noreply.github.com> Date: Sun Dec 19 22:30:22 2021 -0600 Loopback tester pcb (#545) commit46c5c1966f
Author: akuker <34318535+akuker@users.noreply.github.com> Date: Sun Dec 19 22:29:59 2021 -0600 RaSCSI Zero version 1.0 (#546) commitd09df31d67
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 20:25:23 2021 -0800 Remove redundant code from OLED script (#547) commitd8828da690
Author: akuker <34318535+akuker@users.noreply.github.com> Date: Sun Dec 19 21:02:50 2021 -0600 Added list of sponsors commitbcd7e8396d
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 15:51:45 2021 -0800 Second attempt at properly creating the manpage dir (#542) commitc887edfc8c
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 15:50:03 2021 -0800 Remove special elevated privileges for the Web Interface (#536) * Use the pi_shutdown method to restart the rascsi service * Use the pi_shutdown method to restart the rascsi service * Remove modifications to sudoers no longer needed * Introduce sleeps attempting to connect to socket; reduce overall number of retries * Remove systemd helped methods and the functionality that depends on it * Attempts to speed up splash code * Remove unneccessary verbosity * Attempt to optimize service definition commit801aebfb96
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 15:47:22 2021 -0800 More readable message when downloading a file (#531) commit29cf58288f
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 15:47:03 2021 -0800 Add a warning notice when ejecting removable media (#526) commit7efa895239
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 15:46:22 2021 -0800 Unzip zipfiles before storing to iso (#525) * Unzip zipfiles before storing to iso * Add helptext * Skip unzip for MacZip format * Should not be an fstring commit39bc485671
Author: Daniel Markstedt <markstedt@gmail.com> Date: Sun Dec 19 15:28:22 2021 -0800 Add pip3 to global dependencies; remove duplicates from monitor_rascs… (#523) * Add pip3 to global dependencies; remove duplicates from monitor_rascsi dependencies * Cleanup commit a1f4b28f92bd9f7cdba18c04f61d3721fc7c720f Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 21:50:15 2021 +0100 Comment update commit 9cbc06caf5f00ce85e0f7f984c871ed614e2e483 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 21:49:09 2021 +0100 Option update commit c4aa39c2285c5c72e2ea28ad749f5670dd10e89f Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 21:46:37 2021 +0100 Merged feature_folder_filter commitea386fc74c
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 19:05:07 2021 +0100 Code cleanup commit17c3201135
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 19:03:13 2021 +0100 Removed duplicate code commit296f816dd3
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 18:40:49 2021 +0100 Logging update commit73e0df8557
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 17:28:02 2021 +0100 Code cleanup commit405dbb034c
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 17:13:40 2021 +0100 Removed duplicate cod3 commitc7c168a942
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 17:08:02 2021 +0100 Added missing condition commit6af5394f78
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 16:16:12 2021 +0100 Updated assertion handling commit632fe1acd8
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 14:33:01 2021 +0100 Updated handling of mandatory parameters commita4e0d506c6
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 14:14:09 2021 +0100 Aded operation parameters commitfc783e6a43
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 12:38:35 2021 +0100 Fixed typo commitcb1b498459
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 12:37:13 2021 +0100 Interface comment update commite2d4347ce6
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 12:35:50 2021 +0100 Logging update commitcecb72df3e
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 12:13:44 2021 +0100 Added operation commitbbf153ccd6
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 12:09:49 2021 +0100 Fixed typo commit29fa5c2f96
Merge:28a36fa
ec31198
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sun Dec 19 12:07:34 2021 +0100 Merge branch 'develop' into feature_meta_data commit28a36fa308
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 15:51:02 2021 +0100 Description updates commit73df9f136c
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 14:37:58 2021 +0100 Comment update commitc3ea3c8b37
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 14:34:48 2021 +0100 Renaming commit6a84edd0fb
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 14:22:31 2021 +0100 Comment update commitc0d6e66afe
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 14:20:20 2021 +0100 Code cleanup commitcc81b588eb
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 14:18:31 2021 +0100 Renaming commitc88628e12a
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 14:17:50 2021 +0100 Sort map by operation name commitb64001e8a4
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:39:44 2021 +0100 Data type update commit8177cd3062
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:37:23 2021 +0100 Added cast commitb8599ba088
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:35:40 2021 +0100 Added safeguard against unknown operations commit6b14ba6511
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:26:21 2021 +0100 Map operations by ordinal commitee101f2c6b
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:14:31 2021 +0100 Comment update commit67c958ed37
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:13:06 2021 +0100 Comment update commitd1a9c40745
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 13:04:58 2021 +0100 Default value update commitd9dbbc0bb3
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:47:36 2021 +0100 Squashed commit of the following: commit 8171c6ea27982c736c30c0db69a7fdde07ee10ce Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:43:14 2021 +0100 The data type is implicit commit fb01dc9d82e8ff7456b05a0cb9d08069adacc64c Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:37:49 2021 +0100 Renaming commit 057dbf1aca7be3f7e76a5ff89a582a276b6d3089 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:29:54 2021 +0100 Comment update commit 5f699aad2f835f72accdb445d1e59f094aeb108f Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:24:25 2021 +0100 Signature update commit cbcf8b09f9d1ba7b82f816269bcfe91d9f00eb6e Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:22:45 2021 +0100 Signature update commit a8148ef802ca809e5a305d2caa69856c9033d932 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:16:46 2021 +0100 Comment update commit ce685a92d4827e131d80d10ecd56e2b3baf173f8 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 12:15:46 2021 +0100 Use map instead of list commit 454c0438f3589904f5dbe5253963dd200ea416dd Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 10:47:36 2021 +0100 Updated size check commit b386dbba4b0262f4f6f02aecb2a1daeffd41f4a2 Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Sat Dec 18 01:23:43 2021 +0100 Initial improvements commit5d6862b6b0
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 23:22:27 2021 +0100 Description upddate commit69263b3e4b
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 23:21:38 2021 +0100 Description update commit49e14f7078
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 23:19:08 2021 +0100 Removed redundant message field commitff468aafa8
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 23:09:18 2021 +0100 Added support for default value commit2da717a0a0
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 22:57:59 2021 +0100 For completeness sake added permitted boolean values commit5d894d2e4f
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 22:53:07 2021 +0100 Display permitted values commitacc7d3cba5
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 20:37:03 2021 +0100 Comment update commitf846242aea
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 20:35:07 2021 +0100 Code cleanup commit5a9592f102
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 20:24:33 2021 +0100 Added convenience method commit9d258d9979
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 20:05:53 2021 +0100 Added convenience method commit6c4103989b
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:45:20 2021 +0100 Comment update commit7d543451f0
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:27:50 2021 +0100 Description updates commitf4b0e50e66
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:23:29 2021 +0100 Comment update commit35dd3f6282
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:21:12 2021 +0100 Type update commit7a94c0e6e0
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:17:05 2021 +0100 Manpage update commit4179110bac
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:13:42 2021 +0100 rasctl -s returns operation meta data commiteed83bb005
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 19:05:29 2021 +0100 Completed meta data commit4a7528d9d3
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 18:39:18 2021 +0100 Sort operations commitd3af9a142c
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 18:29:22 2021 +0100 Support for localized descriptions commitb4ff4f52ab
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 17:53:11 2021 +0100 Interface update commite8d9e97fe5
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 16:59:49 2021 +0100 Comment update commit22753b1547
Author: Uwe Seimet <Uwe.Seimet@seimet.de> Date: Fri Dec 17 16:57:39 2021 +0100 Added messages
532 lines
20 KiB
C++
532 lines
20 KiB
C++
//---------------------------------------------------------------------------
|
|
//
|
|
// SCSI Target Emulator RaSCSI (*^..^*)
|
|
// for Raspberry Pi
|
|
//
|
|
// Copyright (C) 2021 Uwe Seimet
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
#include "devices/file_support.h"
|
|
#include "devices/disk.h"
|
|
#include "devices/device_factory.h"
|
|
#include "devices/device.h"
|
|
#include "protobuf_util.h"
|
|
#include "rascsi_version.h"
|
|
#include "rascsi_interface.pb.h"
|
|
#include "rascsi_image.h"
|
|
#include "rascsi_response.h"
|
|
#include <sstream>
|
|
|
|
using namespace rascsi_interface;
|
|
using namespace protobuf_util;
|
|
|
|
RascsiResponse::RascsiResponse(DeviceFactory *device_factory, const RascsiImage *rascsi_image)
|
|
{
|
|
this->device_factory = device_factory;
|
|
this->rascsi_image = rascsi_image;
|
|
}
|
|
|
|
PbDeviceProperties *RascsiResponse::GetDeviceProperties(const Device *device)
|
|
{
|
|
PbDeviceProperties *properties = new PbDeviceProperties();
|
|
|
|
properties->set_luns(device->GetSupportedLuns());
|
|
properties->set_read_only(device->IsReadOnly());
|
|
properties->set_protectable(device->IsProtectable());
|
|
properties->set_stoppable(device->IsStoppable());
|
|
properties->set_removable(device->IsRemovable());
|
|
properties->set_lockable(device->IsLockable());
|
|
properties->set_supports_file(dynamic_cast<const FileSupport *>(device));
|
|
properties->set_supports_params(device->SupportsParams());
|
|
|
|
PbDeviceType t = UNDEFINED;
|
|
PbDeviceType_Parse(device->GetType(), &t);
|
|
|
|
if (device->SupportsParams()) {
|
|
for (const auto& param : device_factory->GetDefaultParams(t)) {
|
|
auto& map = *properties->mutable_default_params();
|
|
map[param.first] = param.second;
|
|
}
|
|
}
|
|
|
|
for (const auto& block_size : device_factory->GetSectorSizes(t)) {
|
|
properties->add_block_sizes(block_size);
|
|
}
|
|
|
|
return properties;
|
|
}
|
|
|
|
void RascsiResponse::GetDeviceTypeProperties(PbDeviceTypesInfo& device_types_info, PbDeviceType type)
|
|
{
|
|
PbDeviceTypeProperties *type_properties = device_types_info.add_properties();
|
|
type_properties->set_type(type);
|
|
Device *device = device_factory->CreateDevice(type, "");
|
|
type_properties->set_allocated_properties(GetDeviceProperties(device));
|
|
delete device;
|
|
}
|
|
|
|
void RascsiResponse::GetAllDeviceTypeProperties(PbDeviceTypesInfo& device_types_info)
|
|
{
|
|
GetDeviceTypeProperties(device_types_info, SAHD);
|
|
GetDeviceTypeProperties(device_types_info, SCHD);
|
|
GetDeviceTypeProperties(device_types_info, SCRM);
|
|
GetDeviceTypeProperties(device_types_info, SCMO);
|
|
GetDeviceTypeProperties(device_types_info, SCCD);
|
|
GetDeviceTypeProperties(device_types_info, SCBR);
|
|
GetDeviceTypeProperties(device_types_info, SCDP);
|
|
}
|
|
|
|
void RascsiResponse::GetDevice(const Device *device, PbDevice *pb_device)
|
|
{
|
|
pb_device->set_id(device->GetId());
|
|
pb_device->set_unit(device->GetLun());
|
|
pb_device->set_vendor(device->GetVendor());
|
|
pb_device->set_product(device->GetProduct());
|
|
pb_device->set_revision(device->GetRevision());
|
|
|
|
PbDeviceType type = UNDEFINED;
|
|
PbDeviceType_Parse(device->GetType(), &type);
|
|
pb_device->set_type(type);
|
|
|
|
pb_device->set_allocated_properties(GetDeviceProperties(device));
|
|
|
|
PbDeviceStatus *status = new PbDeviceStatus();
|
|
pb_device->set_allocated_status(status);
|
|
status->set_protected_(device->IsProtected());
|
|
status->set_stopped(device->IsStopped());
|
|
status->set_removed(device->IsRemoved());
|
|
status->set_locked(device->IsLocked());
|
|
|
|
if (device->SupportsParams()) {
|
|
for (const auto& param : device->GetParams()) {
|
|
AddParam(*pb_device, param.first, param.second);
|
|
}
|
|
}
|
|
|
|
const Disk *disk = dynamic_cast<const Disk*>(device);
|
|
if (disk) {
|
|
pb_device->set_block_size(device->IsRemoved()? 0 : disk->GetSectorSizeInBytes());
|
|
pb_device->set_block_count(device->IsRemoved() ? 0: disk->GetBlockCount());
|
|
}
|
|
|
|
const FileSupport *file_support = dynamic_cast<const FileSupport *>(device);
|
|
if (file_support) {
|
|
Filepath filepath;
|
|
file_support->GetPath(filepath);
|
|
PbImageFile *image_file = new PbImageFile();
|
|
GetImageFile(image_file, device->IsRemovable() && !device->IsReady() ? "" : filepath.GetPath());
|
|
pb_device->set_allocated_file(image_file);
|
|
}
|
|
}
|
|
|
|
bool RascsiResponse::GetImageFile(PbImageFile *image_file, const string& filename)
|
|
{
|
|
if (!filename.empty()) {
|
|
image_file->set_name(filename);
|
|
image_file->set_type(device_factory->GetTypeForFile(filename));
|
|
|
|
string f = filename[0] == '/' ? filename : rascsi_image->GetDefaultImageFolder() + "/" + filename;
|
|
|
|
image_file->set_read_only(access(f.c_str(), W_OK));
|
|
|
|
struct stat st;
|
|
if (!stat(f.c_str(), &st) && !S_ISDIR(st.st_mode)) {
|
|
image_file->set_size(st.st_size);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void RascsiResponse::GetAvailableImages(PbImageFilesInfo& image_files_info, const string& default_image_folder,
|
|
const string& folder, const string& folder_pattern, const string& file_pattern, int scan_depth) {
|
|
string folder_pattern_lower = folder_pattern;
|
|
transform(folder_pattern_lower.begin(), folder_pattern_lower.end(), folder_pattern_lower.begin(), ::tolower);
|
|
|
|
string file_pattern_lower = file_pattern;
|
|
transform(file_pattern_lower.begin(), file_pattern_lower.end(), file_pattern_lower.begin(), ::tolower);
|
|
|
|
if (scan_depth-- >= 0) {
|
|
DIR *d = opendir(folder.c_str());
|
|
if (d) {
|
|
struct dirent *dir;
|
|
while ((dir = readdir(d))) {
|
|
bool is_supported_type = dir->d_type == DT_REG || dir->d_type == DT_DIR || dir->d_type == DT_LNK || dir->d_type == DT_BLK;
|
|
if (is_supported_type && dir->d_name[0] != '.') {
|
|
string name_lower = dir->d_name;
|
|
if (!file_pattern.empty()) {
|
|
transform(name_lower.begin(), name_lower.end(), name_lower.begin(), ::tolower);
|
|
}
|
|
|
|
string filename = folder + "/" + dir->d_name;
|
|
|
|
struct stat st;
|
|
if (dir->d_type == DT_REG && !stat(filename.c_str(), &st)) {
|
|
if (!st.st_size) {
|
|
LOGTRACE("File '%s' in image folder '%s' has a size of 0 bytes", dir->d_name, folder.c_str());
|
|
continue;
|
|
}
|
|
} else if (dir->d_type == DT_LNK && stat(filename.c_str(), &st)) {
|
|
LOGTRACE("Symlink '%s' in image folder '%s' is broken", dir->d_name, folder.c_str());
|
|
continue;
|
|
} else if (dir->d_type == DT_DIR) {
|
|
if (folder_pattern_lower.empty() || name_lower.find(folder_pattern_lower) != string::npos) {
|
|
GetAvailableImages(image_files_info, default_image_folder, filename, folder_pattern,
|
|
file_pattern, scan_depth);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (file_pattern_lower.empty() || name_lower.find(file_pattern_lower) != string::npos) {
|
|
PbImageFile *image_file = new PbImageFile();
|
|
if (GetImageFile(image_file, filename)) {
|
|
GetImageFile(image_files_info.add_image_files(), filename.substr(default_image_folder.length() + 1));
|
|
}
|
|
delete image_file;
|
|
}
|
|
}
|
|
}
|
|
|
|
closedir(d);
|
|
}
|
|
}
|
|
}
|
|
|
|
PbImageFilesInfo *RascsiResponse::GetAvailableImages(PbResult& result, const string& folder_pattern,
|
|
const string& file_pattern, int scan_depth)
|
|
{
|
|
PbImageFilesInfo *image_files_info = new PbImageFilesInfo();
|
|
|
|
string default_image_folder = rascsi_image->GetDefaultImageFolder();
|
|
image_files_info->set_default_image_folder(default_image_folder);
|
|
|
|
GetAvailableImages(*image_files_info, default_image_folder, default_image_folder, folder_pattern, file_pattern,
|
|
scan_depth);
|
|
|
|
result.set_status(true);
|
|
|
|
return image_files_info;
|
|
}
|
|
|
|
void RascsiResponse::GetAvailableImages(PbResult& result, PbServerInfo& server_info, const string& folder_pattern,
|
|
const string& file_pattern, int scan_depth)
|
|
{
|
|
PbImageFilesInfo *image_files_info = GetAvailableImages(result, folder_pattern, file_pattern, scan_depth);
|
|
image_files_info->set_default_image_folder(rascsi_image->GetDefaultImageFolder());
|
|
server_info.set_allocated_image_files_info(image_files_info);
|
|
|
|
result.set_status(true);
|
|
}
|
|
|
|
PbReservedIdsInfo *RascsiResponse::GetReservedIds(PbResult& result, const set<int>& ids)
|
|
{
|
|
PbReservedIdsInfo *reserved_ids_info = new PbReservedIdsInfo();
|
|
for (int id : ids) {
|
|
reserved_ids_info->add_ids(id);
|
|
}
|
|
|
|
result.set_status(true);
|
|
|
|
return reserved_ids_info;
|
|
}
|
|
|
|
void RascsiResponse::GetDevices(PbServerInfo& server_info, const vector<Device *>& devices)
|
|
{
|
|
for (const Device *device : devices) {
|
|
// Skip if unit does not exist or is not assigned
|
|
if (device) {
|
|
PbDevice *pb_device = server_info.mutable_devices_info()->add_devices();
|
|
GetDevice(device, pb_device);
|
|
}
|
|
}
|
|
}
|
|
|
|
void RascsiResponse::GetDevicesInfo(PbResult& result, const PbCommand& command, const vector<Device *>& devices,
|
|
int unit_count)
|
|
{
|
|
set<id_set> id_sets;
|
|
if (!command.devices_size()) {
|
|
for (const Device *device : devices) {
|
|
if (device) {
|
|
id_sets.insert(make_pair(device->GetId(), device->GetLun()));
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
for (const auto& device : command.devices()) {
|
|
if (devices[device.id() * unit_count + device.unit()]) {
|
|
id_sets.insert(make_pair(device.id(), device.unit()));
|
|
}
|
|
else {
|
|
ostringstream error;
|
|
error << "No device for ID " << device.id() << ", unit " << device.unit();
|
|
result.set_status(false);
|
|
result.set_msg(error.str());
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
PbDevicesInfo *devices_info = new PbDevicesInfo();
|
|
result.set_allocated_devices_info(devices_info);
|
|
|
|
for (const auto& id_set : id_sets) {
|
|
const Device *device = devices[id_set.first * unit_count + id_set.second];
|
|
GetDevice(device, devices_info->add_devices());
|
|
}
|
|
|
|
result.set_status(true);
|
|
}
|
|
|
|
PbDeviceTypesInfo *RascsiResponse::GetDeviceTypesInfo(PbResult& result, const PbCommand& command)
|
|
{
|
|
PbDeviceTypesInfo *device_types_info = new PbDeviceTypesInfo();
|
|
|
|
GetAllDeviceTypeProperties(*device_types_info);
|
|
|
|
result.set_status(true);
|
|
|
|
return device_types_info;
|
|
}
|
|
|
|
PbServerInfo *RascsiResponse::GetServerInfo(PbResult& result, const vector<Device *>& devices, const set<int>& reserved_ids,
|
|
const string& current_log_level, const string& folder_pattern, const string& file_pattern, int scan_depth)
|
|
{
|
|
PbServerInfo *server_info = new PbServerInfo();
|
|
|
|
server_info->set_allocated_version_info(GetVersionInfo(result));
|
|
server_info->set_allocated_log_level_info(GetLogLevelInfo(result, current_log_level));
|
|
GetAllDeviceTypeProperties(*server_info->mutable_device_types_info());
|
|
GetAvailableImages(result, *server_info, folder_pattern, file_pattern, scan_depth);
|
|
server_info->set_allocated_network_interfaces_info(GetNetworkInterfacesInfo(result));
|
|
server_info->set_allocated_mapping_info(GetMappingInfo(result));
|
|
GetDevices(*server_info, devices);
|
|
server_info->set_allocated_reserved_ids_info(GetReservedIds(result, reserved_ids));
|
|
server_info->set_allocated_operation_info(GetOperationInfo(result, scan_depth));
|
|
|
|
result.set_status(true);
|
|
|
|
return server_info;
|
|
}
|
|
|
|
PbVersionInfo *RascsiResponse::GetVersionInfo(PbResult& result)
|
|
{
|
|
PbVersionInfo *version_info = new PbVersionInfo();
|
|
|
|
version_info->set_major_version(rascsi_major_version);
|
|
version_info->set_minor_version(rascsi_minor_version);
|
|
version_info->set_patch_version(rascsi_patch_version);
|
|
|
|
result.set_status(true);
|
|
|
|
return version_info;
|
|
}
|
|
|
|
PbLogLevelInfo *RascsiResponse::GetLogLevelInfo(PbResult& result, const string& current_log_level)
|
|
{
|
|
PbLogLevelInfo *log_level_info = new PbLogLevelInfo();
|
|
|
|
for (const auto& log_level : log_levels) {
|
|
log_level_info->add_log_levels(log_level);
|
|
}
|
|
|
|
log_level_info->set_current_log_level(current_log_level);
|
|
|
|
result.set_status(true);
|
|
|
|
return log_level_info;
|
|
}
|
|
|
|
PbNetworkInterfacesInfo *RascsiResponse::GetNetworkInterfacesInfo(PbResult& result)
|
|
{
|
|
PbNetworkInterfacesInfo *network_interfaces_info = new PbNetworkInterfacesInfo();
|
|
|
|
for (const auto& network_interface : device_factory->GetNetworkInterfaces()) {
|
|
network_interfaces_info->add_name(network_interface);
|
|
}
|
|
|
|
result.set_status(true);
|
|
|
|
return network_interfaces_info;
|
|
}
|
|
|
|
PbMappingInfo *RascsiResponse::GetMappingInfo(PbResult& result)
|
|
{
|
|
PbMappingInfo *mapping_info = new PbMappingInfo();
|
|
|
|
for (const auto& mapping : device_factory->GetExtensionMapping()) {
|
|
(*mapping_info->mutable_mapping())[mapping.first] = mapping.second;
|
|
}
|
|
|
|
result.set_status(true);
|
|
|
|
return mapping_info;
|
|
}
|
|
|
|
PbOperationInfo *RascsiResponse::GetOperationInfo(PbResult& result, int scan_depth)
|
|
{
|
|
PbOperationInfo *operation_info = new PbOperationInfo();
|
|
|
|
PbOperationMetaData *meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "name", "Image file name in case of a mass storage device");
|
|
AddOperationParameter(meta_data, "interfaces", "Comma-separated prioritized network interface list");
|
|
CreateOperation(operation_info, meta_data, ATTACH, "Attach device, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, DETACH, "Detach device, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, DETACH_ALL, "Detach all devices");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, START, "Start device, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, STOP, "Stop device, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "file", "Image file name", "", true);
|
|
CreateOperation(operation_info, meta_data, INSERT, "Insert medium, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, EJECT, "Eject medium, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, PROTECT, "Protect medium, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, UNPROTECT, "Unprotect medium, device-specific parameters are required");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
if (scan_depth) {
|
|
AddOperationParameter(meta_data, "folder_pattern", "Pattern for filtering folder names");
|
|
}
|
|
AddOperationParameter(meta_data, "file_pattern", "Pattern for filtering image file names");
|
|
CreateOperation(operation_info, meta_data, SERVER_INFO, "Get rascsi server information");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, VERSION_INFO, "Get rascsi server version");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, DEVICES_INFO, "Get information on attached devices");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, DEVICE_TYPES_INFO, "Get device properties by device type");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "filename_pattern", "Pattern for filtering image file names");
|
|
if (scan_depth) {
|
|
AddOperationParameter(meta_data, "folder_pattern", "Pattern for filtering folder names");
|
|
}
|
|
AddOperationParameter(meta_data, "file_pattern", "Pattern for filtering image file names");
|
|
CreateOperation(operation_info, meta_data, DEFAULT_IMAGE_FILES_INFO, "Get information on available image files");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "file", "Image file name", "", true);
|
|
CreateOperation(operation_info, meta_data, IMAGE_FILE_INFO, "Get information on image file");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, LOG_LEVEL_INFO, "Get log level information");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, NETWORK_INTERFACES_INFO, "Get the available network interfaces");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, MAPPING_INFO, "Get mapping of extensions to device types");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, RESERVED_IDS_INFO, "Get list of reserved device IDs");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "folder", "Default image file folder name", "", true);
|
|
CreateOperation(operation_info, meta_data, DEFAULT_FOLDER, "Set default image file folder");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "level", "New log level", "", true);
|
|
CreateOperation(operation_info, meta_data, LOG_LEVEL, "Set log level");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "ids", "Comma-separated device ID list", "", true);
|
|
CreateOperation(operation_info, meta_data, RESERVE_IDS, "Reserve device IDs");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
PbOperationParameter *parameter = AddOperationParameter(meta_data, "mode", "Shutdown mode", "", true);
|
|
parameter->add_permitted_values("rascsi");
|
|
// System shutdown/reboot requires root permissions
|
|
if (!getuid()) {
|
|
parameter->add_permitted_values("system");
|
|
parameter->add_permitted_values("reboot");
|
|
}
|
|
CreateOperation(operation_info, meta_data, SHUT_DOWN, "Shut down or reboot");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "file", "Image file name", "", true);
|
|
AddOperationParameter(meta_data, "size", "Image file size in bytes", "", true);
|
|
parameter = AddOperationParameter(meta_data, "read_only", "Read-only flag", "false");
|
|
parameter->add_permitted_values("true");
|
|
parameter->add_permitted_values("false");
|
|
CreateOperation(operation_info, meta_data, CREATE_IMAGE, "Create an image file");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "file", "Image file name", "", true);
|
|
CreateOperation(operation_info, meta_data, DELETE_IMAGE, "Delete image file");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "from", "Source image file name", "", true);
|
|
AddOperationParameter(meta_data, "to", "Destination image file name", "", true);
|
|
CreateOperation(operation_info, meta_data, RENAME_IMAGE, "Rename image file");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "from", "Source image file name image file name", "", true);
|
|
AddOperationParameter(meta_data, "to", "Destination image file name", "", true);
|
|
parameter = AddOperationParameter(meta_data, "read_only", "Read-only flag", "false");
|
|
parameter->add_permitted_values("true");
|
|
parameter->add_permitted_values("false");
|
|
CreateOperation(operation_info, meta_data, COPY_IMAGE, "Copy image file");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "file", "Image file name", "", true);
|
|
CreateOperation(operation_info, meta_data, PROTECT_IMAGE, "Write-protect image file");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "file", "Image file name", "", true);
|
|
CreateOperation(operation_info, meta_data, UNPROTECT_IMAGE, "Make image file writable");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
AddOperationParameter(meta_data, "token", "Authentication token to be checked", "", true);
|
|
CreateOperation(operation_info, meta_data, CHECK_AUTHENTICATION, "Check whether an authentication token is valid");
|
|
|
|
meta_data = new PbOperationMetaData();
|
|
CreateOperation(operation_info, meta_data, OPERATION_INFO, "Get operation meta data");
|
|
|
|
// Ensure that the complete set of operations is covered
|
|
assert(operation_info->operations_size() == PbOperation_ARRAYSIZE - 1);
|
|
|
|
result.set_status(true);
|
|
|
|
return operation_info;
|
|
}
|
|
|
|
void RascsiResponse::CreateOperation(PbOperationInfo *operation_info, PbOperationMetaData *meta_data,
|
|
const PbOperation& operation, const string& description)
|
|
{
|
|
meta_data->set_server_side_name(PbOperation_Name(operation));
|
|
(*meta_data->mutable_description())["en"] = description;
|
|
int ordinal = PbOperation_descriptor()->FindValueByName(PbOperation_Name(operation))->index();
|
|
(*operation_info->mutable_operations())[ordinal] = *meta_data;
|
|
}
|
|
|
|
PbOperationParameter *RascsiResponse::AddOperationParameter(PbOperationMetaData *meta_data, const string& name,
|
|
const string& description, const string& default_value, bool is_mandatory)
|
|
{
|
|
PbOperationParameter *parameter = meta_data->add_parameters();
|
|
parameter->set_name(name);
|
|
(*parameter->mutable_description())["en"] = description;
|
|
parameter->set_default_value(default_value);
|
|
parameter->set_is_mandatory(is_mandatory);
|
|
|
|
return parameter;
|
|
}
|