analysis/webservice/redirect/RemoteSDAPCache.py (41 lines of code) (raw):
# Licensed to the Apache Software Foundation (ASF) under one or more
# contributor license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright ownership.
# The ASF licenses this file to You under the Apache License, Version 2.0
# (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import requests
import logging
from datetime import datetime
from datetime import timedelta
from dataclasses import dataclass
logger = logging.getLogger(__name__)
@dataclass
class RemoteSDAPList:
list: dict
outdated_at: datetime
class CollectionNotFound(Exception):
pass
class RemoteSDAPCache:
def __init__(self):
self.sdap_lists = {}
def _add(self, url, timeout=2, max_age=3600*24):
list_url = f"{url}/list"
try:
try:
r = requests.get(list_url, timeout=timeout)
except Exception as e:
raise CollectionNotFound(f'URL {list_url} was not reachable')
if r.status_code == 200:
logger.info("Caching list for sdap %s: %s", list_url, r.text)
self.sdap_lists[url] = RemoteSDAPList(
list=r.json(),
outdated_at=datetime.now()+timedelta(seconds=max_age)
)
else:
raise CollectionNotFound(f"url {list_url} was not reachable, responded with status {r.status_code}")
except CollectionNotFound as e:
logger.warning(e)
def get(self, url, short_name):
stripped_url = url.strip('/')
if stripped_url not in self.sdap_lists or self.sdap_lists[stripped_url].outdated_at>datetime.now():
self._add(stripped_url)
if stripped_url in self.sdap_lists:
for collection in self.sdap_lists[stripped_url].list:
if 'shortName' in collection and collection['shortName'] == short_name:
return collection
raise CollectionNotFound(f"collection {short_name} has not been found in url {stripped_url}")