1 /** 2 * Orthanc - A Lightweight, RESTful DICOM Store 3 * Copyright (C) 2012-2016 Sebastien Jodogne, Medical Physics 4 * Department, University Hospital of Liege, Belgium 5 * Copyright (C) 2017-2021 Osimis S.A., Belgium 6 * 7 * This program is free software: you can redistribute it and/or 8 * modify it under the terms of the GNU General Public License as 9 * published by the Free Software Foundation, either version 3 of the 10 * License, or (at your option) any later version. 11 * 12 * In addition, as a special exception, the copyright holders of this 13 * program give permission to link the code of its release with the 14 * OpenSSL project's "OpenSSL" library (or with modified versions of it 15 * that use the same license as the "OpenSSL" library), and distribute 16 * the linked executables. You must obey the GNU General Public License 17 * in all respects for all of the code used other than "OpenSSL". If you 18 * modify file(s) with this exception, you may extend this exception to 19 * your version of the file(s), but you are not obligated to do so. If 20 * you do not wish to do so, delete this exception statement from your 21 * version. If you delete this exception statement from all source files 22 * in the program, then also delete it here. 23 * 24 * This program is distributed in the hope that it will be useful, but 25 * WITHOUT ANY WARRANTY; without even the implied warranty of 26 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 27 * General Public License for more details. 28 * 29 * You should have received a copy of the GNU General Public License 30 * along with this program. If not, see <http://www.gnu.org/licenses/>. 31 **/ 32 33 34 #include "../PrecompiledHeadersServer.h" 35 #include "OrthancRestApi.h" 36 37 #include "../../../OrthancFramework/Sources/DicomParsing/FromDcmtkBridge.h" 38 #include "../../../OrthancFramework/Sources/Logging.h" 39 #include "../../../OrthancFramework/Sources/SerializationToolbox.h" 40 #include "../OrthancConfiguration.h" 41 #include "../ServerContext.h" 42 #include "../ServerJobs/MergeStudyJob.h" 43 #include "../ServerJobs/ResourceModificationJob.h" 44 #include "../ServerJobs/SplitStudyJob.h" 45 46 #include <boost/lexical_cast.hpp> 47 #include <boost/algorithm/string/predicate.hpp> 48 49 #define INFO_SUBSEQUENCES \ 50 "Starting with Orthanc 1.9.4, paths to subsequences can be provided using the "\ 51 "same syntax as the `dcmodify` command-line tool (wildcards are supported as well)." 52 53 54 static const char* const CONTENT = "Content"; 55 static const char* const FORCE = "Force"; 56 static const char* const INSTANCES = "Instances"; 57 static const char* const INTERPRET_BINARY_TAGS = "InterpretBinaryTags"; 58 static const char* const KEEP = "Keep"; 59 static const char* const KEEP_PRIVATE_TAGS = "KeepPrivateTags"; 60 static const char* const KEEP_SOURCE = "KeepSource"; 61 static const char* const LEVEL = "Level"; 62 static const char* const PARENT = "Parent"; 63 static const char* const PRIVATE_CREATOR = "PrivateCreator"; 64 static const char* const REMOVE = "Remove"; 65 static const char* const REPLACE = "Replace"; 66 static const char* const RESOURCES = "Resources"; 67 static const char* const SERIES = "Series"; 68 static const char* const TAGS = "Tags"; 69 static const char* const TRANSCODE = "Transcode"; 70 71 72 namespace Orthanc 73 { 74 // Modification of DICOM instances ------------------------------------------ 75 76 GeneratePatientName(ServerContext & context)77 static std::string GeneratePatientName(ServerContext& context) 78 { 79 uint64_t seq = context.GetIndex().IncrementGlobalSequence(GlobalProperty_AnonymizationSequence, true /* shared */); 80 return "Anonymized" + boost::lexical_cast<std::string>(seq); 81 } 82 83 DocumentKeepSource(RestApiPostCall & call)84 static void DocumentKeepSource(RestApiPostCall& call) 85 { 86 call.GetDocumentation() 87 .SetRequestField(KEEP_SOURCE, RestApiCallDocumentation::Type_Boolean, 88 "If set to `false`, instructs Orthanc to the remove original resources. " 89 "By default, the original resources are kept in Orthanc.", false); 90 } 91 92 DocumentModifyOptions(RestApiPostCall & call)93 static void DocumentModifyOptions(RestApiPostCall& call) 94 { 95 // Check out "DicomModification::ParseModifyRequest()" 96 call.GetDocumentation() 97 .SetRequestField(TRANSCODE, RestApiCallDocumentation::Type_String, 98 "Transcode the DICOM instances to the provided DICOM transfer syntax: " 99 "https://book.orthanc-server.com/faq/transcoding.html", false) 100 .SetRequestField(FORCE, RestApiCallDocumentation::Type_Boolean, 101 "Allow the modification of tags related to DICOM identifiers, at the risk of " 102 "breaking the DICOM model of the real world", false) 103 .SetRequestField("RemovePrivateTags", RestApiCallDocumentation::Type_Boolean, 104 "Remove the private tags from the DICOM instances (defaults to `false`)", false) 105 .SetRequestField(REPLACE, RestApiCallDocumentation::Type_JsonObject, 106 "Associative array to change the value of some DICOM tags in the DICOM instances. " INFO_SUBSEQUENCES, false) 107 .SetRequestField(REMOVE, RestApiCallDocumentation::Type_JsonListOfStrings, 108 "List of tags that must be removed from the DICOM instances. " INFO_SUBSEQUENCES, false) 109 .SetRequestField(KEEP, RestApiCallDocumentation::Type_JsonListOfStrings, 110 "Keep the original value of the specified tags, to be chosen among the `StudyInstanceUID`, " 111 "`SeriesInstanceUID` and `SOPInstanceUID` tags. Avoid this feature as much as possible, " 112 "as this breaks the DICOM model of the real world.", false) 113 .SetRequestField(PRIVATE_CREATOR, RestApiCallDocumentation::Type_String, 114 "The private creator to be used for private tags in `Replace`", false); 115 116 // This was existing, but undocumented in Orthanc <= 1.9.6 117 DocumentKeepSource(call); 118 } 119 120 DocumentAnonymizationOptions(RestApiPostCall & call)121 static void DocumentAnonymizationOptions(RestApiPostCall& call) 122 { 123 // Check out "DicomModification::ParseAnonymizationRequest()" 124 call.GetDocumentation() 125 .SetRequestField(FORCE, RestApiCallDocumentation::Type_Boolean, 126 "Allow the modification of tags related to DICOM identifiers, at the risk of " 127 "breaking the DICOM model of the real world", false) 128 .SetRequestField("DicomVersion", RestApiCallDocumentation::Type_String, 129 "Version of the DICOM standard to be used for anonymization. Check out " 130 "configuration option `DeidentifyLogsDicomVersion` for possible values.", false) 131 .SetRequestField(KEEP_PRIVATE_TAGS, RestApiCallDocumentation::Type_Boolean, 132 "Keep the private tags from the DICOM instances (defaults to `false`)", false) 133 .SetRequestField(REPLACE, RestApiCallDocumentation::Type_JsonObject, 134 "Associative array to change the value of some DICOM tags in the DICOM instances. " INFO_SUBSEQUENCES, false) 135 .SetRequestField(REMOVE, RestApiCallDocumentation::Type_JsonListOfStrings, 136 "List of additional tags to be removed from the DICOM instances. " INFO_SUBSEQUENCES, false) 137 .SetRequestField(KEEP, RestApiCallDocumentation::Type_JsonListOfStrings, 138 "List of DICOM tags whose value must not be destroyed by the anonymization. " INFO_SUBSEQUENCES, false) 139 .SetRequestField(PRIVATE_CREATOR, RestApiCallDocumentation::Type_String, 140 "The private creator to be used for private tags in `Replace`", false); 141 142 // This was existing, but undocumented in Orthanc <= 1.9.6 143 DocumentKeepSource(call); 144 } 145 146 ParseModifyRequest(Json::Value & request,DicomModification & target,const RestApiPostCall & call)147 static void ParseModifyRequest(Json::Value& request, 148 DicomModification& target, 149 const RestApiPostCall& call) 150 { 151 // curl http://localhost:8042/series/95a6e2bf-9296e2cc-bf614e2f-22b391ee-16e010e0/modify -X POST -d '{"Replace":{"InstitutionName":"My own clinic"},"Priority":9}' 152 153 { 154 OrthancConfiguration::ReaderLock lock; 155 target.SetPrivateCreator(lock.GetConfiguration().GetDefaultPrivateCreator()); 156 } 157 158 if (call.ParseJsonRequest(request)) 159 { 160 target.ParseModifyRequest(request); 161 } 162 else 163 { 164 throw OrthancException(ErrorCode_BadFileFormat); 165 } 166 } 167 168 ParseAnonymizationRequest(Json::Value & request,DicomModification & target,RestApiPostCall & call)169 static void ParseAnonymizationRequest(Json::Value& request, 170 DicomModification& target, 171 RestApiPostCall& call) 172 { 173 // curl http://localhost:8042/instances/6e67da51-d119d6ae-c5667437-87b9a8a5-0f07c49f/anonymize -X POST -d '{"Replace":{"PatientName":"hello","0010-0020":"world"},"Keep":["StudyDescription", "SeriesDescription"],"KeepPrivateTags": true,"Remove":["Modality"]}' > Anonymized.dcm 174 175 { 176 OrthancConfiguration::ReaderLock lock; 177 target.SetPrivateCreator(lock.GetConfiguration().GetDefaultPrivateCreator()); 178 } 179 180 if (call.ParseJsonRequest(request) && 181 request.isObject()) 182 { 183 bool patientNameOverridden; 184 target.ParseAnonymizationRequest(patientNameOverridden, request); 185 186 if (!patientNameOverridden) 187 { 188 // Override the random Patient's Name by one that is more 189 // user-friendly (provided none was specified by the user) 190 target.Replace(DICOM_TAG_PATIENT_NAME, GeneratePatientName(OrthancRestApi::GetContext(call)), true); 191 } 192 } 193 else 194 { 195 throw OrthancException(ErrorCode_BadFileFormat); 196 } 197 } 198 199 AnonymizeOrModifyInstance(DicomModification & modification,RestApiPostCall & call,bool transcode,DicomTransferSyntax targetSyntax)200 static void AnonymizeOrModifyInstance(DicomModification& modification, 201 RestApiPostCall& call, 202 bool transcode, 203 DicomTransferSyntax targetSyntax) 204 { 205 ServerContext& context = OrthancRestApi::GetContext(call); 206 std::string id = call.GetUriComponent("id", ""); 207 208 std::unique_ptr<ParsedDicomFile> modified; 209 210 { 211 ServerContext::DicomCacheLocker locker(context, id); 212 modified.reset(locker.GetDicom().Clone(true)); 213 } 214 215 modification.Apply(*modified); 216 217 if (transcode) 218 { 219 IDicomTranscoder::DicomImage source; 220 source.AcquireParsed(*modified); // "modified" is invalid below this point 221 222 IDicomTranscoder::DicomImage transcoded; 223 224 std::set<DicomTransferSyntax> s; 225 s.insert(targetSyntax); 226 227 if (context.Transcode(transcoded, source, s, true)) 228 { 229 call.GetOutput().AnswerBuffer(transcoded.GetBufferData(), 230 transcoded.GetBufferSize(), MimeType_Dicom); 231 } 232 else 233 { 234 throw OrthancException(ErrorCode_InternalError, 235 "Cannot transcode to transfer syntax: " + 236 std::string(GetTransferSyntaxUid(targetSyntax))); 237 } 238 } 239 else 240 { 241 modified->Answer(call.GetOutput()); 242 } 243 } 244 245 DetectModifyLevel(const DicomModification & modification)246 static ResourceType DetectModifyLevel(const DicomModification& modification) 247 { 248 if (modification.IsReplaced(DICOM_TAG_PATIENT_ID)) 249 { 250 return ResourceType_Patient; 251 } 252 else if (modification.IsReplaced(DICOM_TAG_STUDY_INSTANCE_UID)) 253 { 254 return ResourceType_Study; 255 } 256 else if (modification.IsReplaced(DICOM_TAG_SERIES_INSTANCE_UID)) 257 { 258 return ResourceType_Series; 259 } 260 else 261 { 262 return ResourceType_Instance; 263 } 264 } 265 266 ModifyInstance(RestApiPostCall & call)267 static void ModifyInstance(RestApiPostCall& call) 268 { 269 if (call.IsDocumentation()) 270 { 271 DocumentModifyOptions(call); 272 call.GetDocumentation() 273 .SetTag("Instances") 274 .SetSummary("Modify instance") 275 .SetDescription("Download a modified version of the DICOM instance whose Orthanc identifier is provided in the URL: " 276 "https://book.orthanc-server.com/users/anonymization.html#modification-of-a-single-instance") 277 .SetUriArgument("id", "Orthanc identifier of the instance of interest") 278 .AddAnswerType(MimeType_Dicom, "The modified DICOM instance"); 279 return; 280 } 281 282 DicomModification modification; 283 modification.SetAllowManualIdentifiers(true); 284 285 Json::Value request; 286 ParseModifyRequest(request, modification, call); 287 288 modification.SetLevel(DetectModifyLevel(modification)); 289 290 if (request.isMember(TRANSCODE)) 291 { 292 std::string s = SerializationToolbox::ReadString(request, TRANSCODE); 293 294 DicomTransferSyntax syntax; 295 if (LookupTransferSyntax(syntax, s)) 296 { 297 AnonymizeOrModifyInstance(modification, call, true, syntax); 298 } 299 else 300 { 301 throw OrthancException(ErrorCode_ParameterOutOfRange, "Unknown transfer syntax: " + s); 302 } 303 } 304 else 305 { 306 AnonymizeOrModifyInstance(modification, call, false /* no transcoding */, 307 DicomTransferSyntax_LittleEndianImplicit /* unused */); 308 } 309 } 310 311 AnonymizeInstance(RestApiPostCall & call)312 static void AnonymizeInstance(RestApiPostCall& call) 313 { 314 if (call.IsDocumentation()) 315 { 316 DocumentAnonymizationOptions(call); 317 call.GetDocumentation() 318 .SetTag("Instances") 319 .SetSummary("Anonymize instance") 320 .SetDescription("Download an anonymized version of the DICOM instance whose Orthanc identifier is provided in the URL: " 321 "https://book.orthanc-server.com/users/anonymization.html#anonymization-of-a-single-instance") 322 .SetUriArgument("id", "Orthanc identifier of the instance of interest") 323 .AddAnswerType(MimeType_Dicom, "The anonymized DICOM instance"); 324 return; 325 } 326 327 DicomModification modification; 328 modification.SetAllowManualIdentifiers(true); 329 330 Json::Value request; 331 ParseAnonymizationRequest(request, modification, call); 332 333 AnonymizeOrModifyInstance(modification, call, false /* no transcoding */, 334 DicomTransferSyntax_LittleEndianImplicit /* unused */); 335 } 336 337 SetKeepSource(CleaningInstancesJob & job,const Json::Value & body)338 static void SetKeepSource(CleaningInstancesJob& job, 339 const Json::Value& body) 340 { 341 if (body.isMember(KEEP_SOURCE)) 342 { 343 job.SetKeepSource(SerializationToolbox::ReadBoolean(body, KEEP_SOURCE)); 344 } 345 } 346 347 SubmitModificationJob(std::unique_ptr<DicomModification> & modification,bool isAnonymization,RestApiPostCall & call,const Json::Value & body,ResourceType outputLevel,bool isSingleResource,const std::set<std::string> & resources)348 static void SubmitModificationJob(std::unique_ptr<DicomModification>& modification, 349 bool isAnonymization, 350 RestApiPostCall& call, 351 const Json::Value& body, 352 ResourceType outputLevel /* unused for multiple resources */, 353 bool isSingleResource, 354 const std::set<std::string>& resources) 355 { 356 ServerContext& context = OrthancRestApi::GetContext(call); 357 358 std::unique_ptr<ResourceModificationJob> job(new ResourceModificationJob(context)); 359 360 if (isSingleResource) // This notably configures the output format 361 { 362 job->SetSingleResourceModification(modification.release(), outputLevel, isAnonymization); 363 } 364 else 365 { 366 job->SetMultipleResourcesModification(modification.release(), isAnonymization); 367 } 368 369 job->SetOrigin(call); 370 SetKeepSource(*job, body); 371 372 if (body.isMember(TRANSCODE)) 373 { 374 job->SetTranscode(SerializationToolbox::ReadString(body, TRANSCODE)); 375 } 376 377 for (std::set<std::string>::const_iterator 378 it = resources.begin(); it != resources.end(); ++it) 379 { 380 context.AddChildInstances(*job, *it); 381 } 382 383 job->AddTrailingStep(); 384 385 OrthancRestApi::GetApi(call).SubmitCommandsJob 386 (call, job.release(), true /* synchronous by default */, body); 387 } 388 389 SubmitModificationJob(std::unique_ptr<DicomModification> & modification,bool isAnonymization,RestApiPostCall & call,const Json::Value & body,ResourceType outputLevel)390 static void SubmitModificationJob(std::unique_ptr<DicomModification>& modification, 391 bool isAnonymization, 392 RestApiPostCall& call, 393 const Json::Value& body, 394 ResourceType outputLevel) 395 { 396 // This was the only flavor in Orthanc <= 1.9.3 397 std::set<std::string> resources; 398 resources.insert(call.GetUriComponent("id", "")); 399 400 SubmitModificationJob(modification, isAnonymization, call, body, outputLevel, 401 true /* single resource */, resources); 402 } 403 404 SubmitBulkJob(std::unique_ptr<DicomModification> & modification,bool isAnonymization,RestApiPostCall & call,const Json::Value & body)405 static void SubmitBulkJob(std::unique_ptr<DicomModification>& modification, 406 bool isAnonymization, 407 RestApiPostCall& call, 408 const Json::Value& body) 409 { 410 std::set<std::string> resources; 411 SerializationToolbox::ReadSetOfStrings(resources, body, RESOURCES); 412 413 SubmitModificationJob(modification, isAnonymization, 414 call, body, ResourceType_Instance /* arbitrary value, unused */, 415 false /* multiple resources */, resources); 416 } 417 418 419 template <enum ResourceType resourceType> ModifyResource(RestApiPostCall & call)420 static void ModifyResource(RestApiPostCall& call) 421 { 422 if (call.IsDocumentation()) 423 { 424 OrthancRestApi::DocumentSubmitCommandsJob(call); 425 DocumentModifyOptions(call); 426 const std::string r = GetResourceTypeText(resourceType, false /* plural */, false /* lower case */); 427 call.GetDocumentation() 428 .SetTag(GetResourceTypeText(resourceType, true /* plural */, true /* upper case */)) 429 .SetSummary("Modify " + r) 430 .SetDescription("Start a job that will modify all the DICOM instances within the " + r + 431 " whose identifier is provided in the URL. The modified DICOM instances will be " 432 "stored into a brand new " + r + ", whose Orthanc identifiers will be returned by the job. " 433 "https://book.orthanc-server.com/users/anonymization.html#modification-of-studies-or-series") 434 .SetUriArgument("id", "Orthanc identifier of the " + r + " of interest"); 435 return; 436 } 437 438 std::unique_ptr<DicomModification> modification(new DicomModification); 439 440 Json::Value body; 441 ParseModifyRequest(body, *modification, call); 442 443 modification->SetLevel(resourceType); 444 445 SubmitModificationJob(modification, false /* not an anonymization */, 446 call, body, resourceType); 447 } 448 449 450 // New in Orthanc 1.9.4 BulkModify(RestApiPostCall & call)451 static void BulkModify(RestApiPostCall& call) 452 { 453 if (call.IsDocumentation()) 454 { 455 OrthancRestApi::DocumentSubmitCommandsJob(call); 456 DocumentModifyOptions(call); 457 call.GetDocumentation() 458 .SetTag("System") 459 .SetSummary("Modify a set of resources") 460 .SetRequestField(RESOURCES, RestApiCallDocumentation::Type_JsonListOfStrings, 461 "List of the Orthanc identifiers of the patients/studies/series/instances of interest.", true) 462 .SetRequestField(LEVEL, RestApiCallDocumentation::Type_String, 463 "Level of the modification (`Patient`, `Study`, `Series` or `Instance`). If absent, " 464 "the level defaults to `Instance`, but is set to `Patient` if `PatientID` is modified, " 465 "to `Study` if `StudyInstanceUID` is modified, or to `Series` if `SeriesInstancesUID` " 466 "is modified. (new in Orthanc 1.9.7)", false) 467 .SetDescription("Start a job that will modify all the DICOM patients, studies, series or instances " 468 "whose identifiers are provided in the `Resources` field.") 469 .AddAnswerType(MimeType_Json, "The list of all the resources that have been altered by this modification"); 470 return; 471 } 472 473 std::unique_ptr<DicomModification> modification(new DicomModification); 474 475 Json::Value body; 476 ParseModifyRequest(body, *modification, call); 477 478 if (body.isMember(LEVEL)) 479 { 480 // This case was introduced in Orthanc 1.9.7 481 modification->SetLevel(StringToResourceType(body[LEVEL].asCString())); 482 } 483 else 484 { 485 modification->SetLevel(DetectModifyLevel(*modification)); 486 } 487 488 SubmitBulkJob(modification, false /* not an anonymization */, call, body); 489 } 490 491 492 template <enum ResourceType resourceType> AnonymizeResource(RestApiPostCall & call)493 static void AnonymizeResource(RestApiPostCall& call) 494 { 495 if (call.IsDocumentation()) 496 { 497 OrthancRestApi::DocumentSubmitCommandsJob(call); 498 DocumentAnonymizationOptions(call); 499 const std::string r = GetResourceTypeText(resourceType, false /* plural */, false /* lower case */); 500 call.GetDocumentation() 501 .SetTag(GetResourceTypeText(resourceType, true /* plural */, true /* upper case */)) 502 .SetSummary("Anonymize " + r) 503 .SetDescription("Start a job that will anonymize all the DICOM instances within the " + r + 504 " whose identifier is provided in the URL. The modified DICOM instances will be " 505 "stored into a brand new " + r + ", whose Orthanc identifiers will be returned by the job. " 506 "https://book.orthanc-server.com/users/anonymization.html#anonymization-of-patients-studies-or-series") 507 .SetUriArgument("id", "Orthanc identifier of the " + r + " of interest"); 508 return; 509 } 510 511 std::unique_ptr<DicomModification> modification(new DicomModification); 512 513 Json::Value body; 514 ParseAnonymizationRequest(body, *modification, call); 515 516 SubmitModificationJob(modification, true /* anonymization */, 517 call, body, resourceType); 518 } 519 520 521 // New in Orthanc 1.9.4 BulkAnonymize(RestApiPostCall & call)522 static void BulkAnonymize(RestApiPostCall& call) 523 { 524 if (call.IsDocumentation()) 525 { 526 OrthancRestApi::DocumentSubmitCommandsJob(call); 527 DocumentAnonymizationOptions(call); 528 call.GetDocumentation() 529 .SetTag("System") 530 .SetSummary("Anonymize a set of resources") 531 .SetRequestField(RESOURCES, RestApiCallDocumentation::Type_JsonListOfStrings, 532 "List of the Orthanc identifiers of the patients/studies/series/instances of interest.", true) 533 .SetDescription("Start a job that will anonymize all the DICOM patients, studies, series or instances " 534 "whose identifiers are provided in the `Resources` field.") 535 .AddAnswerType(MimeType_Json, "The list of all the resources that have been created by this anonymization"); 536 return; 537 } 538 539 std::unique_ptr<DicomModification> modification(new DicomModification); 540 541 Json::Value body; 542 ParseAnonymizationRequest(body, *modification, call); 543 544 SubmitBulkJob(modification, true /* anonymization */, call, body); 545 } 546 547 StoreCreatedInstance(std::string & id,RestApiPostCall & call,ParsedDicomFile & dicom,bool sendAnswer)548 static void StoreCreatedInstance(std::string& id /* out */, 549 RestApiPostCall& call, 550 ParsedDicomFile& dicom, 551 bool sendAnswer) 552 { 553 std::unique_ptr<DicomInstanceToStore> toStore(DicomInstanceToStore::CreateFromParsedDicomFile(dicom)); 554 toStore->SetOrigin(DicomInstanceOrigin::FromRest(call)); 555 556 ServerContext& context = OrthancRestApi::GetContext(call); 557 StoreStatus status = context.Store(id, *toStore, StoreInstanceMode_Default); 558 559 if (status == StoreStatus_Failure) 560 { 561 throw OrthancException(ErrorCode_CannotStoreInstance); 562 } 563 564 if (sendAnswer) 565 { 566 OrthancRestApi::GetApi(call).AnswerStoredInstance(call, *toStore, status, id); 567 } 568 } 569 570 CreateDicomV1(ParsedDicomFile & dicom,RestApiPostCall & call,const Json::Value & request)571 static void CreateDicomV1(ParsedDicomFile& dicom, 572 RestApiPostCall& call, 573 const Json::Value& request) 574 { 575 // curl http://localhost:8042/tools/create-dicom -X POST -d '{"PatientName":"Hello^World"}' 576 // curl http://localhost:8042/tools/create-dicom -X POST -d '{"PatientName":"Hello^World","PixelData":""}' 577 578 assert(request.isObject()); 579 LOG(WARNING) << "Using a deprecated call to /tools/create-dicom"; 580 581 Json::Value::Members members = request.getMemberNames(); 582 for (size_t i = 0; i < members.size(); i++) 583 { 584 const std::string& name = members[i]; 585 if (request[name].type() != Json::stringValue) 586 { 587 throw OrthancException(ErrorCode_CreateDicomNotString); 588 } 589 590 std::string value = request[name].asString(); 591 592 DicomTag tag = FromDcmtkBridge::ParseTag(name); 593 if (tag == DICOM_TAG_PIXEL_DATA) 594 { 595 dicom.EmbedContent(value); 596 } 597 else 598 { 599 // This is V1, don't try and decode data URI scheme 600 dicom.ReplacePlainString(tag, value); 601 } 602 } 603 } 604 605 InjectTags(ParsedDicomFile & dicom,const Json::Value & tags,bool decodeBinaryTags,const std::string & privateCreator,bool force)606 static void InjectTags(ParsedDicomFile& dicom, 607 const Json::Value& tags, 608 bool decodeBinaryTags, 609 const std::string& privateCreator, 610 bool force) 611 { 612 if (tags.type() != Json::objectValue) 613 { 614 throw OrthancException(ErrorCode_BadRequest, "Tags field is not an array"); 615 } 616 617 // Inject the user-specified tags 618 Json::Value::Members members = tags.getMemberNames(); 619 for (size_t i = 0; i < members.size(); i++) 620 { 621 const std::string& name = members[i]; 622 DicomTag tag = FromDcmtkBridge::ParseTag(name); 623 624 if (tag != DICOM_TAG_SPECIFIC_CHARACTER_SET) 625 { 626 if (!force && 627 tag != DICOM_TAG_PATIENT_ID && 628 tag != DICOM_TAG_ACQUISITION_DATE && 629 tag != DICOM_TAG_ACQUISITION_TIME && 630 tag != DICOM_TAG_CONTENT_DATE && 631 tag != DICOM_TAG_CONTENT_TIME && 632 tag != DICOM_TAG_INSTANCE_CREATION_DATE && 633 tag != DICOM_TAG_INSTANCE_CREATION_TIME && 634 tag != DICOM_TAG_SERIES_DATE && 635 tag != DICOM_TAG_SERIES_TIME && 636 tag != DICOM_TAG_STUDY_DATE && 637 tag != DICOM_TAG_STUDY_TIME && 638 dicom.HasTag(tag)) 639 { 640 throw OrthancException(ErrorCode_CreateDicomOverrideTag, name); 641 } 642 643 if (tag == DICOM_TAG_PIXEL_DATA) 644 { 645 throw OrthancException(ErrorCode_CreateDicomUseContent); 646 } 647 else 648 { 649 dicom.Replace(tag, tags[name], decodeBinaryTags, DicomReplaceMode_InsertIfAbsent, privateCreator); 650 } 651 } 652 } 653 } 654 655 CreateSeries(RestApiPostCall & call,ParsedDicomFile & base,const Json::Value & content,bool decodeBinaryTags,const std::string & privateCreator,bool force)656 static void CreateSeries(RestApiPostCall& call, 657 ParsedDicomFile& base /* in */, 658 const Json::Value& content, 659 bool decodeBinaryTags, 660 const std::string& privateCreator, 661 bool force) 662 { 663 assert(content.isArray()); 664 assert(content.size() > 0); 665 ServerContext& context = OrthancRestApi::GetContext(call); 666 667 base.ReplacePlainString(DICOM_TAG_IMAGES_IN_ACQUISITION, boost::lexical_cast<std::string>(content.size())); 668 base.ReplacePlainString(DICOM_TAG_NUMBER_OF_TEMPORAL_POSITIONS, "1"); 669 670 std::string someInstance; 671 672 try 673 { 674 for (Json::ArrayIndex i = 0; i < content.size(); i++) 675 { 676 std::unique_ptr<ParsedDicomFile> dicom(base.Clone(false)); 677 const Json::Value* payload = NULL; 678 679 if (content[i].type() == Json::stringValue) 680 { 681 payload = &content[i]; 682 } 683 else if (content[i].type() == Json::objectValue) 684 { 685 if (!content[i].isMember(CONTENT)) 686 { 687 throw OrthancException(ErrorCode_CreateDicomNoPayload); 688 } 689 690 payload = &content[i][CONTENT]; 691 692 if (content[i].isMember(TAGS)) 693 { 694 InjectTags(*dicom, content[i][TAGS], decodeBinaryTags, privateCreator, force); 695 } 696 } 697 698 if (payload == NULL || 699 payload->type() != Json::stringValue) 700 { 701 throw OrthancException(ErrorCode_CreateDicomUseDataUriScheme); 702 } 703 704 dicom->EmbedContent(payload->asString()); 705 dicom->ReplacePlainString(DICOM_TAG_INSTANCE_NUMBER, boost::lexical_cast<std::string>(i + 1)); 706 dicom->ReplacePlainString(DICOM_TAG_IMAGE_INDEX, boost::lexical_cast<std::string>(i + 1)); 707 708 StoreCreatedInstance(someInstance, call, *dicom, false); 709 } 710 } 711 catch (OrthancException&) 712 { 713 // Error: Remove the newly-created series 714 715 std::string series; 716 if (context.GetIndex().LookupParent(series, someInstance)) 717 { 718 Json::Value dummy; 719 context.GetIndex().DeleteResource(dummy, series, ResourceType_Series); 720 } 721 722 throw; 723 } 724 725 std::string series; 726 if (context.GetIndex().LookupParent(series, someInstance)) 727 { 728 OrthancRestApi::GetApi(call).AnswerStoredResource(call, series, ResourceType_Series, StoreStatus_Success); 729 } 730 } 731 732 CreateDicomV2(RestApiPostCall & call,const Json::Value & request)733 static void CreateDicomV2(RestApiPostCall& call, 734 const Json::Value& request) 735 { 736 static const char* const SPECIFIC_CHARACTER_SET_2 = "SpecificCharacterSet"; 737 static const char* const TYPE = "Type"; 738 static const char* const VALUE = "Value"; 739 740 assert(request.isObject()); 741 ServerContext& context = OrthancRestApi::GetContext(call); 742 743 if (!request.isMember(TAGS) || 744 request[TAGS].type() != Json::objectValue) 745 { 746 throw OrthancException(ErrorCode_BadRequest); 747 } 748 749 ParsedDicomFile dicom(true); 750 751 { 752 Encoding encoding; 753 754 if (request[TAGS].isMember(SPECIFIC_CHARACTER_SET_2)) 755 { 756 const char* tmp = request[TAGS][SPECIFIC_CHARACTER_SET_2].asCString(); 757 if (!GetDicomEncoding(encoding, tmp)) 758 { 759 throw OrthancException(ErrorCode_ParameterOutOfRange, 760 "Unknown specific character set: " + std::string(tmp)); 761 } 762 } 763 else 764 { 765 encoding = GetDefaultDicomEncoding(); 766 } 767 768 dicom.SetEncoding(encoding); 769 } 770 771 ResourceType parentType = ResourceType_Instance; 772 773 if (request.isMember(PARENT)) 774 { 775 // Locate the parent tags 776 std::string parent = request[PARENT].asString(); 777 if (!context.GetIndex().LookupResourceType(parentType, parent)) 778 { 779 throw OrthancException(ErrorCode_CreateDicomBadParent); 780 } 781 782 if (parentType == ResourceType_Instance) 783 { 784 throw OrthancException(ErrorCode_CreateDicomParentIsInstance); 785 } 786 787 // Select one existing child instance of the parent resource, to 788 // retrieve all its tags 789 Json::Value siblingTags; 790 std::string siblingInstanceId; 791 792 { 793 // Retrieve all the instances of the parent resource 794 std::list<std::string> siblingInstances; 795 context.GetIndex().GetChildInstances(siblingInstances, parent); 796 797 if (siblingInstances.empty()) 798 { 799 // Error: No instance (should never happen) 800 throw OrthancException(ErrorCode_InternalError); 801 } 802 803 siblingInstanceId = siblingInstances.front(); 804 context.ReadDicomAsJson(siblingTags, siblingInstanceId); 805 } 806 807 808 // Choose the same encoding as the parent resource 809 { 810 static const char* const SPECIFIC_CHARACTER_SET = "0008,0005"; 811 812 if (siblingTags.isMember(SPECIFIC_CHARACTER_SET)) 813 { 814 Encoding encoding; 815 816 if (!siblingTags[SPECIFIC_CHARACTER_SET].isMember(VALUE) || 817 siblingTags[SPECIFIC_CHARACTER_SET][VALUE].type() != Json::stringValue || 818 !GetDicomEncoding(encoding, siblingTags[SPECIFIC_CHARACTER_SET][VALUE].asCString())) 819 { 820 LOG(WARNING) << "Instance with an incorrect Specific Character Set, " 821 << "using the default Orthanc encoding: " << siblingInstanceId; 822 encoding = GetDefaultDicomEncoding(); 823 } 824 825 dicom.SetEncoding(encoding); 826 } 827 } 828 829 830 // Retrieve the tags for all the parent modules 831 typedef std::set<DicomTag> ModuleTags; 832 ModuleTags moduleTags; 833 834 ResourceType type = parentType; 835 for (;;) 836 { 837 DicomTag::AddTagsForModule(moduleTags, GetModule(type)); 838 839 if (type == ResourceType_Patient) 840 { 841 break; // We're done 842 } 843 844 // Go up 845 std::string tmp; 846 if (!context.GetIndex().LookupParent(tmp, parent)) 847 { 848 throw OrthancException(ErrorCode_InternalError); 849 } 850 851 parent = tmp; 852 type = GetParentResourceType(type); 853 } 854 855 for (ModuleTags::const_iterator it = moduleTags.begin(); 856 it != moduleTags.end(); ++it) 857 { 858 std::string t = it->Format(); 859 if (siblingTags.isMember(t)) 860 { 861 const Json::Value& tag = siblingTags[t]; 862 if (tag[TYPE] == "Null") 863 { 864 dicom.ReplacePlainString(*it, ""); 865 } 866 else if (tag[TYPE] == "String") 867 { 868 std::string value = tag[VALUE].asString(); // This is an UTF-8 value (as it comes from JSON) 869 dicom.ReplacePlainString(*it, value); 870 } 871 } 872 } 873 } 874 875 876 bool decodeBinaryTags = true; 877 if (request.isMember(INTERPRET_BINARY_TAGS)) 878 { 879 const Json::Value& v = request[INTERPRET_BINARY_TAGS]; 880 if (v.type() != Json::booleanValue) 881 { 882 throw OrthancException(ErrorCode_BadRequest); 883 } 884 885 decodeBinaryTags = v.asBool(); 886 } 887 888 889 // New argument in Orthanc 1.6.0 890 std::string privateCreator; 891 if (request.isMember(PRIVATE_CREATOR)) 892 { 893 const Json::Value& v = request[PRIVATE_CREATOR]; 894 if (v.type() != Json::stringValue) 895 { 896 throw OrthancException(ErrorCode_BadRequest); 897 } 898 899 privateCreator = v.asString(); 900 } 901 else 902 { 903 OrthancConfiguration::ReaderLock lock; 904 privateCreator = lock.GetConfiguration().GetDefaultPrivateCreator(); 905 } 906 907 908 // New in Orthanc 1.9.0 909 bool force = false; 910 if (request.isMember(FORCE)) 911 { 912 const Json::Value& v = request[FORCE]; 913 if (v.type() != Json::booleanValue) 914 { 915 throw OrthancException(ErrorCode_BadRequest); 916 } 917 918 force = v.asBool(); 919 } 920 921 922 // Inject time-related information 923 std::string date, time; 924 SystemToolbox::GetNowDicom(date, time, true /* use UTC time (not local time) */); 925 dicom.ReplacePlainString(DICOM_TAG_ACQUISITION_DATE, date); 926 dicom.ReplacePlainString(DICOM_TAG_ACQUISITION_TIME, time); 927 dicom.ReplacePlainString(DICOM_TAG_CONTENT_DATE, date); 928 dicom.ReplacePlainString(DICOM_TAG_CONTENT_TIME, time); 929 dicom.ReplacePlainString(DICOM_TAG_INSTANCE_CREATION_DATE, date); 930 dicom.ReplacePlainString(DICOM_TAG_INSTANCE_CREATION_TIME, time); 931 932 if (parentType == ResourceType_Patient || 933 parentType == ResourceType_Study || 934 parentType == ResourceType_Instance /* no parent */) 935 { 936 dicom.ReplacePlainString(DICOM_TAG_SERIES_DATE, date); 937 dicom.ReplacePlainString(DICOM_TAG_SERIES_TIME, time); 938 } 939 940 if (parentType == ResourceType_Patient || 941 parentType == ResourceType_Instance /* no parent */) 942 { 943 dicom.ReplacePlainString(DICOM_TAG_STUDY_DATE, date); 944 dicom.ReplacePlainString(DICOM_TAG_STUDY_TIME, time); 945 } 946 947 948 InjectTags(dicom, request[TAGS], decodeBinaryTags, privateCreator, force); 949 950 951 // Inject the content (either an image, or a PDF file) 952 if (request.isMember(CONTENT)) 953 { 954 const Json::Value& content = request[CONTENT]; 955 956 if (content.type() == Json::stringValue) 957 { 958 dicom.EmbedContent(request[CONTENT].asString()); 959 960 } 961 else if (content.type() == Json::arrayValue) 962 { 963 if (content.size() > 0) 964 { 965 // Let's create a series instead of a single instance 966 CreateSeries(call, dicom, content, decodeBinaryTags, privateCreator, force); 967 return; 968 } 969 } 970 else 971 { 972 throw OrthancException(ErrorCode_CreateDicomUseDataUriScheme); 973 } 974 } 975 976 std::string id; 977 StoreCreatedInstance(id, call, dicom, true); 978 } 979 980 CreateDicom(RestApiPostCall & call)981 static void CreateDicom(RestApiPostCall& call) 982 { 983 if (call.IsDocumentation()) 984 { 985 call.GetDocumentation() 986 .SetTag("System") 987 .SetSummary("Create one DICOM instance") 988 .SetDescription("Create one DICOM instance, and store it into Orthanc") 989 .SetRequestField(TAGS, RestApiCallDocumentation::Type_JsonObject, 990 "Associative array containing the tags of the new instance to be created", true) 991 .SetRequestField(CONTENT, RestApiCallDocumentation::Type_String, 992 "This field can be used to embed an image (pixel data) or a PDF inside the created DICOM instance. " 993 "The PNG image, the JPEG image or the PDF file must be provided using their " 994 "[data URI scheme encoding](https://en.wikipedia.org/wiki/Data_URI_scheme). " 995 "This field can possibly contain a JSON array, in which case a DICOM series is created " 996 "containing one DICOM instance for each item in the `Content` field.", false) 997 .SetRequestField(PARENT, RestApiCallDocumentation::Type_String, 998 "If present, the newly created instance will be attached to the parent DICOM resource " 999 "whose Orthanc identifier is contained in this field. The DICOM tags of the parent " 1000 "modules in the DICOM hierarchy will be automatically copied to the newly created instance.", false) 1001 .SetRequestField(INTERPRET_BINARY_TAGS, RestApiCallDocumentation::Type_Boolean, 1002 "If some value in the `Tags` associative array is formatted according to some " 1003 "[data URI scheme encoding](https://en.wikipedia.org/wiki/Data_URI_scheme), " 1004 "whether this value is decoded to a binary value or kept as such (`true` by default)", false) 1005 .SetRequestField(PRIVATE_CREATOR, RestApiCallDocumentation::Type_String, 1006 "The private creator to be used for private tags in `Tags`", false) 1007 .SetRequestField(FORCE, RestApiCallDocumentation::Type_Boolean, 1008 "Avoid the consistency checks for the DICOM tags that enforce the DICOM model of the real-world. " 1009 "You can notably use this flag if you need to manually set the tags `StudyInstanceUID`, " 1010 "`SeriesInstanceUID`, or `SOPInstanceUID`. Be careful with this feature.", false) 1011 .SetAnswerField("ID", RestApiCallDocumentation::Type_String, "Orthanc identifier of the newly created instance") 1012 .SetAnswerField("Path", RestApiCallDocumentation::Type_String, "Path to access the instance in the REST API"); 1013 return; 1014 } 1015 1016 Json::Value request; 1017 if (!call.ParseJsonRequest(request) || 1018 !request.isObject()) 1019 { 1020 throw OrthancException(ErrorCode_BadRequest); 1021 } 1022 1023 if (request.isMember(TAGS)) 1024 { 1025 CreateDicomV2(call, request); 1026 } 1027 else 1028 { 1029 // Compatibility with Orthanc <= 0.9.3 1030 ParsedDicomFile dicom(true); 1031 CreateDicomV1(dicom, call, request); 1032 1033 std::string id; 1034 StoreCreatedInstance(id, call, dicom, true); 1035 } 1036 } 1037 1038 SplitStudy(RestApiPostCall & call)1039 static void SplitStudy(RestApiPostCall& call) 1040 { 1041 if (call.IsDocumentation()) 1042 { 1043 OrthancRestApi::DocumentSubmitCommandsJob(call); 1044 call.GetDocumentation() 1045 .SetTag("Studies") 1046 .SetSummary("Split study") 1047 .SetDescription("Start a new job so as to split the DICOM study whose Orthanc identifier is provided in the URL, " 1048 "by taking some of its children series or instances out of it and putting them into a brand new study " 1049 "(this new study is created by setting the `StudyInstanceUID` tag to a random identifier): " 1050 "https://book.orthanc-server.com/users/anonymization.html#splitting") 1051 .SetUriArgument("id", "Orthanc identifier of the study of interest") 1052 .SetRequestField(SERIES, RestApiCallDocumentation::Type_JsonListOfStrings, 1053 "The list of series to be separated from the parent study. " 1054 "These series must all be children of the same source study, that is specified in the URI.", false) 1055 .SetRequestField(REPLACE, RestApiCallDocumentation::Type_JsonObject, 1056 "Associative array to change the value of some DICOM tags in the new study. " 1057 "These tags must be part of the \"Patient Module Attributes\" or the \"General Study " 1058 "Module Attributes\", as specified by the DICOM 2011 standard in Tables C.7-1 and C.7-3.", false) 1059 .SetRequestField(REMOVE, RestApiCallDocumentation::Type_JsonListOfStrings, 1060 "List of tags that must be removed in the new study (from the same modules as in the `Replace` option)", false) 1061 .SetRequestField(KEEP_SOURCE, RestApiCallDocumentation::Type_Boolean, 1062 "If set to `true`, instructs Orthanc to keep a copy of the original series/instances in the source study. " 1063 "By default, the original series/instances are deleted from Orthanc.", false) 1064 .SetRequestField(INSTANCES, RestApiCallDocumentation::Type_JsonListOfStrings, 1065 "The list of instances to be separated from the parent study. " 1066 "These instances must all be children of the same source study, that is specified in the URI.", false); 1067 return; 1068 } 1069 1070 ServerContext& context = OrthancRestApi::GetContext(call); 1071 1072 Json::Value request; 1073 if (!call.ParseJsonRequest(request)) 1074 { 1075 // Bad JSON request 1076 throw OrthancException(ErrorCode_BadFileFormat); 1077 } 1078 1079 const std::string study = call.GetUriComponent("id", ""); 1080 1081 std::unique_ptr<SplitStudyJob> job(new SplitStudyJob(context, study)); 1082 job->SetOrigin(call); 1083 1084 bool ok = false; 1085 if (request.isMember(SERIES)) 1086 { 1087 std::vector<std::string> series; 1088 SerializationToolbox::ReadArrayOfStrings(series, request, SERIES); 1089 1090 for (size_t i = 0; i < series.size(); i++) 1091 { 1092 job->AddSourceSeries(series[i]); 1093 ok = true; 1094 } 1095 } 1096 1097 if (request.isMember(INSTANCES)) 1098 { 1099 std::vector<std::string> instances; 1100 SerializationToolbox::ReadArrayOfStrings(instances, request, INSTANCES); 1101 1102 for (size_t i = 0; i < instances.size(); i++) 1103 { 1104 job->AddSourceInstance(instances[i]); 1105 ok = true; 1106 } 1107 } 1108 1109 if (!ok) 1110 { 1111 throw OrthancException(ErrorCode_BadRequest, "Both the \"Series\" and the \"Instances\" fields are missing"); 1112 } 1113 1114 job->AddTrailingStep(); 1115 1116 SetKeepSource(*job, request); 1117 1118 if (request.isMember(REMOVE)) 1119 { 1120 if (request[REMOVE].type() != Json::arrayValue) 1121 { 1122 throw OrthancException(ErrorCode_BadFileFormat); 1123 } 1124 1125 for (Json::Value::ArrayIndex i = 0; i < request[REMOVE].size(); i++) 1126 { 1127 if (request[REMOVE][i].type() != Json::stringValue) 1128 { 1129 throw OrthancException(ErrorCode_BadFileFormat); 1130 } 1131 else 1132 { 1133 job->Remove(FromDcmtkBridge::ParseTag(request[REMOVE][i].asCString())); 1134 } 1135 } 1136 } 1137 1138 if (request.isMember(REPLACE)) 1139 { 1140 if (request[REPLACE].type() != Json::objectValue) 1141 { 1142 throw OrthancException(ErrorCode_BadFileFormat); 1143 } 1144 1145 Json::Value::Members tags = request[REPLACE].getMemberNames(); 1146 1147 for (size_t i = 0; i < tags.size(); i++) 1148 { 1149 const Json::Value& value = request[REPLACE][tags[i]]; 1150 1151 if (value.type() != Json::stringValue) 1152 { 1153 throw OrthancException(ErrorCode_BadFileFormat); 1154 } 1155 else 1156 { 1157 job->Replace(FromDcmtkBridge::ParseTag(tags[i]), value.asString()); 1158 } 1159 } 1160 } 1161 1162 OrthancRestApi::GetApi(call).SubmitCommandsJob 1163 (call, job.release(), true /* synchronous by default */, request); 1164 } 1165 1166 MergeStudy(RestApiPostCall & call)1167 static void MergeStudy(RestApiPostCall& call) 1168 { 1169 if (call.IsDocumentation()) 1170 { 1171 OrthancRestApi::DocumentSubmitCommandsJob(call); 1172 call.GetDocumentation() 1173 .SetTag("Studies") 1174 .SetSummary("Merge study") 1175 .SetDescription("Start a new job so as to move some DICOM resources into the DICOM study whose Orthanc identifier " 1176 "is provided in the URL: https://book.orthanc-server.com/users/anonymization.html#merging") 1177 .SetUriArgument("id", "Orthanc identifier of the study of interest") 1178 .SetRequestField(RESOURCES, RestApiCallDocumentation::Type_JsonListOfStrings, 1179 "The list of DICOM resources (studies, series, and/or instances) to be merged " 1180 "into the study of interest (mandatory option)", true) 1181 .SetRequestField(KEEP_SOURCE, RestApiCallDocumentation::Type_Boolean, 1182 "If set to `true`, instructs Orthanc to keep a copy of the original resources in their source study. " 1183 "By default, the original resources are deleted from Orthanc.", false); 1184 return; 1185 } 1186 1187 ServerContext& context = OrthancRestApi::GetContext(call); 1188 1189 Json::Value request; 1190 if (!call.ParseJsonRequest(request)) 1191 { 1192 // Bad JSON request 1193 throw OrthancException(ErrorCode_BadFileFormat); 1194 } 1195 1196 const std::string study = call.GetUriComponent("id", ""); 1197 1198 std::unique_ptr<MergeStudyJob> job(new MergeStudyJob(context, study)); 1199 job->SetOrigin(call); 1200 1201 std::vector<std::string> resources; 1202 SerializationToolbox::ReadArrayOfStrings(resources, request, RESOURCES); 1203 1204 for (size_t i = 0; i < resources.size(); i++) 1205 { 1206 job->AddSource(resources[i]); 1207 } 1208 1209 job->AddTrailingStep(); 1210 1211 SetKeepSource(*job, request); 1212 1213 OrthancRestApi::GetApi(call).SubmitCommandsJob 1214 (call, job.release(), true /* synchronous by default */, request); 1215 } 1216 1217 RegisterAnonymizeModify()1218 void OrthancRestApi::RegisterAnonymizeModify() 1219 { 1220 Register("/instances/{id}/modify", ModifyInstance); 1221 Register("/series/{id}/modify", ModifyResource<ResourceType_Series>); 1222 Register("/studies/{id}/modify", ModifyResource<ResourceType_Study>); 1223 Register("/patients/{id}/modify", ModifyResource<ResourceType_Patient>); 1224 Register("/tools/bulk-modify", BulkModify); 1225 1226 Register("/instances/{id}/anonymize", AnonymizeInstance); 1227 Register("/series/{id}/anonymize", AnonymizeResource<ResourceType_Series>); 1228 Register("/studies/{id}/anonymize", AnonymizeResource<ResourceType_Study>); 1229 Register("/patients/{id}/anonymize", AnonymizeResource<ResourceType_Patient>); 1230 Register("/tools/bulk-anonymize", BulkAnonymize); 1231 1232 Register("/tools/create-dicom", CreateDicom); 1233 1234 Register("/studies/{id}/split", SplitStudy); 1235 Register("/studies/{id}/merge", MergeStudy); 1236 } 1237 } 1238