Mercurial > hg > orthanc-dicomweb
view Plugin/DicomWebServers.cpp @ 307:5a9238e974ea refactoring
first version of the DICOMweb client
author | Sebastien Jodogne <s.jodogne@gmail.com> |
---|---|
date | Tue, 18 Jun 2019 18:38:30 +0200 |
parents | ebd8a1cd97da |
children | 46bf8c8047de |
line wrap: on
line source
/** * Orthanc - A Lightweight, RESTful DICOM Store * Copyright (C) 2012-2016 Sebastien Jodogne, Medical Physics * Department, University Hospital of Liege, Belgium * Copyright (C) 2017-2019 Osimis S.A., Belgium * * This program is free software: you can redistribute it and/or * modify it under the terms of the GNU Affero General Public License * as published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. **/ #include "DicomWebServers.h" #include "Configuration.h" #include <Core/Toolbox.h> #include <boost/algorithm/string/predicate.hpp> namespace OrthancPlugins { void DicomWebServers::Clear() { for (Servers::iterator it = servers_.begin(); it != servers_.end(); ++it) { delete it->second; } } void DicomWebServers::Load(const Json::Value& servers) { boost::mutex::scoped_lock lock(mutex_); Clear(); bool ok = true; try { if (servers.type() != Json::objectValue) { ok = false; } else { Json::Value::Members members = servers.getMemberNames(); for (size_t i = 0; i < members.size(); i++) { std::auto_ptr<Orthanc::WebServiceParameters> parameters (new Orthanc::WebServiceParameters(servers[members[i]])); servers_[members[i]] = parameters.release(); } } } catch (Orthanc::OrthancException& e) { OrthancPlugins::LogError("Exception while parsing the \"DicomWeb.Servers\" section " "of the configuration file: " + std::string(e.What())); throw; } if (!ok) { throw Orthanc::OrthancException( Orthanc::ErrorCode_BadFileFormat, "Cannot parse the \"DicomWeb.Servers\" section of the configuration file"); } } DicomWebServers& DicomWebServers::GetInstance() { static DicomWebServers singleton; return singleton; } Orthanc::WebServiceParameters DicomWebServers::GetServer(const std::string& name) { boost::mutex::scoped_lock lock(mutex_); Servers::const_iterator server = servers_.find(name); if (server == servers_.end() || server->second == NULL) { throw Orthanc::OrthancException(Orthanc::ErrorCode_InexistentItem, "Inexistent server: " + name); } else { return *server->second; } } void DicomWebServers::ListServers(std::list<std::string>& servers) { boost::mutex::scoped_lock lock(mutex_); servers.clear(); for (Servers::const_iterator it = servers_.begin(); it != servers_.end(); ++it) { servers.push_back(it->first); } } static std::string RemoveMultipleSlashes(const std::string& source) { std::string target; target.reserve(source.size()); size_t prefix = 0; if (boost::starts_with(source, "https://")) { prefix = 8; } else if (boost::starts_with(source, "http://")) { prefix = 7; } for (size_t i = 0; i < prefix; i++) { target.push_back(source[i]); } bool isLastSlash = false; for (size_t i = prefix; i < source.size(); i++) { if (source[i] == '/') { if (!isLastSlash) { target.push_back('/'); isLastSlash = true; } } else { target.push_back(source[i]); isLastSlash = false; } } return target; } void DicomWebServers::ConfigureHttpClient(HttpClient& client, const std::string& name, const std::string& uri) { const Orthanc::WebServiceParameters parameters = GetServer(name); client.SetUrl(RemoveMultipleSlashes(parameters.GetUrl() + "/" + uri)); client.SetHeaders(parameters.GetHttpHeaders()); if (!parameters.GetUsername().empty()) { client.SetCredentials(parameters.GetUsername(), parameters.GetPassword()); } } void DicomWebServers::DeleteServer(const std::string& name) { boost::mutex::scoped_lock lock(mutex_); Servers::iterator found = servers_.find(name); if (found == servers_.end()) { throw Orthanc::OrthancException(Orthanc::ErrorCode_ParameterOutOfRange, "Unknown DICOMweb server: " + name); } else { assert(found->second != NULL); delete found->second; servers_.erase(found); } } void DicomWebServers::SetServer(const std::string& name, const Orthanc::WebServiceParameters& parameters) { boost::mutex::scoped_lock lock(mutex_); Servers::iterator found = servers_.find(name); if (found != servers_.end()) { assert(found->second != NULL); delete found->second; servers_.erase(found); } servers_[name] = new Orthanc::WebServiceParameters(parameters); } static const char* ConvertToCString(const std::string& s) { if (s.empty()) { return NULL; } else { return s.c_str(); } } void CallServer(OrthancPlugins::MemoryBuffer& answerBody /* out */, std::map<std::string, std::string>& answerHeaders /* out */, const Orthanc::WebServiceParameters& server, OrthancPluginHttpMethod method, const std::map<std::string, std::string>& httpHeaders, const std::string& uri, const std::string& body) { answerBody.Clear(); answerHeaders.clear(); std::string url = server.GetUrl(); assert(!url.empty() && url[url.size() - 1] == '/'); // Remove the leading "/" in the URI if need be if (!uri.empty() && uri[0] == '/') { url += uri.substr(1); } else { url += uri; } std::map<std::string, std::string> allHttpHeaders = server.GetHttpHeaders(); { // Add the user-specified HTTP headers to the HTTP headers // coming from the Orthanc configuration file for (std::map<std::string, std::string>::const_iterator it = httpHeaders.begin(); it != httpHeaders.end(); ++it) { allHttpHeaders[it->first] = it->second; } } std::vector<const char*> httpHeadersKeys(allHttpHeaders.size()); std::vector<const char*> httpHeadersValues(allHttpHeaders.size()); { size_t pos = 0; for (std::map<std::string, std::string>::const_iterator it = allHttpHeaders.begin(); it != allHttpHeaders.end(); ++it) { httpHeadersKeys[pos] = it->first.c_str(); httpHeadersValues[pos] = it->second.c_str(); pos += 1; } assert(pos == allHttpHeaders.size()); } const char* bodyContent = NULL; size_t bodySize = 0; if ((method == OrthancPluginHttpMethod_Put || method == OrthancPluginHttpMethod_Post) && !body.empty()) { bodyContent = body.c_str(); bodySize = body.size(); } OrthancPluginContext* context = OrthancPlugins::GetGlobalContext(); uint16_t status = 0; MemoryBuffer answerHeadersTmp; OrthancPluginErrorCode code = OrthancPluginHttpClient( context, /* Outputs */ *answerBody, *answerHeadersTmp, &status, method, url.c_str(), /* HTTP headers*/ allHttpHeaders.size(), httpHeadersKeys.empty() ? NULL : &httpHeadersKeys[0], httpHeadersValues.empty() ? NULL : &httpHeadersValues[0], bodyContent, bodySize, ConvertToCString(server.GetUsername()), /* Authentication */ ConvertToCString(server.GetPassword()), 0, /* Timeout */ ConvertToCString(server.GetCertificateFile()), ConvertToCString(server.GetCertificateKeyFile()), ConvertToCString(server.GetCertificateKeyPassword()), server.IsPkcs11Enabled() ? 1 : 0); if (code != OrthancPluginErrorCode_Success || (status < 200 || status >= 300)) { throw Orthanc::OrthancException( static_cast<Orthanc::ErrorCode>(code), "Cannot issue an HTTP query to " + url + " (HTTP status: " + boost::lexical_cast<std::string>(status) + ")"); } Json::Value json; answerHeadersTmp.ToJson(json); answerHeadersTmp.Clear(); if (json.type() != Json::objectValue) { throw Orthanc::OrthancException(Orthanc::ErrorCode_InternalError); } Json::Value::Members members = json.getMemberNames(); for (size_t i = 0; i < members.size(); i++) { const std::string& key = members[i]; if (json[key].type() != Json::stringValue) { throw Orthanc::OrthancException(Orthanc::ErrorCode_InternalError); } else { answerHeaders[key] = json[key].asString(); } } } void DicomWebServers::UriEncode(std::string& uri, const std::string& resource, const std::map<std::string, std::string>& getArguments) { if (resource.find('?') != std::string::npos) { throw Orthanc::OrthancException(Orthanc::ErrorCode_BadFileFormat, "The GET arguments must be provided in a separate field " "(explicit \"?\" is disallowed): " + resource); } uri = resource; bool isFirst = true; for (std::map<std::string, std::string>::const_iterator it = getArguments.begin(); it != getArguments.end(); ++it) { if (isFirst) { uri += '?'; isFirst = false; } else { uri += '&'; } std::string key, value; Orthanc::Toolbox::UriEncode(key, it->first); Orthanc::Toolbox::UriEncode(value, it->second); if (value.empty()) { uri += key; } else { uri += key + "=" + value; } } } }