* feat: basic blobstore infrastructure for dev * refactor: (broken) attempt to put minio console behind nginx * feat: initialize blobstore with boto3 * fix: abandon attempt to proxy minio. Use docker compose instead. * feat: beginning of blob writes * feat: storage utilities * feat: test buckets * chore: black * chore: remove unused import * chore: avoid f string when not needed * fix: inform all settings files about blobstores * fix: declare types for some settings * ci: point to new target base * ci: adjust test workflow * fix: give the tests debug environment a blobstore * fix: "better" name declarations * ci: use devblobstore container * chore: identify places to write to blobstorage * chore: remove unreachable code * feat: store materials * feat: store statements * feat: store status changes * feat: store liaison attachments * feat: store agendas provided with Interim session requests * chore: capture TODOs * feat: store polls and chatlogs * chore: remove unneeded TODO * feat: store drafts on submit and post * fix: handle storage during doc expiration and resurrection * fix: mirror an unlink * chore: add/refine TODOs * feat: store slide submissions * fix: structure slide test correctly * fix: correct sense of existence check * feat: store some indexes * feat: BlobShadowFileSystemStorage * feat: shadow floorplans / host logos to the blob * chore: remove unused import * feat: strip path from blob shadow names * feat: shadow photos / thumbs * refactor: combine photo and photothumb blob kinds The photos / thumbs were already dropped in the same directory, so let's not add a distinction at this point. * style: whitespace * refactor: use kwargs consistently * chore: migrations * refactor: better deconstruct(); rebuild migrations * fix: use new class in mack patch * chore: add TODO * feat: store group index documents * chore: identify more TODO * feat: store reviews * fix: repair merge * chore: remove unnecessary TODO * feat: StoredObject metadata * fix: deburr some debugging code * fix: only set the deleted timestamp once * chore: correct typo * fix: get_or_create vs get and test * fix: avoid the questionable is_seekable helper * chore: capture future design consideration * chore: blob store cfg for k8s * chore: black * chore: copyright * ci: bucket name prefix option + run Black Adds/uses DATATRACKER_BLOB_STORE_BUCKET_PREFIX option. Other changes are just Black styling. * ci: fix typo in bucket name expression * chore: parameters in app-configure-blobstore Allows use with other blob stores. * ci: remove verify=False option * fix: don't return value from __init__ * feat: option to log timing of S3Storage calls * chore: units * fix: deleted->null when storing a file * style: Black * feat: log as JSON; refactor to share code; handle exceptions * ci: add ietf_log_blob_timing option for k8s * test: --no-manage-blobstore option for running tests * test: use blob store settings from env, if set * test: actually set a couple more storage opts * feat: offswitch (#8541) * feat: offswitch * fix: apply ENABLE_BLOBSTORAGE to BlobShadowFileSystemStorage behavior * chore: log timing of blob reads * chore: import Config from botocore.config * chore(deps): import boto3-stubs / botocore botocore is implicitly imported, but make it explicit since we refer to it directly * chore: drop type annotation that mypy loudly ignores * refactor: add storage methods via mixin Shares code between Document and DocHistory without putting it in the base DocumentInfo class, which lacks the name field. Also makes mypy happy. * feat: add timeout / retry limit to boto client * ci: let k8s config the timeouts via env * chore: repair merge resolution typo * chore: tweak settings imports * chore: simplify k8s/settings_local.py imports --------- Co-authored-by: Jennifer Richards <jennifer@staff.ietf.org>
116 lines
4.1 KiB
Python
116 lines
4.1 KiB
Python
# Copyright The IETF Trust 2024, All Rights Reserved
|
|
#
|
|
# Celery task definitions
|
|
#
|
|
import shutil
|
|
|
|
from celery import shared_task
|
|
from pathlib import Path
|
|
|
|
from django.conf import settings
|
|
from django.template.loader import render_to_string
|
|
|
|
from ietf.doc.storage_utils import store_file
|
|
from ietf.utils import log
|
|
|
|
from .models import Group
|
|
from .utils import fill_in_charter_info, fill_in_wg_drafts, fill_in_wg_roles
|
|
from .views import extract_last_name, roles
|
|
|
|
|
|
@shared_task
|
|
def generate_wg_charters_files_task():
|
|
areas = Group.objects.filter(type="area", state="active").order_by("name")
|
|
groups = (
|
|
Group.objects.filter(type="wg", state="active")
|
|
.exclude(parent=None)
|
|
.order_by("acronym")
|
|
)
|
|
for group in groups:
|
|
fill_in_charter_info(group)
|
|
fill_in_wg_roles(group)
|
|
fill_in_wg_drafts(group)
|
|
for area in areas:
|
|
area.groups = [g for g in groups if g.parent_id == area.pk]
|
|
charter_path = Path(settings.CHARTER_PATH)
|
|
charters_file = charter_path / "1wg-charters.txt"
|
|
charters_file.write_text(
|
|
render_to_string("group/1wg-charters.txt", {"areas": areas}),
|
|
encoding="utf8",
|
|
)
|
|
charters_by_acronym_file = charter_path / "1wg-charters-by-acronym.txt"
|
|
charters_by_acronym_file.write_text(
|
|
render_to_string("group/1wg-charters-by-acronym.txt", {"groups": groups}),
|
|
encoding="utf8",
|
|
)
|
|
|
|
with charters_file.open("rb") as f:
|
|
store_file("indexes", "1wg-charters.txt", f, allow_overwrite=True)
|
|
with charters_by_acronym_file.open("rb") as f:
|
|
store_file("indexes", "1wg-charters-by-acronym.txt", f, allow_overwrite=True)
|
|
|
|
charter_copy_dests = [
|
|
getattr(settings, "CHARTER_COPY_PATH", None),
|
|
getattr(settings, "CHARTER_COPY_OTHER_PATH", None),
|
|
getattr(settings, "CHARTER_COPY_THIRD_PATH", None),
|
|
]
|
|
for charter_copy_dest in charter_copy_dests:
|
|
if charter_copy_dest is not None:
|
|
if not Path(charter_copy_dest).is_dir():
|
|
log.log(
|
|
f"Error copying 1wg-charter files to {charter_copy_dest}: it does not exist or is not a directory"
|
|
)
|
|
else:
|
|
try:
|
|
shutil.copy2(charters_file, charter_copy_dest)
|
|
except IOError as err:
|
|
log.log(f"Error copying {charters_file} to {charter_copy_dest}: {err}")
|
|
try:
|
|
shutil.copy2(charters_by_acronym_file, charter_copy_dest)
|
|
except IOError as err:
|
|
log.log(
|
|
f"Error copying {charters_by_acronym_file} to {charter_copy_dest}: {err}"
|
|
)
|
|
|
|
|
|
@shared_task
|
|
def generate_wg_summary_files_task():
|
|
# Active WGs (all should have a parent, but filter to be sure)
|
|
groups = (
|
|
Group.objects.filter(type="wg", state="active")
|
|
.exclude(parent=None)
|
|
.order_by("acronym")
|
|
)
|
|
# Augment groups with chairs list
|
|
for group in groups:
|
|
group.chairs = sorted(roles(group, "chair"), key=extract_last_name)
|
|
|
|
# Active areas with one or more active groups in them
|
|
areas = Group.objects.filter(
|
|
type="area",
|
|
state="active",
|
|
group__in=groups,
|
|
).distinct().order_by("name")
|
|
# Augment areas with their groups
|
|
for area in areas:
|
|
area.groups = [g for g in groups if g.parent_id == area.pk]
|
|
summary_path = Path(settings.GROUP_SUMMARY_PATH)
|
|
summary_file = summary_path / "1wg-summary.txt"
|
|
summary_file.write_text(
|
|
render_to_string("group/1wg-summary.txt", {"areas": areas}),
|
|
encoding="utf8",
|
|
)
|
|
summary_by_acronym_file = summary_path / "1wg-summary-by-acronym.txt"
|
|
summary_by_acronym_file.write_text(
|
|
render_to_string(
|
|
"group/1wg-summary-by-acronym.txt",
|
|
{"areas": areas, "groups": groups},
|
|
),
|
|
encoding="utf8",
|
|
)
|
|
|
|
with summary_file.open("rb") as f:
|
|
store_file("indexes", "1wg-summary.txt", f, allow_overwrite=True)
|
|
with summary_by_acronym_file.open("rb") as f:
|
|
store_file("indexes", "1wg-summary-by-acronym.txt", f, allow_overwrite=True)
|