mirror of
https://github.com/Telecominfraproject/wlan-cloud-owprov.git
synced 2025-11-01 02:58:00 +00:00
433 lines
19 KiB
C++
433 lines
19 KiB
C++
//
|
|
// License type: BSD 3-Clause License
|
|
// License copy: https://github.com/Telecominfraproject/wlan-cloud-ucentralgw/blob/master/LICENSE
|
|
//
|
|
// Created by Stephane Bourque on 2021-03-04.
|
|
// Arilia Wireless Inc.
|
|
//
|
|
|
|
#include "RESTAPI_inventory_handler.h"
|
|
|
|
#include "StorageService.h"
|
|
#include "APConfig.h"
|
|
#include "AutoDiscovery.h"
|
|
#include "sdks/SDK_gw.h"
|
|
#include "sdks/SDK_sec.h"
|
|
#include "RESTAPI/RESTAPI_db_helpers.h"
|
|
#include "SerialNumberCache.h"
|
|
#include "DeviceTypeCache.h"
|
|
|
|
#define __DBG__ std::cout << __LINE__ << std::endl;
|
|
|
|
namespace OpenWifi{
|
|
|
|
void GetRejectedLines(const Poco::JSON::Object::Ptr &Response, Types::StringVec & Warnings) {
|
|
try {
|
|
if(Response->has("results")) {
|
|
auto Results = Response->get("results").extract<Poco::JSON::Object::Ptr>();
|
|
auto Status = Results->get("status").extract<Poco::JSON::Object::Ptr>();
|
|
auto Rejected = Status->getArray("rejected");
|
|
std::transform(Rejected->begin(),Rejected->end(),std::back_inserter(Warnings), [](auto i) -> auto { return i.toString(); });
|
|
// for(const auto &i:*Rejected)
|
|
// Warnings.push_back(i.toString());
|
|
}
|
|
} catch (...) {
|
|
}
|
|
}
|
|
|
|
void RESTAPI_inventory_handler::DoGet() {
|
|
|
|
ProvObjects::InventoryTag Existing;
|
|
__DBG__
|
|
std::string SerialNumber = GetBinding(RESTAPI::Protocol::SERIALNUMBER,"");
|
|
__DBG__
|
|
Logger().debug(Poco::format("%s: Retrieving inventory information.",SerialNumber));
|
|
__DBG__
|
|
if(SerialNumber.empty() || !DB_.GetRecord(RESTAPI::Protocol::SERIALNUMBER,SerialNumber,Existing)) {
|
|
return NotFound();
|
|
}
|
|
__DBG__
|
|
Logger().debug(Poco::format("%s,%s: Retrieving inventory information.", Existing.serialNumber, Existing.info.id ));
|
|
|
|
__DBG__
|
|
Poco::JSON::Object Answer;
|
|
std::string Arg;
|
|
__DBG__
|
|
if(HasParameter("config",Arg) && Arg=="true") {
|
|
__DBG__
|
|
bool Explain = (HasParameter("explain",Arg) && Arg == "true");
|
|
__DBG__
|
|
APConfig Device(SerialNumber,Existing.deviceType,Logger(), Explain);
|
|
|
|
__DBG__
|
|
Poco::JSON::Object::Ptr Configuration;
|
|
if(Device.Get(Configuration)) {
|
|
__DBG__
|
|
Answer.set("config", Configuration);
|
|
if(Explain)
|
|
__DBG__
|
|
Answer.set("explanation", Device.Explanation());
|
|
} else {
|
|
__DBG__
|
|
Answer.set("config","none");
|
|
__DBG__
|
|
}
|
|
__DBG__
|
|
return ReturnObject(Answer);
|
|
} else if(HasParameter("firmwareOptions", Arg) && Arg=="true") {
|
|
ProvObjects::FIRMWARE_UPGRADE_RULES Rules;
|
|
|
|
StorageService()->InventoryDB().FindFirmwareOptions(SerialNumber,Rules);
|
|
|
|
if(Rules == ProvObjects::dont_upgrade) {
|
|
Answer.set("firmwareUpgrade","no");
|
|
} else {
|
|
Answer.set("firmwareUpgrade","yes");
|
|
if(Rules == ProvObjects::upgrade_release_only)
|
|
Answer.set("firmwareRCOnly", Rules == ProvObjects::upgrade_release_only );
|
|
}
|
|
return ReturnObject(Answer);
|
|
} else if(HasParameter("applyConfiguration",Arg) && Arg=="true") {
|
|
Logger().debug(Poco::format("%s: Retrieving configuration.",Existing.serialNumber));
|
|
auto Device = std::make_shared<APConfig>(SerialNumber, Existing.deviceType, Logger(), false);
|
|
auto Configuration = Poco::makeShared<Poco::JSON::Object>();
|
|
Poco::JSON::Object ErrorsObj, WarningsObj;
|
|
ProvObjects::InventoryConfigApplyResult Results;
|
|
Logger().debug(Poco::format("%s: Computing configuration.",Existing.serialNumber));
|
|
if (Device->Get(Configuration)) {
|
|
std::ostringstream OS;
|
|
Configuration->stringify(OS);
|
|
Results.appliedConfiguration = OS.str();
|
|
Poco::JSON::Object::Ptr Response;
|
|
Logger().debug(Poco::format("%s: Sending configuration push.",Existing.serialNumber));
|
|
if (SDK::GW::Device::Configure(this, SerialNumber, Configuration, Response)) {
|
|
Logger().debug(Poco::format("%s: Sending configuration pushed.",Existing.serialNumber));
|
|
GetRejectedLines(Response, Results.warnings);
|
|
Results.errorCode = 0;
|
|
} else {
|
|
Logger().debug(Poco::format("%s: Sending configuration failed.",Existing.serialNumber));
|
|
Results.errorCode = 1;
|
|
}
|
|
} else {
|
|
Logger().debug(Poco::format("%s: Configuration is bad.",Existing.serialNumber));
|
|
Results.errorCode = 1;
|
|
}
|
|
Results.to_json(Answer);
|
|
return ReturnObject(Answer);
|
|
} else if(QB_.AdditionalInfo) {
|
|
AddExtendedInfo(Existing,Answer);
|
|
}
|
|
Existing.to_json(Answer);
|
|
ReturnObject(Answer);
|
|
}
|
|
|
|
void RESTAPI_inventory_handler::DoDelete() {
|
|
ProvObjects::InventoryTag Existing;
|
|
std::string SerialNumber = GetBinding(RESTAPI::Protocol::SERIALNUMBER,"");
|
|
if(SerialNumber.empty() || !DB_.GetRecord(RESTAPI::Protocol::SERIALNUMBER,SerialNumber,Existing)) {
|
|
return NotFound();
|
|
}
|
|
|
|
MoveUsage(StorageService()->PolicyDB(),DB_,Existing.managementPolicy,"",Existing.info.id);
|
|
RemoveMembership(StorageService()->VenueDB(),&ProvObjects::Venue::configurations,Existing.venue,Existing.info.id);
|
|
RemoveMembership(StorageService()->EntityDB(),&ProvObjects::Entity::configurations,Existing.entity,Existing.info.id);
|
|
MoveUsage(StorageService()->LocationDB(),DB_,Existing.location,"",Existing.info.id);
|
|
MoveUsage(StorageService()->ContactDB(),DB_,Existing.contact,"",Existing.info.id);
|
|
|
|
if(!Existing.deviceConfiguration.empty()) {
|
|
ProvObjects::DeviceConfiguration DC;
|
|
if(StorageService()->ConfigurationDB().GetRecord("id", Existing.deviceConfiguration, DC)) {
|
|
if(DC.subscriberOnly)
|
|
StorageService()->ConfigurationDB().DeleteRecord("id", Existing.deviceConfiguration);
|
|
else
|
|
StorageService()->ConfigurationDB().DeleteInUse("id", Existing.deviceConfiguration, DB_.Prefix(),
|
|
Existing.info.id);
|
|
}
|
|
}
|
|
|
|
MoveUsage(StorageService()->PolicyDB(),DB_,Existing.managementPolicy,"",Existing.info.id);
|
|
MoveUsage(StorageService()->LocationDB(),DB_,Existing.location,"",Existing.info.id);
|
|
MoveUsage(StorageService()->ContactDB(),DB_,Existing.contact,"",Existing.info.id);
|
|
MoveUsage(StorageService()->ConfigurationDB(),DB_,Existing.deviceConfiguration,"",Existing.info.id);
|
|
ManageMembership(StorageService()->EntityDB(),&ProvObjects::Entity::devices,Existing.entity,"",Existing.info.id);
|
|
ManageMembership(StorageService()->VenueDB(),&ProvObjects::Venue::devices,Existing.venue,"",Existing.info.id);
|
|
DB_.DeleteRecord("id", Existing.info.id);
|
|
SerialNumberCache()->DeleteSerialNumber(SerialNumber);
|
|
return OK();
|
|
}
|
|
|
|
|
|
void RESTAPI_inventory_handler::DoPost() {
|
|
std::string SerialNumber = GetBinding(RESTAPI::Protocol::SERIALNUMBER,"");
|
|
if(SerialNumber.empty()) {
|
|
return BadRequest(RESTAPI::Errors::MissingSerialNumber);
|
|
}
|
|
|
|
__DBG__
|
|
if(!NormalizeMac(SerialNumber)) {
|
|
return BadRequest(RESTAPI::Errors::InvalidSerialNumber);
|
|
}
|
|
__DBG__
|
|
|
|
if(DB_.Exists(RESTAPI::Protocol::SERIALNUMBER,SerialNumber)) {
|
|
return BadRequest(RESTAPI::Errors::SerialNumberExists + " (" + SerialNumber + ")");
|
|
}
|
|
__DBG__
|
|
|
|
auto Obj = ParseStream();
|
|
__DBG__
|
|
ProvObjects::InventoryTag NewObject;
|
|
__DBG__
|
|
if (!NewObject.from_json(Obj)) {
|
|
return BadRequest(RESTAPI::Errors::InvalidJSONDocument);
|
|
}
|
|
__DBG__
|
|
|
|
if(!Provisioning::DeviceClass::Validate(NewObject.devClass.c_str())) {
|
|
return BadRequest(RESTAPI::Errors::InvalidDeviceClass);
|
|
}
|
|
__DBG__
|
|
|
|
if(NewObject.devClass.empty()) {
|
|
NewObject.devClass = Provisioning::DeviceClass::ANY;
|
|
}
|
|
__DBG__
|
|
|
|
if(!ProvObjects::CreateObjectInfo(Obj, UserInfo_.userinfo, NewObject.info)) {
|
|
return BadRequest( RESTAPI::Errors::NameMustBeSet);
|
|
}
|
|
__DBG__
|
|
|
|
if(NewObject.deviceType.empty() || !DeviceTypeCache()->IsAcceptableDeviceType(NewObject.deviceType)) {
|
|
return BadRequest(RESTAPI::Errors::InvalidDeviceTypes);
|
|
}
|
|
__DBG__
|
|
|
|
if(OpenWifi::EntityDB::IsRoot(NewObject.entity) || (!NewObject.entity.empty() && !StorageService()->EntityDB().Exists("id",NewObject.entity))) {
|
|
return BadRequest(RESTAPI::Errors::ValidNonRootUUID);
|
|
}
|
|
__DBG__
|
|
|
|
if(!NewObject.venue.empty() && !StorageService()->VenueDB().Exists("id",NewObject.venue)) {
|
|
return BadRequest(RESTAPI::Errors::VenueMustExist);
|
|
}
|
|
__DBG__
|
|
|
|
if(!NewObject.venue.empty() && !NewObject.entity.empty()) {
|
|
return BadRequest(RESTAPI::Errors::NotBoth);
|
|
}
|
|
__DBG__
|
|
|
|
if(!NewObject.location.empty() && !StorageService()->LocationDB().Exists("id",NewObject.location)) {
|
|
return BadRequest(RESTAPI::Errors::LocationMustExist);
|
|
}
|
|
__DBG__
|
|
|
|
if(!NewObject.contact.empty() && !StorageService()->ContactDB().Exists("id",NewObject.contact)) {
|
|
return BadRequest(RESTAPI::Errors::ContactMustExist);
|
|
}
|
|
|
|
__DBG__
|
|
if(!NewObject.deviceConfiguration.empty() && !StorageService()->ConfigurationDB().Exists("id",NewObject.deviceConfiguration)) {
|
|
return BadRequest(RESTAPI::Errors::ConfigurationMustExist);
|
|
}
|
|
|
|
__DBG__
|
|
if(!NewObject.managementPolicy.empty() && !StorageService()->PolicyDB().Exists("id",NewObject.managementPolicy)) {
|
|
return BadRequest(RESTAPI::Errors::UnknownManagementPolicyUUID);
|
|
}
|
|
|
|
__DBG__
|
|
/*
|
|
if(!NewObject.venue.empty()) {
|
|
nlohmann::json state;
|
|
state["method"] = "assignedTo";
|
|
state["venue"] = NewObject.venue;
|
|
state["date"] = std::time(nullptr);
|
|
NewObject.state = to_string(state);
|
|
} else if (!NewObject.entity.empty()) {
|
|
nlohmann::json state;
|
|
state["method"] = "assignedTo";
|
|
state["entity"] = NewObject.entity;
|
|
state["date"] = std::time(nullptr);
|
|
NewObject.state = to_string(state);
|
|
} else {
|
|
nlohmann::json state;
|
|
state["method"] = "created";
|
|
state["date"] = std::time(nullptr);
|
|
NewObject.state = to_string(state);
|
|
}
|
|
*/
|
|
|
|
__DBG__
|
|
if(DB_.CreateRecord(NewObject)) {
|
|
__DBG__
|
|
SerialNumberCache()->AddSerialNumber(SerialNumber,NewObject.deviceType);
|
|
|
|
__DBG__
|
|
MoveUsage(StorageService()->PolicyDB(),DB_,NewObject.managementPolicy,"",NewObject.info.id);
|
|
__DBG__
|
|
MoveUsage(StorageService()->LocationDB(),DB_,NewObject.location,"",NewObject.info.id);
|
|
__DBG__
|
|
MoveUsage(StorageService()->ContactDB(),DB_,NewObject.contact,"",NewObject.info.id);
|
|
__DBG__
|
|
MoveUsage(StorageService()->ConfigurationDB(),DB_,NewObject.deviceConfiguration,"",NewObject.info.id);
|
|
__DBG__
|
|
ManageMembership(StorageService()->EntityDB(),&ProvObjects::Entity::devices,NewObject.entity,"",NewObject.info.id);
|
|
__DBG__
|
|
ManageMembership(StorageService()->VenueDB(),&ProvObjects::Venue::devices,NewObject.venue,"",NewObject.info.id);
|
|
__DBG__
|
|
|
|
ProvObjects::InventoryTag NewTag;
|
|
__DBG__
|
|
DB_.GetRecord("id",NewObject.info.id,NewTag);
|
|
__DBG__
|
|
Poco::JSON::Object Answer;
|
|
NewTag.to_json(Answer);
|
|
__DBG__
|
|
return ReturnObject(Answer);
|
|
}
|
|
InternalError(RESTAPI::Errors::RecordNotCreated);
|
|
}
|
|
|
|
|
|
#define __DBG__ std::cout << __LINE__ << std::endl;
|
|
|
|
void RESTAPI_inventory_handler::DoPut() {
|
|
|
|
std::string SerialNumber = GetBinding(RESTAPI::Protocol::SERIALNUMBER,"");
|
|
if(SerialNumber.empty() || !Utils::ValidSerialNumber(SerialNumber)) {
|
|
return BadRequest(RESTAPI::Errors::MissingOrInvalidParameters);
|
|
}
|
|
|
|
ProvObjects::InventoryTag Existing;
|
|
if(SerialNumber.empty() || !DB_.GetRecord(RESTAPI::Protocol::SERIALNUMBER,SerialNumber,Existing)) {
|
|
return NotFound();
|
|
}
|
|
|
|
auto RemoveSubscriber = GetParameter("removeSubscriber","");
|
|
if(!RemoveSubscriber.empty()) {
|
|
if(Existing.subscriber == RemoveSubscriber) {
|
|
Logger().information(Poco::format("%s: removing subscriber (%s)", SerialNumber, RemoveSubscriber));
|
|
ProvObjects::DeviceConfiguration DC;
|
|
if(StorageService()->ConfigurationDB().GetRecord("id",Existing.deviceConfiguration,DC)) {
|
|
Logger().information(Poco::format("%s: removing configuration for subscriber (%s)", SerialNumber, RemoveSubscriber));
|
|
if(DC.subscriberOnly) {
|
|
if(!StorageService()->ConfigurationDB().DeleteRecord("id", Existing.deviceConfiguration))
|
|
Logger().debug("Could not delete the subscriber configuration");
|
|
}
|
|
else
|
|
Logger().debug("Configurations is not for a subscriber.");
|
|
Existing.deviceConfiguration = "";
|
|
}
|
|
Existing.subscriber = "";
|
|
Poco::JSON::Object state;
|
|
state.set("date",OpenWifi::Now());
|
|
state.set("method","auto-discovery");
|
|
state.set("last-operation", "returned to inventory");
|
|
std::ostringstream OO;
|
|
state.stringify(OO);
|
|
Existing.state = OO.str();
|
|
StorageService()->InventoryDB().UpdateRecord("id",Existing.info.id,Existing);
|
|
Poco::JSON::Object Answer;
|
|
Existing.to_json(Answer);
|
|
return ReturnObject(Answer);
|
|
} else {
|
|
Logger().information(Poco::format("%s: wrong subscriber (%s)", SerialNumber, RemoveSubscriber));
|
|
}
|
|
return BadRequest(RESTAPI::Errors::MissingOrInvalidParameters);
|
|
}
|
|
|
|
auto RawObject = ParseStream();
|
|
ProvObjects::InventoryTag NewObject;
|
|
if(!NewObject.from_json(RawObject)) {
|
|
return BadRequest(RESTAPI::Errors::InvalidJSONDocument);
|
|
}
|
|
|
|
if(!Provisioning::DeviceClass::Validate(NewObject.devClass.c_str())) {
|
|
return BadRequest(RESTAPI::Errors::InvalidDeviceClass);
|
|
}
|
|
|
|
if(!NewObject.deviceType.empty()) {
|
|
if(!DeviceTypeCache()->IsAcceptableDeviceType(NewObject.deviceType)) {
|
|
return BadRequest(RESTAPI::Errors::InvalidDeviceTypes);
|
|
}
|
|
}
|
|
|
|
if(!UpdateObjectInfo(RawObject, UserInfo_.userinfo, Existing.info)) {
|
|
return BadRequest(RESTAPI::Errors::NameMustBeSet);
|
|
}
|
|
|
|
AssignIfPresent(RawObject, "rrm",Existing.rrm);
|
|
|
|
std::string FromPolicy, ToPolicy;
|
|
if(!CreateMove(RawObject,"managementPolicy",&InventoryDB::RecordName::managementPolicy, Existing, FromPolicy,
|
|
ToPolicy, StorageService()->PolicyDB()))
|
|
return BadRequest(RESTAPI::Errors::EntityMustExist);
|
|
|
|
std::string FromEntity, ToEntity;
|
|
if(!CreateMove(RawObject,"entity",&InventoryDB::RecordName::entity, Existing, FromEntity, ToEntity,
|
|
StorageService()->EntityDB()))
|
|
return BadRequest(RESTAPI::Errors::EntityMustExist);
|
|
|
|
std::string FromVenue, ToVenue;
|
|
if(!CreateMove(RawObject,"venue",&InventoryDB::RecordName::venue, Existing, FromVenue, ToVenue,
|
|
StorageService()->VenueDB()))
|
|
return BadRequest(RESTAPI::Errors::VenueMustExist);
|
|
|
|
std::string FromLocation, ToLocation;
|
|
if(!CreateMove(RawObject,"location",&InventoryDB::RecordName::location, Existing, FromLocation, ToLocation,
|
|
StorageService()->LocationDB()))
|
|
return BadRequest(RESTAPI::Errors::VenueMustExist);
|
|
|
|
std::string FromContact, ToContact;
|
|
if(!CreateMove(RawObject,"contact",&InventoryDB::RecordName::contact, Existing, FromContact, ToContact,
|
|
StorageService()->ContactDB()))
|
|
return BadRequest(RESTAPI::Errors::VenueMustExist);
|
|
|
|
std::string FromConfiguration, ToConfiguration;
|
|
if(!CreateMove(RawObject,"deviceConfiguration",&InventoryDB::RecordName::deviceConfiguration, Existing,
|
|
FromConfiguration, ToConfiguration, StorageService()->ConfigurationDB()))
|
|
return BadRequest(RESTAPI::Errors::ConfigurationMustExist);
|
|
|
|
std::string NewSubScriber;
|
|
if(AssignIfPresent(RawObject, "subscriber", NewSubScriber)) {
|
|
if(!NewSubScriber.empty()) {
|
|
if(NewSubScriber!=Existing.subscriber) {
|
|
SecurityObjects::UserInfo U;
|
|
if(SDK::Sec::Subscriber::Get(this, NewSubScriber, U)) {
|
|
Existing.subscriber = NewSubScriber;
|
|
} else {
|
|
return BadRequest(RESTAPI::Errors::SubscriberMustExist);
|
|
}
|
|
}
|
|
} else {
|
|
Existing.subscriber = "";
|
|
}
|
|
}
|
|
|
|
if( RawObject->has("devClass") && NewObject.devClass!= Existing.devClass) {
|
|
Existing.devClass = NewObject.devClass;
|
|
}
|
|
|
|
if( RawObject->has("state") && NewObject.state!= Existing.state) {
|
|
Existing.state = NewObject.state;
|
|
}
|
|
|
|
if(StorageService()->InventoryDB().UpdateRecord("id", Existing.info.id, Existing)) {
|
|
MoveUsage(StorageService()->PolicyDB(),DB_,FromPolicy,ToPolicy,Existing.info.id);
|
|
MoveUsage(StorageService()->LocationDB(),DB_,FromLocation,ToLocation,Existing.info.id);
|
|
MoveUsage(StorageService()->ContactDB(),DB_,FromContact,ToContact,Existing.info.id);
|
|
MoveUsage(StorageService()->ConfigurationDB(),DB_,FromConfiguration,ToConfiguration,Existing.info.id);
|
|
ManageMembership(StorageService()->EntityDB(),&ProvObjects::Entity::devices,FromEntity,ToEntity,Existing.info.id);
|
|
ManageMembership(StorageService()->VenueDB(),&ProvObjects::Venue::devices,FromVenue,ToVenue,Existing.info.id);
|
|
|
|
ProvObjects::InventoryTag NewObjectCreated;
|
|
DB_.GetRecord("id", Existing.info.id, NewObjectCreated);
|
|
Poco::JSON::Object Answer;
|
|
NewObject.to_json(Answer);
|
|
return ReturnObject(Answer);
|
|
}
|
|
InternalError(RESTAPI::Errors::RecordNotUpdated);
|
|
}
|
|
} |