protobuf interface returns assumed device type (#259)

* Added device type to PbImageFile

* Comment update

* Code cleanup

* More code cleanup

* Extraced methods

* Extracted methods

* Renaming

* Moved code

* Visility update

* Updated error handling

* Removed duplicate error handling

* Removed duplicate code

* Code cleanup

* Moved code

* Moved code
This commit is contained in:
Uwe Seimet 2021-09-24 08:48:48 +02:00 committed by GitHub
parent fdee57e421
commit 829356a32c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 478 additions and 376 deletions

View File

@ -83,6 +83,7 @@ SRC_RASCSI = \
fileio.cpp\ fileio.cpp\
rascsi_version.cpp \ rascsi_version.cpp \
rasutil.cpp \ rasutil.cpp \
protobuf_response_helper.cpp \
protobuf_util.cpp protobuf_util.cpp
SRC_RASCSI += $(shell find ./controllers -name '*.cpp') SRC_RASCSI += $(shell find ./controllers -name '*.cpp')
SRC_RASCSI += $(shell find ./devices -name '*.cpp') SRC_RASCSI += $(shell find ./devices -name '*.cpp')

View File

@ -72,33 +72,50 @@ DeviceFactory& DeviceFactory::instance()
return instance; return instance;
} }
Device *DeviceFactory::CreateDevice(PbDeviceType type, const string& filename, const string& extension) string DeviceFactory::GetExtension(const string& filename)
{ {
string ext = extension; string ext;
size_t separator = filename.rfind('.');
if (separator != string::npos) {
ext = filename.substr(separator + 1);
}
std::transform(ext.begin(), ext.end(), ext.begin(), [](unsigned char c){ return std::tolower(c); }); std::transform(ext.begin(), ext.end(), ext.begin(), [](unsigned char c){ return std::tolower(c); });
// If no type was specified try to derive the device type from the filename and extension return ext;
if (type == UNDEFINED) { }
PbDeviceType DeviceFactory::GetTypeForFile(const string& filename)
{
string ext = GetExtension(filename);
if (ext == "hdf") { if (ext == "hdf") {
type = SAHD; return SAHD;
} }
else if (ext == "hds" || ext == "hdn" || ext == "hdi" || ext == "nhd" || ext == "hda") { else if (ext == "hds" || ext == "hdn" || ext == "hdi" || ext == "nhd" || ext == "hda") {
type = SCHD; return SCHD;
} }
else if (ext == "hdr") { else if (ext == "hdr") {
type = SCRM; return SCRM;
} else if (ext == "mos") { } else if (ext == "mos") {
type = SCMO; return SCMO;
} else if (ext == "iso") { } else if (ext == "iso") {
type = SCCD; return SCCD;
} }
else if (filename == "bridge") { else if (filename == "bridge") {
type = SCBR; return SCBR;
} }
else if (filename == "daynaport") { else if (filename == "daynaport") {
type = SCDP; return SCDP;
} }
else {
return UNDEFINED;
}
Device *DeviceFactory::CreateDevice(PbDeviceType type, const string& filename)
{
// If no type was specified try to derive the device type from the filename
if (type == UNDEFINED) {
type = GetTypeForFile(filename);
if (type == UNDEFINED) {
return NULL; return NULL;
} }
} }
@ -112,7 +129,8 @@ Device *DeviceFactory::CreateDevice(PbDeviceType type, const string& filename, c
((Disk *)device)->SetSectorSizes(sector_sizes[SAHD]); ((Disk *)device)->SetSectorSizes(sector_sizes[SAHD]);
break; break;
case SCHD: case SCHD: {
string ext = GetExtension(filename);
if (ext == "hdn" || ext == "hdi" || ext == "nhd") { if (ext == "hdn" || ext == "hdi" || ext == "nhd") {
device = new SCSIHD_NEC(); device = new SCSIHD_NEC();
((Disk *)device)->SetSectorSizes({ 512 }); ((Disk *)device)->SetSectorSizes({ 512 });
@ -124,6 +142,7 @@ Device *DeviceFactory::CreateDevice(PbDeviceType type, const string& filename, c
device->SetProtectable(true); device->SetProtectable(true);
device->SetStoppable(true); device->SetStoppable(true);
break; break;
}
case SCRM: case SCRM:
device = new SCSIHD(true); device = new SCSIHD(true);

View File

@ -5,7 +5,7 @@
// //
// Copyright (C) 2021 Uwe Seimet // Copyright (C) 2021 Uwe Seimet
// //
// The DeviceFactory singleton creates devices based on their type and the extension of their image file // The DeviceFactory singleton creates devices based on their type and the image file extension
// //
//--------------------------------------------------------------------------- //---------------------------------------------------------------------------
@ -33,7 +33,9 @@ public:
static DeviceFactory& instance(); static DeviceFactory& instance();
Device *CreateDevice(PbDeviceType type, const string& filename, const string& ext); static PbDeviceType GetTypeForFile(const string&);
Device *CreateDevice(PbDeviceType, const string&);
const set<uint32_t>& GetSectorSizes(PbDeviceType type) { return sector_sizes[type]; } const set<uint32_t>& GetSectorSizes(PbDeviceType type) { return sector_sizes[type]; }
const set<uint32_t>& GetSectorSizes(const string&); const set<uint32_t>& GetSectorSizes(const string&);
@ -48,4 +50,6 @@ private:
map<PbDeviceType, map<uint64_t, Geometry>> geometries; map<PbDeviceType, map<uint64_t, Geometry>> geometries;
map<PbDeviceType, map<string, string>> default_params; map<PbDeviceType, map<string, string>> default_params;
static string GetExtension(const string&);
}; };

View File

@ -0,0 +1,306 @@
//---------------------------------------------------------------------------
//
// 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 "protobuf_response_helper.h"
#include <sstream>
using namespace rascsi_interface;
ProtobufResponseHandler::ProtobufResponseHandler()
{
device_factory = DeviceFactory::instance();
log_levels.push_back("trace");
log_levels.push_back("debug");
log_levels.push_back("info");
log_levels.push_back("warn");
log_levels.push_back("err");
log_levels.push_back("critical");
log_levels.push_back("off");
}
ProtobufResponseHandler& ProtobufResponseHandler::instance()
{
static ProtobufResponseHandler instance;
return instance;
}
PbDeviceProperties *ProtobufResponseHandler::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);
}
for (const auto& capacity : device_factory.GetCapacities(t)) {
properties->add_capacities(capacity);
}
return properties;
}
void ProtobufResponseHandler::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 ProtobufResponseHandler::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 ProtobufResponseHandler::GetDevice(const Device *device, PbDevice *pb_device, const string& image_folder)
{
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(), image_folder);
pb_device->set_allocated_file(image_file);
}
}
void ProtobufResponseHandler::GetImageFile(PbImageFile *image_file, const string& filename, const string& image_folder)
{
image_file->set_name(filename);
if (!filename.empty()) {
image_file->set_type(DeviceFactory::GetTypeForFile(filename));
string f = filename[0] == '/' ? filename : image_folder + "/" + filename;
image_file->set_read_only(access(f.c_str(), W_OK));
struct stat st;
if (!stat(f.c_str(), &st)) {
image_file->set_size(st.st_size);
}
}
}
PbImageFilesInfo *ProtobufResponseHandler::GetAvailableImages(PbResult& result, const string& image_folder)
{
PbImageFilesInfo *image_files_info = new PbImageFilesInfo();
image_files_info->set_default_image_folder(image_folder);
// filesystem::directory_iterator cannot be used because libstdc++ 8.3.0 does not support big files
DIR *d = opendir(image_folder.c_str());
if (d) {
struct dirent *dir;
while ((dir = readdir(d))) {
if (dir->d_type == DT_REG || dir->d_type == DT_LNK || dir->d_type == DT_BLK) {
string filename = image_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, image_folder.c_str());
continue;
}
if (st.st_size % 512) {
LOGTRACE("Size of file '%s' in image folder '%s' is not a multiple of 512", dir->d_name, image_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, image_folder.c_str());
continue;
}
GetImageFile(image_files_info->add_image_files(), dir->d_name, image_folder);
}
}
closedir(d);
}
result.set_status(true);
return image_files_info;
}
void ProtobufResponseHandler::GetAvailableImages(PbResult& result, PbServerInfo& server_info, const string& image_folder)
{
PbImageFilesInfo *image_files_info = GetAvailableImages(result, image_folder);
image_files_info->set_default_image_folder(image_folder);
server_info.set_allocated_image_files_info(image_files_info);
result.set_status(true);
}
void ProtobufResponseHandler::GetDevices(PbServerInfo& serverInfo, const vector<Device *>& devices, const string& image_folder)
{
for (const Device *device : devices) {
// Skip if unit does not exist or is not assigned
if (device) {
PbDevice *pb_device = serverInfo.mutable_devices()->add_devices();
GetDevice(device, pb_device, image_folder);
}
}
}
void ProtobufResponseHandler::GetDevicesInfo(PbResult& result, const PbCommand& command, const vector<Device *>& devices,
const string& image_folder, 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;
}
}
}
PbDevices *pb_devices = new PbDevices();
result.set_allocated_device_info(pb_devices);
for (const auto& id_set : id_sets) {
const Device *device = devices[id_set.first * unit_count + id_set.second];
GetDevice(device, pb_devices->add_devices(), image_folder);
}
result.set_status(true);
}
PbDeviceTypesInfo *ProtobufResponseHandler::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 *ProtobufResponseHandler::GetServerInfo(PbResult& result, const vector<Device *>& devices, const set<int>& reserved_ids,
const string& image_folder, const string& log_level)
{
PbServerInfo *server_info = new PbServerInfo();
server_info->set_major_version(rascsi_major_version);
server_info->set_minor_version(rascsi_minor_version);
server_info->set_patch_version(rascsi_patch_version);
GetLogLevels(*server_info);
server_info->set_current_log_level(log_level);
GetAllDeviceTypeProperties(*server_info->mutable_device_types_info());
GetAvailableImages(result, *server_info, image_folder);
server_info->set_allocated_network_interfaces_info(GetNetworkInterfacesInfo(result));
GetDevices(*server_info, devices, image_folder);
for (int id : reserved_ids) {
server_info->add_reserved_ids(id);
}
result.set_status(true);
return server_info;
}
void ProtobufResponseHandler::GetLogLevels(PbServerInfo& server_info)
{
for (const auto& log_level : log_levels) {
server_info.add_log_levels(log_level);
}
}
PbNetworkInterfacesInfo *ProtobufResponseHandler::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;
}

View File

@ -0,0 +1,53 @@
//---------------------------------------------------------------------------
//
// SCSI Target Emulator RaSCSI (*^..^*)
// for Raspberry Pi
//
// Copyright (C) 2021 Uwe Seimet
//
// A singleton that creates responses for protobuf interface requests
//
//---------------------------------------------------------------------------
#pragma once
#include "devices/device_factory.h"
#include "rascsi_interface.pb.h"
#include <vector>
#include <string>
using namespace std;
using namespace rascsi_interface;
class Device;
class ProtobufResponseHandler
{
public:
ProtobufResponseHandler();
~ProtobufResponseHandler() {};
static ProtobufResponseHandler& instance();
PbImageFilesInfo *GetAvailableImages(PbResult&, const string&);
void GetDevices(PbServerInfo&, const vector<Device *>&, const string&);
void GetDevicesInfo(PbResult&, const PbCommand&, const vector<Device *>&, const string&, int);
PbDeviceTypesInfo *GetDeviceTypesInfo(PbResult&, const PbCommand&);
PbServerInfo *GetServerInfo(PbResult&, const vector<Device *>&, const set<int>&, const string&, const string&);
PbNetworkInterfacesInfo *GetNetworkInterfacesInfo(PbResult&);
private:
DeviceFactory device_factory;
vector<string> log_levels;
PbDeviceProperties *GetDeviceProperties(const Device *);
void GetDevice(const Device *, PbDevice *, const string&);
void GetAllDeviceTypeProperties(PbDeviceTypesInfo&);
void GetDeviceTypeProperties(PbDeviceTypesInfo&, PbDeviceType);
void GetAvailableImages(PbResult& result, PbServerInfo&, const string&);
void GetImageFile(PbImageFile *, const string&, const string&);
void GetLogLevels(PbServerInfo&);
};

View File

@ -10,7 +10,6 @@
//--------------------------------------------------------------------------- //---------------------------------------------------------------------------
#include "rascsi.h" #include "rascsi.h"
#include "os.h" #include "os.h"
#include "filepath.h" #include "filepath.h"
#include "fileio.h" #include "fileio.h"
@ -24,6 +23,7 @@
#include "devices/file_support.h" #include "devices/file_support.h"
#include "gpiobus.h" #include "gpiobus.h"
#include "exceptions.h" #include "exceptions.h"
#include "protobuf_response_helper.h"
#include "protobuf_util.h" #include "protobuf_util.h"
#include "rascsi_version.h" #include "rascsi_version.h"
#include "rasutil.h" #include "rasutil.h"
@ -69,11 +69,11 @@ int monsocket; // Monitor Socket
pthread_t monthread; // Monitor Thread pthread_t monthread; // Monitor Thread
pthread_mutex_t ctrl_mutex; // Semaphore for the ctrl array pthread_mutex_t ctrl_mutex; // Semaphore for the ctrl array
static void *MonThread(void *param); static void *MonThread(void *param);
vector<string> log_levels;
string current_log_level; // Some versions of spdlog do not support get_log_level() string current_log_level; // Some versions of spdlog do not support get_log_level()
string default_image_folder; string default_image_folder;
set<int> reserved_ids; set<int> reserved_ids;
DeviceFactory& device_factory = DeviceFactory::instance(); DeviceFactory& device_factory = DeviceFactory::instance();
ProtobufResponseHandler& response_helper = ProtobufResponseHandler::instance();
//--------------------------------------------------------------------------- //---------------------------------------------------------------------------
// //
@ -255,21 +255,6 @@ void Reset()
bus->Reset(); bus->Reset();
} }
void GetImageFile(PbImageFile *image_file, const string& filename)
{
image_file->set_name(filename);
if (!filename.empty()) {
string f = filename[0] == '/' ? filename : default_image_folder + "/" + filename;
image_file->set_read_only(access(f.c_str(), W_OK));
struct stat st;
if (!stat(f.c_str(), &st)) {
image_file->set_size(st.st_size);
}
}
}
//--------------------------------------------------------------------------- //---------------------------------------------------------------------------
// //
// Controller Mapping // Controller Mapping
@ -466,241 +451,6 @@ void LogDevices(const string& devices)
} }
} }
void GetLogLevels(PbServerInfo& server_info)
{
for (const auto& log_level : log_levels) {
server_info.add_log_levels(log_level);
}
}
PbDeviceProperties *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);
}
for (const auto& capacity : device_factory.GetCapacities(t)) {
properties->add_capacities(capacity);
}
return properties;
}
void 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 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 GetAvailableImages(PbImageFilesInfo& image_files_info)
{
image_files_info.set_default_image_folder(default_image_folder);
// filesystem::directory_iterator cannot be used because libstdc++ 8.3.0 does not support big files
DIR *d = opendir(default_image_folder.c_str());
if (d) {
struct dirent *dir;
while ((dir = readdir(d))) {
if (dir->d_type == DT_REG || dir->d_type == DT_LNK || dir->d_type == DT_BLK) {
string filename = default_image_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,
default_image_folder.c_str());
continue;
}
if (st.st_size % 512) {
LOGTRACE("Size of file '%s' in image folder '%s' is not a multiple of 512", dir->d_name,
default_image_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,
default_image_folder.c_str());
continue;
}
GetImageFile(image_files_info.add_image_files(), dir->d_name);
}
}
closedir(d);
}
}
void GetAvailableImages(PbServerInfo& server_info)
{
PbImageFilesInfo *image_files_info = new PbImageFilesInfo();
server_info.set_allocated_image_files_info(image_files_info);
image_files_info->set_default_image_folder(default_image_folder);
GetAvailableImages(*image_files_info);
}
void GetNetworkInterfacesInfo(PbNetworkInterfacesInfo& network_interfaces_info)
{
for (const auto& network_interface : device_factory.GetNetworkInterfaces()) {
network_interfaces_info.add_name(network_interface);
}
}
void 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);
}
}
void GetDevices(PbServerInfo& serverInfo)
{
for (const Device *device : devices) {
// Skip if unit does not exist or is not assigned
if (device) {
PbDevice *pb_device = serverInfo.mutable_devices()->add_devices();
GetDevice(device, pb_device);
}
}
}
void GetDevicesInfo(const PbCommand& command, PbResult& result)
{
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() * UnitNum + 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;
}
}
}
PbDevices *pb_devices = new PbDevices();
result.set_allocated_device_info(pb_devices);
for (const auto& id_set : id_sets) {
Device *device = devices[id_set.first * UnitNum + id_set.second];
GetDevice(device, pb_devices->add_devices());
}
}
void GetDeviceTypesInfo(const PbCommand& command, PbResult& result)
{
PbDeviceTypesInfo *device_types_info = new PbDeviceTypesInfo();
GetAllDeviceTypeProperties(*device_types_info);
result.set_allocated_device_types_info(device_types_info);
}
void GetServerInfo(PbResult& result)
{
PbServerInfo *server_info = new PbServerInfo();
server_info->set_major_version(rascsi_major_version);
server_info->set_minor_version(rascsi_minor_version);
server_info->set_patch_version(rascsi_patch_version);
GetLogLevels(*server_info);
server_info->set_current_log_level(current_log_level);
GetAllDeviceTypeProperties(*server_info->mutable_device_types_info());
GetAvailableImages(*server_info);
PbNetworkInterfacesInfo * network_interfaces_info = new PbNetworkInterfacesInfo();
server_info->set_allocated_network_interfaces_info(network_interfaces_info);
GetNetworkInterfacesInfo(*network_interfaces_info);
GetDevices(*server_info);
for (int id : reserved_ids) {
server_info->add_reserved_ids(id);
}
result.set_allocated_server_info(server_info);
}
bool SetDefaultImageFolder(const string& f) bool SetDefaultImageFolder(const string& f)
{ {
string folder = f; string folder = f;
@ -777,10 +527,18 @@ string SetReservedIds(const string& ids)
return ""; return "";
} }
bool IsValidFilename(const string& filename) bool IsValidSrcFilename(const string& filename)
{ {
// Source file must exist and must be a regular file or a symlink
struct stat st; struct stat st;
return stat(filename.c_str(), &st) || !S_ISREG(st.st_mode); return !stat(filename.c_str(), &st) && (S_ISREG(st.st_mode) || S_ISLNK(st.st_mode));
}
bool IsValidDstFilename(const string& filename)
{
// Destination file must not yet exist
struct stat st;
return stat(filename.c_str(), &st);
} }
bool CreateImage(int fd, const PbCommand& command) bool CreateImage(int fd, const PbCommand& command)
@ -789,17 +547,20 @@ bool CreateImage(int fd, const PbCommand& command)
if (filename.empty()) { if (filename.empty()) {
return ReturnStatus(fd, false, "Can't create image file: Missing image filename"); return ReturnStatus(fd, false, "Can't create image file: Missing image filename");
} }
if (filename.find('/') != string::npos) {
if (!IsValidFilename(filename)) { return ReturnStatus(fd, false, "Can't create image file '" + filename + "': Filename must not contain a path");
return ReturnStatus(fd, false, "Can't create image file: '" + filename + "': Invalid filename"); }
filename = default_image_folder + "/" + filename;
if (!IsValidDstFilename(filename)) {
return ReturnStatus(fd, false, "Can't create image file: '" + filename + "': File already exists");
} }
string size = GetParam(command, "size"); const string size = GetParam(command, "size");
if (size.empty()) { if (size.empty()) {
return ReturnStatus(fd, false, "Can't create image file '" + filename + "': Missing image size"); return ReturnStatus(fd, false, "Can't create image file '" + filename + "': Missing image size");
} }
string permission = GetParam(command, "read_only"); const string permission = GetParam(command, "read_only");
if (permission.empty()) { if (permission.empty()) {
return ReturnStatus(fd, false, "Can't create image file'" + filename + "': Missing read-only flag"); return ReturnStatus(fd, false, "Can't create image file'" + filename + "': Missing read-only flag");
} }
@ -811,12 +572,6 @@ bool CreateImage(int fd, const PbCommand& command)
int permissions = !strcasecmp(permission.c_str(), "true") ? int permissions = !strcasecmp(permission.c_str(), "true") ?
S_IRUSR | S_IRGRP | S_IROTH : S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH; S_IRUSR | S_IRGRP | S_IROTH : S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH;
if (filename.find('/') != string::npos) {
return ReturnStatus(fd, false, "Can't create image file '" + filename + "': Filename must not contain a path");
}
filename = default_image_folder + "/" + filename;
off_t len; off_t len;
try { try {
len = stoul(size); len = stoul(size);
@ -866,8 +621,8 @@ bool DeleteImage(int fd, const PbCommand& command)
return ReturnStatus(fd, false, "Missing image filename"); return ReturnStatus(fd, false, "Missing image filename");
} }
if (!IsValidFilename(filename)) { if (!IsValidDstFilename(filename)) {
return ReturnStatus(fd, false, "Can't delete image file '" + filename + "': Invalid filename"); return ReturnStatus(fd, false, "Can't delete image file '" + filename + "': File already exists");
} }
if (filename.find('/') != string::npos) { if (filename.find('/') != string::npos) {
@ -901,33 +656,24 @@ bool RenameImage(int fd, const PbCommand& command)
if (from.empty()) { if (from.empty()) {
return ReturnStatus(fd, false, "Can't rename image file: Missing source filename"); return ReturnStatus(fd, false, "Can't rename image file: Missing source filename");
} }
if (from.find('/') != string::npos) {
return ReturnStatus(fd, false, "The source filename '" + from + "' must not contain a path");
}
from = default_image_folder + "/" + from;
if (!IsValidSrcFilename(from)) {
return ReturnStatus(fd, false, "Can't rename image file: '" + from + "': Invalid name or type");
}
string to = GetParam(command, "to"); string to = GetParam(command, "to");
if (to.empty()) { if (to.empty()) {
return ReturnStatus(fd, false, "Can't rename image file '" + from + "': Missing destination filename"); return ReturnStatus(fd, false, "Can't rename image file '" + from + "': Missing destination filename");
} }
if (!IsValidFilename(from)) {
return ReturnStatus(fd, false, "Can't rename image file: '" + from + "': Invalid filename");
}
if (!IsValidFilename(to)) {
return ReturnStatus(fd, false, "Can't rename image file '" + from + "' to '" + to + "': Invalid filename");
}
if (from.find('/') != string::npos) {
return ReturnStatus(fd, false, "The source filename '" + from + "' must not contain a path");
}
if (to.find('/') != string::npos) { if (to.find('/') != string::npos) {
return ReturnStatus(fd, false, "The destination filename '" + to + "' must not contain a path"); return ReturnStatus(fd, false, "The destination filename '" + to + "' must not contain a path");
} }
from = default_image_folder + "/" + from;
to = default_image_folder + "/" + to; to = default_image_folder + "/" + to;
if (!IsValidDstFilename(to)) {
struct stat st; return ReturnStatus(fd, false, "Can't rename image file '" + from + "' to '" + to + "': File already exists");
if (!stat(to.c_str(), &st)) {
return ReturnStatus(fd, false, "Image file '" + to + "' already exists");
} }
if (rename(from.c_str(), to.c_str())) { if (rename(from.c_str(), to.c_str())) {
@ -945,42 +691,33 @@ bool CopyImage(int fd, const PbCommand& command)
if (from.empty()) { if (from.empty()) {
return ReturnStatus(fd, false, "Can't copy image file: Missing source filename"); return ReturnStatus(fd, false, "Can't copy image file: Missing source filename");
} }
if (from.find('/') != string::npos) {
return ReturnStatus(fd, false, "The source filename '" + from + "' must not contain a path");
}
from = default_image_folder + "/" + from;
if (!IsValidSrcFilename(from)) {
return ReturnStatus(fd, false, "Can't copy image file: '" + from + "': Invalid name or type");
}
string to = GetParam(command, "to"); string to = GetParam(command, "to");
if (to.empty()) { if (to.empty()) {
return ReturnStatus(fd, false, "Can't copy image file '" + from + "': Missing destination filename"); return ReturnStatus(fd, false, "Can't copy image file '" + from + "': Missing destination filename");
} }
if (!IsValidFilename(from)) {
return ReturnStatus(fd, false, "Can't copy image file: '" + from + "': Invalid filename");
}
if (!IsValidFilename(to)) {
return ReturnStatus(fd, false, "Can't copy image file '" + from + "' to '" + to + "': Invalid filename");
}
if (from.find('/') != string::npos) {
return ReturnStatus(fd, false, "The source filename '" + from + "' must not contain a path");
}
if (to.find('/') != string::npos) { if (to.find('/') != string::npos) {
return ReturnStatus(fd, false, "The destination filename '" + to + "' must not contain a path"); return ReturnStatus(fd, false, "The destination filename '" + to + "' must not contain a path");
} }
from = default_image_folder + "/" + from;
to = default_image_folder + "/" + to; to = default_image_folder + "/" + to;
if (!IsValidDstFilename(to)) {
struct stat st_dst; return ReturnStatus(fd, false, "Can't copy image file '" + from + "' to '" + to + "': File already exists");
if (!stat(to.c_str(), &st_dst)) {
return ReturnStatus(fd, false, "Image file '" + to + "' already exists");
} }
struct stat st_src; struct stat st;
if (lstat(from.c_str(), &st_src)) { if (lstat(from.c_str(), &st)) {
return ReturnStatus(fd, false, "Can't access source image file '" + from + "': " + string(strerror(errno))); return ReturnStatus(fd, false, "Can't access source image file '" + from + "': " + string(strerror(errno)));
} }
// Symbolic links need a special handling // Symbolic links need a special handling
if ((st_src.st_mode & S_IFMT) == S_IFLNK) { if ((st.st_mode & S_IFMT) == S_IFLNK) {
if (symlink(filesystem::read_symlink(from).c_str(), to.c_str())) { if (symlink(filesystem::read_symlink(from).c_str(), to.c_str())) {
return ReturnStatus(fd, false, "Can't copy symlink '" + from + "': " + string(strerror(errno))); return ReturnStatus(fd, false, "Can't copy symlink '" + from + "': " + string(strerror(errno)));
} }
@ -995,14 +732,14 @@ bool CopyImage(int fd, const PbCommand& command)
return ReturnStatus(fd, false, "Can't open source image file '" + from + "': " + string(strerror(errno))); return ReturnStatus(fd, false, "Can't open source image file '" + from + "': " + string(strerror(errno)));
} }
int fd_dst = open(to.c_str(), O_WRONLY | O_CREAT, st_src.st_mode); int fd_dst = open(to.c_str(), O_WRONLY | O_CREAT, st.st_mode);
if (fd_dst == -1) { if (fd_dst == -1) {
close(fd_src); close(fd_src);
return ReturnStatus(fd, false, "Can't open destination image file '" + to + "': " + string(strerror(errno))); return ReturnStatus(fd, false, "Can't open destination image file '" + to + "': " + string(strerror(errno)));
} }
if (sendfile(fd_dst, fd_src, 0, st_src.st_size) == -1) { if (sendfile(fd_dst, fd_src, 0, st.st_size) == -1) {
close(fd_dst); close(fd_dst);
close(fd_src); close(fd_src);
@ -1023,16 +760,13 @@ bool SetImagePermissions(int fd, const PbCommand& command)
if (filename.empty()) { if (filename.empty()) {
return ReturnStatus(fd, false, "Missing image filename"); return ReturnStatus(fd, false, "Missing image filename");
} }
if (!IsValidFilename(filename)) {
return ReturnStatus(fd, false, "Can't modify image file '" + filename + "': Invalid filename");
}
if (filename.find('/') != string::npos) { if (filename.find('/') != string::npos) {
return ReturnStatus(fd, false, "The image filename '" + filename + "' must not contain a path"); return ReturnStatus(fd, false, "The image filename '" + filename + "' must not contain a path");
} }
filename = default_image_folder + "/" + filename; filename = default_image_folder + "/" + filename;
if (!IsValidSrcFilename(filename)) {
return ReturnStatus(fd, false, "Can't modify image file '" + filename + "': Invalid name or type");
}
bool protect = command.operation() == PROTECT_IMAGE; bool protect = command.operation() == PROTECT_IMAGE;
@ -1081,17 +815,12 @@ bool Attach(int fd, const PbDeviceDefinition& pb_device, Device *map[], bool dry
} }
string filename = GetParam(pb_device, "file"); string filename = GetParam(pb_device, "file");
string ext;
size_t separator = filename.rfind('.');
if (separator != string::npos) {
ext = filename.substr(separator + 1);
}
// Create a new device, based upon the provided type or filename extension // Create a new device, based on the provided type or filename
Device *device = device_factory.CreateDevice(type, filename, ext); Device *device = device_factory.CreateDevice(type, filename);
if (!device) { if (!device) {
if (type == UNDEFINED) { if (type == UNDEFINED) {
return ReturnStatus(fd, false, "No device type provided for unknown file extension '" + ext + "'"); return ReturnStatus(fd, false, "Device type required for unknown extension of file '" + filename + "'");
} }
else { else {
return ReturnStatus(fd, false, "Unknown device type " + PbDeviceType_Name(type)); return ReturnStatus(fd, false, "Unknown device type " + PbDeviceType_Name(type));
@ -1708,7 +1437,7 @@ bool ParseArgument(int argc, char* argv[], int& port)
// Display and log the device list // Display and log the device list
PbServerInfo server_info; PbServerInfo server_info;
GetDevices(server_info); response_helper.GetDevices(server_info, devices, default_image_folder);
const list<PbDevice>& devices = { server_info.devices().devices().begin(), server_info.devices().devices().end() }; const list<PbDevice>& devices = { server_info.devices().devices().begin(), server_info.devices().devices().end() };
const string device_list = ListDevices(devices); const string device_list = ListDevices(devices);
LogDevices(device_list); LogDevices(device_list);
@ -1814,13 +1543,12 @@ static void *MonThread(void *param)
LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str()); LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str());
PbResult result; PbResult result;
result.set_status(true); response_helper.GetDevicesInfo(result, command, devices, default_image_folder, UnitNum);
GetDevicesInfo(command, result);
SerializeMessage(fd, result); SerializeMessage(fd, result);
const list<PbDevice>& devices ={ result.device_info().devices().begin(), result.device_info().devices().end() };
// For backwards compatibility: Log device list if information on all devices was requested. // For backwards compatibility: Log device list if information on all devices was requested.
if (command.devices_size() == 0) { if (!command.devices_size()) {
const list<PbDevice>& devices = { result.device_info().devices().begin(), result.device_info().devices().end() };
LogDevices(ListDevices(devices)); LogDevices(ListDevices(devices));
} }
break; break;
@ -1830,8 +1558,7 @@ static void *MonThread(void *param)
LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str()); LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str());
PbResult result; PbResult result;
result.set_status(true); result.set_allocated_device_types_info(response_helper.GetDeviceTypesInfo(result, command));
GetDeviceTypesInfo(command, result);
SerializeMessage(fd, result); SerializeMessage(fd, result);
break; break;
} }
@ -1841,8 +1568,8 @@ static void *MonThread(void *param)
LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str()); LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str());
PbResult result; PbResult result;
result.set_status(true); result.set_allocated_server_info(response_helper.GetServerInfo(
GetServerInfo(result); result, devices, reserved_ids, default_image_folder, current_log_level));
SerializeMessage(fd, result); SerializeMessage(fd, result);
break; break;
} }
@ -1850,11 +1577,8 @@ static void *MonThread(void *param)
case IMAGE_FILES_INFO: { case IMAGE_FILES_INFO: {
LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str()); LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str());
PbImageFilesInfo *image_files_info = new PbImageFilesInfo();
GetAvailableImages(*image_files_info);
PbResult result; PbResult result;
result.set_status(true); result.set_allocated_image_files_info(response_helper.GetAvailableImages(result, default_image_folder));
result.set_allocated_image_files_info(image_files_info);
SerializeMessage(fd, result); SerializeMessage(fd, result);
break; break;
} }
@ -1862,11 +1586,8 @@ static void *MonThread(void *param)
case NETWORK_INTERFACES_INFO: { case NETWORK_INTERFACES_INFO: {
LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str()); LOGTRACE("Received %s command", PbOperation_Name(command.operation()).c_str());
PbNetworkInterfacesInfo *network_interfaces_info = new PbNetworkInterfacesInfo();
GetNetworkInterfacesInfo(*network_interfaces_info);
PbResult result; PbResult result;
result.set_status(true); result.set_allocated_network_interfaces_info(response_helper.GetNetworkInterfacesInfo(result));
result.set_allocated_network_interfaces_info(network_interfaces_info);
SerializeMessage(fd, result); SerializeMessage(fd, result);
break; break;
} }
@ -1923,13 +1644,6 @@ int main(int argc, char* argv[])
} }
} }
log_levels.push_back("trace");
log_levels.push_back("debug");
log_levels.push_back("info");
log_levels.push_back("warn");
log_levels.push_back("err");
log_levels.push_back("critical");
log_levels.push_back("off");
SetLogLevel("info"); SetLogLevel("info");
// Create a thread-safe stdout logger to process the log messages // Create a thread-safe stdout logger to process the log messages

View File

@ -181,9 +181,11 @@ message PbDeviceTypesInfo {
// The image file data // The image file data
message PbImageFile { message PbImageFile {
string name = 1; string name = 1;
bool read_only = 2; // The assumed device type, based on the filename extension
PbDeviceType type = 2;
bool read_only = 3;
// The file size in bytes, 0 for block devices // The file size in bytes, 0 for block devices
int64 size = 3; int64 size = 4;
} }
// The default image folder and the image files it contains // The default image folder and the image files it contains

View File

@ -265,9 +265,12 @@ void DisplayImageFiles(const list<PbImageFile> image_files, const string& defaul
cout << "Available image files:" << endl; cout << "Available image files:" << endl;
for (const auto& file : files) { for (const auto& file : files) {
cout << " " << file.name() << " (" << file.size() << " bytes)"; cout << " " << file.name() << " " << file.size() << " bytes";
if (file.read_only()) { if (file.read_only()) {
cout << ", read-only"; cout << " read-only";
}
if (file.type() != UNDEFINED) {
cout << " " << PbDeviceType_Name(file.type());
} }
cout << endl; cout << endl;
} }