Skip to content

Commit

Permalink
Merge branch 'develop' into feature/arche-import
Browse files Browse the repository at this point in the history
  • Loading branch information
BernhardKoschicek committed Oct 18, 2023
2 parents 0e6d577 + d4eb118 commit 600b646
Show file tree
Hide file tree
Showing 25 changed files with 545 additions and 145 deletions.
11 changes: 8 additions & 3 deletions .github/workflows/starter.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -112,9 +112,14 @@ jobs:
WTF_CSRF_METHODS: list[str] = []
ARCHE = {
'id': 0,
'collection_ids': [0],
'base_url': 'https://arche-curation.acdh-dev.oeaw.ac.at/',
'thumbnail_url': 'https://arche-thumbnails.acdh.oeaw.ac.at/'}
'url': 'https://arche-curation.acdh-dev.oeaw.ac.at/'}
IIIF = {
'enabled': True,
'path': '/var/www/iipsrv/',
'url': 'http://localhost:8080/iiif/',
'version': 2,
'conversion': True,
'compression': 'jpeg'}
EOF
# production.py is recreated on every start
sudo chown 33:33 testing.py
Expand Down
File renamed without changes.
25 changes: 11 additions & 14 deletions config/default.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,33 +23,30 @@
'es': 'Español',
'fr': 'Français'}

# Files with these extensions are can be displayed in the browser
DISPLAY_FILE_EXTENSIONS = \
['.bmp', '.gif', '.ico', '.jpeg', '.jpg', '.png', '.svg']

# Paths are implemented operating system independent using pathlib.
# To override them (in instance/production.py) either use them like here
# or use absolute paths like e.g. pathlib.Path('/some/location/somewhere')
FILES_PATH = Path(__file__).parent.parent / 'files'
EXPORT_PATH = Path(FILES_PATH) / 'export'
UPLOAD_PATH = Path(FILES_PATH) / 'uploads'
TMP_PATH = Path('/tmp') # used e.g. for processing imports and export files
TMP_PATH = Path('/tmp') # For processing files e.g. at import and export

# Image processing
DISPLAY_FILE_EXT = ['.bmp', '.gif', '.ico', '.jpeg', '.jpg', '.png', '.svg']
PROCESSABLE_EXT = ['.tiff', '.tif']
PROCESSED_EXT = '.jpeg'
PROCESSED_IMAGE_PATH = Path(FILES_PATH) / 'processed_images'
RESIZED_IMAGES = Path(PROCESSED_IMAGE_PATH) / 'resized'
IMAGE_SIZE = {
'thumbnail': '200',
'table': '100'}
NONE_DISPLAY_EXT = ['.tiff', '.tif']
ALLOWED_IMAGE_EXT = DISPLAY_FILE_EXTENSIONS + NONE_DISPLAY_EXT
PROCESSED_EXT = '.jpeg'

# For system checks
WRITABLE_PATHS = [
UPLOAD_PATH,
EXPORT_PATH,
RESIZED_IMAGES]
IIIF = {
'enabled': False,
'path': '',
'url': '',
'version': 2,
'conversion': True,
'compression': 'deflate'} # 'deflate' or 'jpeg'

# Security
SESSION_COOKIE_SECURE = False # Should be True in production.py if using HTTPS
Expand Down
8 changes: 8 additions & 0 deletions install/Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,16 @@ RUN --mount=type=cache,target=/var/cache/apt \
apt install -y --no-install-recommends python3-pandas python3-jinja2 python3-flask-cors python3-flask-restful p7zip-full &&\
apt install -y --no-install-recommends python3-wand python3-rdflib python3-requests python3-dicttoxml python3-rdflib-jsonld python3-flasgger &&\
apt install -y --no-install-recommends apache2 libapache2-mod-wsgi-py3 python3-coverage python3-nose exiftran &&\
apt install -y --no-install-recommends iipimage-server libvips-tools &&\
apt install -y --no-install-recommends gettext npm python3-pip git postgresql-client-13 &&\
apt install -y --no-install-recommends dos2unix locales locales-all &&\
mkdir -p /var/www/openatlas /var/www/.cache /var/www/.local /var/www/.npm &&\
chown -R www-data:www-data /var/www/.cache /var/www/.local /var/www/.npm /var/log/apache2 /var/run/apache2
RUN cp -rp /usr/lib/iipimage-server/ /var/www/iipsrv/ &&\
chown -R www-data /var/www/iipsrv/ &&\
chmod 777 -R /var/www/iipsrv/
RUN rm /etc/apache2/mods-available/iipsrv.conf
COPY /install/iipsrv.conf /etc/apache2/mods-available/iipsrv.conf
COPY --chown=www-data:www-data / /var/www/openatlas/
RUN cd /var/www/openatlas && cp install/entrypoint.sh /entrypoint.sh &&\
cp install/example_apache.conf /etc/apache2/sites-available/000-default.conf &&\
Expand All @@ -26,6 +32,8 @@ RUN cd /var/www/openatlas &&\
cd openatlas/static &&\
pip3 install -e ./ &&\
~/.local/bin/calmjs npm --install openatlas
RUN a2enmod iipsrv &&\
service apache2 restart

EXPOSE 8080
ENV APACHE_CONFDIR /etc/apache2
Expand Down
34 changes: 34 additions & 0 deletions install/iipsrv.conf
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
# Create a directory for the iipsrv binary
ScriptAlias /iiif "/var/www/iipsrv/iipsrv.fcgi"

# Set the options on that directory
<Location "iipsrv/">
AllowOverride None
Options None
<IfModule mod_version.c>
<IfVersion < 2.4>
Order allow,deny
Allow from all
</IfVersion>
<IfVersion >= 2.4>
Require all granted
</IfVersion>
</IfModule>
# Set the module handler
AddHandler fcgid-script .fcgi
</Location>

# Set our environment variables for the IIP server
FcgidInitialEnv VERBOSITY "6"
FcgidInitialEnv LOGFILE "/var/log/iipsrv.log"
FcgidInitialEnv MAX_IMAGE_CACHE_SIZE "10"
FcgidInitialEnv JPEG_QUALITY "90"
FcgidInitialEnv MAX_CVT "5000"
FcgidInitialEnv MEMCACHED_SERVERS "localhost"
FcgidInitialEnv CORS "*"
FcgidInitialEnv URI_MAP "iiif=>IIIF"

# Define the idle timeout as unlimited and the number of
# processes we want
FcgidIdleTimeout 0
FcgidMaxProcessesPerClass 1
20 changes: 20 additions & 0 deletions install/upgrade/upgrade.md
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,26 @@ then run the database upgrade script, then restart Apache:
sudo python3 install/upgrade/database_upgrade.py
sudo service apache2 restart

### 7.16.x to 7.17.0

#### Configuration

The configuration was refactored, especially path parameters. You should
compare the instance/production.py with config/default.py in case you made
adaptions to these. The same goes for instance/tests.py in case you are also
using tests.

#### IIIF
For the IIIF implementation new NPM packages are needed:

$ cd openatlas/static
$ rm package.json
$ pip3 install -e ./
$ ~/.local/bin/calmjs npm --install openatlas

If you want to use IIIF, please read the
[instructions](https://redmine.openatlas.eu/projects/uni/wiki/IIIF).

### 7.16.0 to 7.16.1
A code base update (e.g. with git pull) and a webserver restart is sufficient.

Expand Down
14 changes: 10 additions & 4 deletions openatlas/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@
app: Flask = Flask(__name__, instance_relative_config=True)
csrf = CSRFProtect(app) # Make sure all forms are CSRF protected
app.config.from_object('config.default')
app.config.from_object('config.api_config')
app.config.from_object('config.api')
app.config.from_pyfile('production.py')
app.config['WTF_CSRF_TIME_LIMIT'] = None # Set CSRF token valid for session

Expand Down Expand Up @@ -69,10 +69,16 @@ def before_request() -> None:
for file_ in app.config['UPLOAD_PATH'].iterdir():
if file_.stem.isdigit():
g.files[int(file_.stem)] = file_
# Set max file upload in MB
app.config['MAX_CONTENT_LENGTH'] = \
g.settings['file_upload_max_size'] * 1024 * 1024

g.settings['file_upload_max_size'] * 1024 * 1024 # Max upload in MB
g.display_file_ext = app.config['DISPLAY_FILE_EXT']
if g.settings['image_processing']:
g.display_file_ext += app.config['PROCESSABLE_EXT']
g.writable_paths = [
app.config['EXPORT_PATH'],
app.config['RESIZED_IMAGES'],
app.config['UPLOAD_PATH'],
app.config['TMP_PATH']]
if request.path.startswith('/api/'):
ip = request.environ.get('HTTP_X_REAL_IP', request.remote_addr)
if not current_user.is_authenticated \
Expand Down
148 changes: 148 additions & 0 deletions openatlas/api/endpoints/iiif.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
from typing import Any

import requests
from flask import jsonify, Response, url_for, g
from flask_restful import Resource

from openatlas import app
from openatlas.api.resources.model_mapper import get_entity_by_id
from openatlas.api.resources.util import get_license_name
from openatlas.models.entity import Entity


class IIIFSequenceV2(Resource):
@staticmethod
def get(id_: int) -> Response:
return jsonify(
{"@context": "https://iiif.io/api/presentation/2/context.json"} |
IIIFSequenceV2.build_sequence(get_metadata(get_entity_by_id(id_))))

@staticmethod
def build_sequence(metadata: dict[str, Any]) -> dict[str, Any]:
return {
"@id": url_for(
'api.iiif_sequence',
id_=metadata['entity'].id,
version=2,
_external=True),
"@type": "sc:Sequence",
"label": [{
"@value": "Normal Sequence",
"@language": "en"}],
"canvases": [
IIIFCanvasV2.build_canvas(metadata)]}


class IIIFCanvasV2(Resource):
@staticmethod
def get(id_: int) -> Response:
return jsonify(
{"@context": "https://iiif.io/api/presentation/2/context.json"} |
IIIFCanvasV2.build_canvas(get_metadata(get_entity_by_id(id_))))

@staticmethod
def build_canvas(metadata: dict[str, Any]) -> dict[str, Any]:
entity = metadata['entity']
return {
"@id": url_for(
'api.iiif_canvas', id_=entity.id, version=2, _external=True),
"@type": "sc:Canvas",
"label": entity.name,
"height": metadata['img_api']['height'],
"width": metadata['img_api']['width'],
"description": {
"@value": entity.description,
"@language": "en"},
"images": [IIIFImageV2.build_image(metadata)],
"related": "",
"thumbnail": {
"@id": f'{metadata["img_url"]}/full/!200,200/0/default.jpg',
"@type": "dctypes:Image",
"format": "image/jpeg",
"height": 200,
"width": 200,
"service": {
"@context": "https://iiif.io/api/image/2/context.json",
"@id": metadata['img_url'],
"profile": metadata['img_api']['profile']}}}


class IIIFImageV2(Resource):
@staticmethod
def get(id_: int) -> Response:
return jsonify(
IIIFImageV2.build_image(get_metadata(get_entity_by_id(id_))))

@staticmethod
def build_image(metadata: dict[str, Any]) -> dict[str, Any]:
id_ = metadata['entity'].id
return {
"@context": "https://iiif.io/api/presentation/2/context.json",
"@id":
url_for('api.iiif_image', id_=id_, version=2, _external=True),
"@type": "oa:Annotation",
"motivation": "sc:painting",
"resource": {
"@id": metadata['img_url'],
"@type": "dctypes:Image",
"format": "image/jpeg",
"service": {
"@context": "https://iiif.io/api/image/2/context.json",
"@id": metadata['img_url'],
"profile": metadata['img_api']['profile']},
"height": metadata['img_api']['height'],
"width": metadata['img_api']['width']},
"on":
url_for('api.iiif_canvas', id_=id_, version=2, _external=True)}


class IIIFManifest(Resource):
@staticmethod
def get(version: int, id_: int) -> Response:
operation = getattr(IIIFManifest, f'get_manifest_version_{version}')
return jsonify(operation(id_))

@staticmethod
def get_manifest_version_2(id_: int) -> dict[str, Any]:
entity = get_entity_by_id(id_)
return {
"@context": "https://iiif.io/api/presentation/2/context.json",
"@id":
url_for(
'api.iiif_manifest',
id_=id_,
version=2,
_external=True),
"@type": "sc:Manifest",
"label": entity.name,
"metadata": [],
"description": [{
"@value": entity.description,
"@language": "en"}],
"license": get_license_name(entity),
"attribution": "By OpenAtlas",
"logo": get_logo(),
"sequences": [
IIIFSequenceV2.build_sequence(get_metadata(entity))],
"structures": []}


def get_metadata(entity: Entity) -> dict[str, Any]:
ext = '.tiff' if app.config['IIIF']['conversion'] \
else entity.get_file_ext()
image_url = f"{app.config['IIIF']['url']}{entity.id}{ext}"
req = requests.get(f"{image_url}/info.json")
image_api = req.json()
return {'entity': entity, 'img_url': image_url, 'img_api': image_api}


def get_logo() -> dict[str, Any]:
return {
"@id": url_for(
'api.display',
filename=g.settings['logo_file_id'],
_external=True),
"service": {
"@context": "https://iiif.io/api/image/2/context.json",
"@id": url_for('overview', _external=True),
"profile": "https://iiif.io/api/image/2/level2.json"}}
8 changes: 4 additions & 4 deletions openatlas/api/import_scripts/vocabs.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,8 +27,8 @@ def fetch_top_groups(
duplicates = []
if ref := get_vocabs_reference_system(details):
for group in form_data['choices']:
if (not Type.check_hierarchy_exists(group[1]) and
group[0] in form_data['top_concepts']):
if not Type.check_hierarchy_exists(group[1]) and \
group[0] in form_data['top_concepts']:
hierarchy = Entity.insert(
'type',
group[1],
Expand All @@ -47,8 +47,8 @@ def fetch_top_groups(
ref,
hierarchy)
count.append(group[0])
if (Type.check_hierarchy_exists(group[1])
and group[0] in form_data['top_concepts']):
if Type.check_hierarchy_exists(group[1]) \
and group[0] in form_data['top_concepts']:
duplicates.append(group[1])
return count, duplicates

Expand Down
27 changes: 23 additions & 4 deletions openatlas/api/routes.py
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
from flask_restful import Api

from openatlas.api.endpoints.iiif import \
(IIIFManifest, IIIFImageV2, IIIFCanvasV2, IIIFSequenceV2)
from openatlas.api.endpoints.content import ClassMapping, \
GetContent, SystemClassCount
from openatlas.api.endpoints.special import GetGeometricEntities, \
ExportDatabase, GetSubunits
from openatlas.api.endpoints.display_image import DisplayImage, LicensedFileOverview
from openatlas.api.endpoints.display_image import \
(DisplayImage, LicensedFileOverview)
from openatlas.api.endpoints.entities import GetByCidocClass, \
GetBySystemClass, GetByViewClass, GetEntitiesLinkedToEntity, GetEntity, \
GetLatest, GetQuery, GetTypeEntities, GetTypeEntitiesAll
from openatlas.api.endpoints.type import GetTypeByViewClass, \
GetTypeOverview, \
GetTypeTree
from openatlas.api.endpoints.type import \
(GetTypeByViewClass, GetTypeOverview, GetTypeTree)


def add_routes_v03(api: Api) -> None:
Expand Down Expand Up @@ -97,3 +99,20 @@ def add_routes_v03(api: Api) -> None:
LicensedFileOverview,
'/licensed_file_overview/',
endpoint='licensed_file_overview')

api.add_resource(
IIIFManifest,
'/iiif_manifest/<int:version>/<int:id_>',
endpoint='iiif_manifest')
api.add_resource(
IIIFImageV2,
'/iiif_image/<int:id_>.json',
endpoint='iiif_image')
api.add_resource(
IIIFCanvasV2,
'/iiif_canvas/<int:id_>.json',
endpoint='iiif_canvas')
api.add_resource(
IIIFSequenceV2,
'/iiif_sequence/<int:id_>.json',
endpoint='iiif_sequence')
Loading

0 comments on commit 600b646

Please sign in to comment.