Orthanc/OrthancFramework/Resources/EmbedResources.py
2025-06-23 19:07:37 +05:30

447 lines
12 KiB
Python
Executable File

#!/usr/bin/python
# Orthanc - A Lightweight, RESTful DICOM Store
# Copyright (C) 2012-2016 Sebastien Jodogne, Medical Physics
# Department, University Hospital of Liege, Belgium
# Copyright (C) 2017-2023 Osimis S.A., Belgium
# Copyright (C) 2024-2025 Orthanc Team SRL, Belgium
# Copyright (C) 2021-2025 Sebastien Jodogne, ICTEAM UCLouvain, Belgium
#
# This program is free software: you can redistribute it and/or
# modify it under the terms of the GNU Lesser 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
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this program. If not, see
# <http://www.gnu.org/licenses/>.
import sys
import os
import os.path
import pprint
import re
UPCASE_CHECK = True
USE_SYSTEM_EXCEPTION = False
EXCEPTION_CLASS = 'OrthancException'
OUT_OF_RANGE_EXCEPTION = '::Orthanc::OrthancException(::Orthanc::ErrorCode_ParameterOutOfRange)'
INEXISTENT_PATH_EXCEPTION = '::Orthanc::OrthancException(::Orthanc::ErrorCode_InexistentItem)'
NAMESPACE = 'Orthanc.EmbeddedResources'
FRAMEWORK_PATH = None
ARGS = []
for i in range(len(sys.argv)):
if not sys.argv[i].startswith('--'):
ARGS.append(sys.argv[i])
elif sys.argv[i].lower() == '--no-upcase-check':
UPCASE_CHECK = False
elif sys.argv[i].lower() == '--system-exception':
USE_SYSTEM_EXCEPTION = True
EXCEPTION_CLASS = '::std::runtime_error'
OUT_OF_RANGE_EXCEPTION = '%s("Parameter out of range")' % EXCEPTION_CLASS
INEXISTENT_PATH_EXCEPTION = '%s("Unknown path in a directory resource")' % EXCEPTION_CLASS
elif sys.argv[i].startswith('--namespace='):
NAMESPACE = sys.argv[i][sys.argv[i].find('=') + 1 : ]
elif sys.argv[i].startswith('--framework-path='):
FRAMEWORK_PATH = sys.argv[i][sys.argv[i].find('=') + 1 : ]
if len(ARGS) < 2 or len(ARGS) % 2 != 0:
print ('Usage:')
print ('python %s [--no-upcase-check] [--system-exception] [--namespace=<Namespace>] <TargetBaseFilename> [ <Name> <Source> ]*' % sys.argv[0])
exit(-1)
TARGET_BASE_FILENAME = ARGS[1]
SOURCES = ARGS[2:]
try:
# Make sure the destination directory exists
os.makedirs(os.path.normpath(os.path.join(TARGET_BASE_FILENAME, '..')))
except:
pass
#####################################################################
## Read each resource file
#####################################################################
def CheckNoUpcase(s):
global UPCASE_CHECK
if (UPCASE_CHECK and
re.search('[A-Z]', s) != None):
raise Exception("Path in a directory with an upcase letter: %s" % s)
resources = {}
counter = 0
i = 0
while i < len(SOURCES):
resourceName = SOURCES[i].upper()
pathName = SOURCES[i + 1]
if not os.path.exists(pathName):
raise Exception("Non existing path: %s" % pathName)
if resourceName in resources:
raise Exception("Twice the same resource: " + resourceName)
if os.path.isdir(pathName):
# The resource is a directory: Recursively explore its files
content = {}
for root, dirs, files in os.walk(pathName):
dirs.sort()
files.sort()
base = os.path.relpath(root, pathName)
# Fix issue #24 (Build fails on OSX when directory has .DS_Store files):
# Ignore folders whose name starts with a dot (".")
if base.find('/.') != -1:
print('Ignoring folder: %s' % root)
continue
for f in files:
if f.find('~') == -1: # Ignore Emacs backup files
if base == '.':
r = f
else:
r = os.path.join(base, f)
CheckNoUpcase(r)
r = '/' + r.replace('\\', '/')
if r in content:
raise Exception("Twice the same filename (check case): " + r)
content[r] = {
'Filename' : os.path.join(root, f),
'Index' : counter
}
counter += 1
resources[resourceName] = {
'Type' : 'Directory',
'Files' : content
}
elif os.path.isfile(pathName):
resources[resourceName] = {
'Type' : 'File',
'Index' : counter,
'Filename' : pathName
}
counter += 1
else:
raise Exception("Not a regular file, nor a directory: " + pathName)
i += 2
#pprint.pprint(resources)
#####################################################################
## Write .h header
#####################################################################
header = open(TARGET_BASE_FILENAME + '.h', 'w')
header.write("""
#pragma once
#include <string>
#include <list>
#if defined(_MSC_VER)
# pragma warning(disable: 4065) // "Switch statement contains 'default' but no 'case' labels"
#endif
""")
for ns in NAMESPACE.split('.'):
header.write('namespace %s {\n' % ns)
header.write("""
enum FileResourceId
{
""")
isFirst = True
for name in resources:
if resources[name]['Type'] == 'File':
if isFirst:
isFirst = False
else:
header.write(',\n')
header.write(' %s' % name)
header.write("""
};
enum DirectoryResourceId
{
""")
isFirst = True
for name in resources:
if resources[name]['Type'] == 'Directory':
if isFirst:
isFirst = False
else:
header.write(',\n')
header.write(' %s' % name)
header.write("""
};
const void* GetFileResourceBuffer(FileResourceId id);
size_t GetFileResourceSize(FileResourceId id);
void GetFileResource(std::string& result, FileResourceId id);
const void* GetDirectoryResourceBuffer(DirectoryResourceId id, const char* path);
size_t GetDirectoryResourceSize(DirectoryResourceId id, const char* path);
void GetDirectoryResource(std::string& result, DirectoryResourceId id, const char* path);
void ListResources(std::list<std::string>& result, DirectoryResourceId id);
""")
for ns in NAMESPACE.split('.'):
header.write('}\n')
header.close()
#####################################################################
## Write the resource content in the .cpp source
#####################################################################
PYTHON_MAJOR_VERSION = sys.version_info[0]
def WriteResource(cpp, item):
cpp.write(' static const uint8_t resource%dBuffer[] = {' % item['Index'])
f = open(item['Filename'], "rb")
content = f.read()
f.close()
# http://stackoverflow.com/a/1035360
pos = 0
buffer = [] # instead of appending a few bytes at a time to the cpp file,
# we first append each chunk to a list, join it and write it
# to the file. We've measured that it was 2-3 times faster in python3.
# Note that speed is important since if generation is too slow,
# cmake might try to compile the EmbeddedResources.cpp file while it is
# still being generated !
for b in content:
if PYTHON_MAJOR_VERSION == 2:
c = ord(b[0])
else:
c = b
if pos > 0:
buffer.append(",")
if (pos % 16) == 0:
buffer.append("\n")
if c < 0:
raise Exception("Internal error")
buffer.append("0x%02x" % c)
pos += 1
cpp.write("".join(buffer))
# Zero-size array are disallowed, so we put one single void character in it.
if pos == 0:
cpp.write(' 0')
cpp.write(' };\n')
cpp.write(' static const size_t resource%dSize = %d;\n' % (item['Index'], pos))
cpp = open(TARGET_BASE_FILENAME + '.cpp', 'w')
cpp.write('#include "%s.h"\n' % os.path.basename(TARGET_BASE_FILENAME))
if USE_SYSTEM_EXCEPTION:
cpp.write('#include <stdexcept>')
elif FRAMEWORK_PATH != None:
cpp.write('#include "%s/OrthancException.h"' % FRAMEWORK_PATH)
else:
cpp.write('#include <OrthancException.h>')
cpp.write("""
#include <stdint.h>
#include <string.h>
""")
for ns in NAMESPACE.split('.'):
cpp.write('namespace %s {\n' % ns)
for name in resources:
if resources[name]['Type'] == 'File':
WriteResource(cpp, resources[name])
else:
for f in resources[name]['Files']:
WriteResource(cpp, resources[name]['Files'][f])
#####################################################################
## Write the accessors to the file resources in .cpp
#####################################################################
cpp.write("""
const void* GetFileResourceBuffer(FileResourceId id)
{
switch (id)
{
""")
for name in resources:
if resources[name]['Type'] == 'File':
cpp.write(' case %s:\n' % name)
cpp.write(' return resource%dBuffer;\n' % resources[name]['Index'])
cpp.write("""
default:
throw %s;
}
}
size_t GetFileResourceSize(FileResourceId id)
{
switch (id)
{
""" % OUT_OF_RANGE_EXCEPTION)
for name in resources:
if resources[name]['Type'] == 'File':
cpp.write(' case %s:\n' % name)
cpp.write(' return resource%dSize;\n' % resources[name]['Index'])
cpp.write("""
default:
throw %s;
}
}
""" % OUT_OF_RANGE_EXCEPTION)
#####################################################################
## Write the accessors to the directory resources in .cpp
#####################################################################
cpp.write("""
const void* GetDirectoryResourceBuffer(DirectoryResourceId id, const char* path)
{
switch (id)
{
""")
for name in resources:
if resources[name]['Type'] == 'Directory':
cpp.write(' case %s:\n' % name)
isFirst = True
for path in resources[name]['Files']:
cpp.write(' if (!strcmp(path, "%s"))\n' % path)
cpp.write(' return resource%dBuffer;\n' % resources[name]['Files'][path]['Index'])
cpp.write(' throw %s;\n\n' % INEXISTENT_PATH_EXCEPTION)
cpp.write(""" default:
throw %s;
}
}
size_t GetDirectoryResourceSize(DirectoryResourceId id, const char* path)
{
switch (id)
{
""" % OUT_OF_RANGE_EXCEPTION)
for name in resources:
if resources[name]['Type'] == 'Directory':
cpp.write(' case %s:\n' % name)
isFirst = True
for path in resources[name]['Files']:
cpp.write(' if (!strcmp(path, "%s"))\n' % path)
cpp.write(' return resource%dSize;\n' % resources[name]['Files'][path]['Index'])
cpp.write(' throw %s;\n\n' % INEXISTENT_PATH_EXCEPTION)
cpp.write(""" default:
throw %s;
}
}
""" % OUT_OF_RANGE_EXCEPTION)
#####################################################################
## List the resources in a directory
#####################################################################
cpp.write("""
void ListResources(std::list<std::string>& result, DirectoryResourceId id)
{
result.clear();
switch (id)
{
""")
for name in resources:
if resources[name]['Type'] == 'Directory':
cpp.write(' case %s:\n' % name)
for path in sorted(resources[name]['Files']):
cpp.write(' result.push_back("%s");\n' % path)
cpp.write(' break;\n\n')
cpp.write(""" default:
throw %s;
}
}
""" % OUT_OF_RANGE_EXCEPTION)
#####################################################################
## Write the convenience wrappers in .cpp
#####################################################################
cpp.write("""
void GetFileResource(std::string& result, FileResourceId id)
{
size_t size = GetFileResourceSize(id);
result.resize(size);
if (size > 0)
memcpy(&result[0], GetFileResourceBuffer(id), size);
}
void GetDirectoryResource(std::string& result, DirectoryResourceId id, const char* path)
{
size_t size = GetDirectoryResourceSize(id, path);
result.resize(size);
if (size > 0)
memcpy(&result[0], GetDirectoryResourceBuffer(id, path), size);
}
""")
for ns in NAMESPACE.split('.'):
cpp.write('}\n')
cpp.close()