forked from github/plane
e1ad385688
* fix: issue exports in self hosted instances * dev: remove print logs * dev: update url creation function * fix: changed the presigned url for self hosted exports --------- Co-authored-by: NarayanBavisetti <narayan3119@gmail.com>
376 lines
12 KiB
Python
376 lines
12 KiB
Python
# Python imports
|
|
import csv
|
|
import io
|
|
import json
|
|
import boto3
|
|
import zipfile
|
|
from urllib.parse import urlparse, urlunparse
|
|
|
|
# Django imports
|
|
from django.conf import settings
|
|
from django.utils import timezone
|
|
|
|
# Third party imports
|
|
from celery import shared_task
|
|
from sentry_sdk import capture_exception
|
|
from botocore.client import Config
|
|
from openpyxl import Workbook
|
|
|
|
# Module imports
|
|
from plane.db.models import Issue, ExporterHistory
|
|
|
|
|
|
def dateTimeConverter(time):
|
|
if time:
|
|
return time.strftime("%a, %d %b %Y %I:%M:%S %Z%z")
|
|
|
|
|
|
def dateConverter(time):
|
|
if time:
|
|
return time.strftime("%a, %d %b %Y")
|
|
|
|
|
|
def create_csv_file(data):
|
|
csv_buffer = io.StringIO()
|
|
csv_writer = csv.writer(csv_buffer, delimiter=",", quoting=csv.QUOTE_ALL)
|
|
|
|
for row in data:
|
|
csv_writer.writerow(row)
|
|
|
|
csv_buffer.seek(0)
|
|
return csv_buffer.getvalue()
|
|
|
|
|
|
def create_json_file(data):
|
|
return json.dumps(data)
|
|
|
|
|
|
def create_xlsx_file(data):
|
|
workbook = Workbook()
|
|
sheet = workbook.active
|
|
|
|
for row in data:
|
|
sheet.append(row)
|
|
|
|
xlsx_buffer = io.BytesIO()
|
|
workbook.save(xlsx_buffer)
|
|
xlsx_buffer.seek(0)
|
|
return xlsx_buffer.getvalue()
|
|
|
|
|
|
def create_zip_file(files):
|
|
zip_buffer = io.BytesIO()
|
|
with zipfile.ZipFile(zip_buffer, "w", zipfile.ZIP_DEFLATED) as zipf:
|
|
for filename, file_content in files:
|
|
zipf.writestr(filename, file_content)
|
|
|
|
zip_buffer.seek(0)
|
|
return zip_buffer
|
|
|
|
|
|
def upload_to_s3(zip_file, workspace_id, token_id, slug):
|
|
file_name = f"{workspace_id}/export-{slug}-{token_id[:6]}-{timezone.now()}.zip"
|
|
expires_in = 7 * 24 * 60 * 60
|
|
|
|
if settings.DOCKERIZED and settings.USE_MINIO:
|
|
s3 = boto3.client(
|
|
"s3",
|
|
endpoint_url=settings.AWS_S3_ENDPOINT_URL,
|
|
aws_access_key_id=settings.AWS_ACCESS_KEY_ID,
|
|
aws_secret_access_key=settings.AWS_SECRET_ACCESS_KEY,
|
|
config=Config(signature_version="s3v4"),
|
|
)
|
|
s3.upload_fileobj(
|
|
zip_file,
|
|
settings.AWS_STORAGE_BUCKET_NAME,
|
|
file_name,
|
|
ExtraArgs={"ACL": "public-read", "ContentType": "application/zip"},
|
|
)
|
|
presigned_url = s3.generate_presigned_url(
|
|
"get_object",
|
|
Params={"Bucket": settings.AWS_STORAGE_BUCKET_NAME, "Key": file_name},
|
|
ExpiresIn=expires_in,
|
|
)
|
|
# Create the new url with updated domain and protocol
|
|
presigned_url = presigned_url.replace(
|
|
"http://plane-minio:9000/uploads/",
|
|
f"{settings.AWS_S3_URL_PROTOCOL}//{settings.AWS_S3_CUSTOM_DOMAIN}/",
|
|
)
|
|
else:
|
|
s3 = boto3.client(
|
|
"s3",
|
|
region_name=settings.AWS_REGION,
|
|
aws_access_key_id=settings.AWS_ACCESS_KEY_ID,
|
|
aws_secret_access_key=settings.AWS_SECRET_ACCESS_KEY,
|
|
config=Config(signature_version="s3v4"),
|
|
)
|
|
s3.upload_fileobj(
|
|
zip_file,
|
|
settings.AWS_S3_BUCKET_NAME,
|
|
file_name,
|
|
ExtraArgs={"ACL": "public-read", "ContentType": "application/zip"},
|
|
)
|
|
|
|
presigned_url = s3.generate_presigned_url(
|
|
"get_object",
|
|
Params={"Bucket": settings.AWS_S3_BUCKET_NAME, "Key": file_name},
|
|
ExpiresIn=expires_in,
|
|
)
|
|
|
|
exporter_instance = ExporterHistory.objects.get(token=token_id)
|
|
|
|
if presigned_url:
|
|
exporter_instance.url = presigned_url
|
|
exporter_instance.status = "completed"
|
|
exporter_instance.key = file_name
|
|
else:
|
|
exporter_instance.status = "failed"
|
|
|
|
exporter_instance.save(update_fields=["status", "url", "key"])
|
|
|
|
|
|
def generate_table_row(issue):
|
|
return [
|
|
f"""{issue["project__identifier"]}-{issue["sequence_id"]}""",
|
|
issue["project__name"],
|
|
issue["name"],
|
|
issue["description_stripped"],
|
|
issue["state__name"],
|
|
issue["priority"],
|
|
f"{issue['created_by__first_name']} {issue['created_by__last_name']}"
|
|
if issue["created_by__first_name"] and issue["created_by__last_name"]
|
|
else "",
|
|
f"{issue['assignees__first_name']} {issue['assignees__last_name']}"
|
|
if issue["assignees__first_name"] and issue["assignees__last_name"]
|
|
else "",
|
|
issue["labels__name"],
|
|
issue["issue_cycle__cycle__name"],
|
|
dateConverter(issue["issue_cycle__cycle__start_date"]),
|
|
dateConverter(issue["issue_cycle__cycle__end_date"]),
|
|
issue["issue_module__module__name"],
|
|
dateConverter(issue["issue_module__module__start_date"]),
|
|
dateConverter(issue["issue_module__module__target_date"]),
|
|
dateTimeConverter(issue["created_at"]),
|
|
dateTimeConverter(issue["updated_at"]),
|
|
dateTimeConverter(issue["completed_at"]),
|
|
dateTimeConverter(issue["archived_at"]),
|
|
]
|
|
|
|
|
|
def generate_json_row(issue):
|
|
return {
|
|
"ID": f"""{issue["project__identifier"]}-{issue["sequence_id"]}""",
|
|
"Project": issue["project__name"],
|
|
"Name": issue["name"],
|
|
"Description": issue["description_stripped"],
|
|
"State": issue["state__name"],
|
|
"Priority": issue["priority"],
|
|
"Created By": f"{issue['created_by__first_name']} {issue['created_by__last_name']}"
|
|
if issue["created_by__first_name"] and issue["created_by__last_name"]
|
|
else "",
|
|
"Assignee": f"{issue['assignees__first_name']} {issue['assignees__last_name']}"
|
|
if issue["assignees__first_name"] and issue["assignees__last_name"]
|
|
else "",
|
|
"Labels": issue["labels__name"],
|
|
"Cycle Name": issue["issue_cycle__cycle__name"],
|
|
"Cycle Start Date": dateConverter(issue["issue_cycle__cycle__start_date"]),
|
|
"Cycle End Date": dateConverter(issue["issue_cycle__cycle__end_date"]),
|
|
"Module Name": issue["issue_module__module__name"],
|
|
"Module Start Date": dateConverter(issue["issue_module__module__start_date"]),
|
|
"Module Target Date": dateConverter(issue["issue_module__module__target_date"]),
|
|
"Created At": dateTimeConverter(issue["created_at"]),
|
|
"Updated At": dateTimeConverter(issue["updated_at"]),
|
|
"Completed At": dateTimeConverter(issue["completed_at"]),
|
|
"Archived At": dateTimeConverter(issue["archived_at"]),
|
|
}
|
|
|
|
|
|
def update_json_row(rows, row):
|
|
matched_index = next(
|
|
(
|
|
index
|
|
for index, existing_row in enumerate(rows)
|
|
if existing_row["ID"] == row["ID"]
|
|
),
|
|
None,
|
|
)
|
|
|
|
if matched_index is not None:
|
|
existing_assignees, existing_labels = (
|
|
rows[matched_index]["Assignee"],
|
|
rows[matched_index]["Labels"],
|
|
)
|
|
assignee, label = row["Assignee"], row["Labels"]
|
|
|
|
if assignee is not None and assignee not in existing_assignees:
|
|
rows[matched_index]["Assignee"] += f", {assignee}"
|
|
if label is not None and label not in existing_labels:
|
|
rows[matched_index]["Labels"] += f", {label}"
|
|
else:
|
|
rows.append(row)
|
|
|
|
|
|
def update_table_row(rows, row):
|
|
matched_index = next(
|
|
(index for index, existing_row in enumerate(rows) if existing_row[0] == row[0]),
|
|
None,
|
|
)
|
|
|
|
if matched_index is not None:
|
|
existing_assignees, existing_labels = rows[matched_index][7:9]
|
|
assignee, label = row[7:9]
|
|
|
|
if assignee is not None and assignee not in existing_assignees:
|
|
rows[matched_index][7] += f", {assignee}"
|
|
if label is not None and label not in existing_labels:
|
|
rows[matched_index][8] += f", {label}"
|
|
else:
|
|
rows.append(row)
|
|
|
|
|
|
def generate_csv(header, project_id, issues, files):
|
|
"""
|
|
Generate CSV export for all the passed issues.
|
|
"""
|
|
rows = [
|
|
header,
|
|
]
|
|
for issue in issues:
|
|
row = generate_table_row(issue)
|
|
update_table_row(rows, row)
|
|
csv_file = create_csv_file(rows)
|
|
files.append((f"{project_id}.csv", csv_file))
|
|
|
|
|
|
def generate_json(header, project_id, issues, files):
|
|
rows = []
|
|
for issue in issues:
|
|
row = generate_json_row(issue)
|
|
update_json_row(rows, row)
|
|
json_file = create_json_file(rows)
|
|
files.append((f"{project_id}.json", json_file))
|
|
|
|
|
|
def generate_xlsx(header, project_id, issues, files):
|
|
rows = [header]
|
|
for issue in issues:
|
|
row = generate_table_row(issue)
|
|
update_table_row(rows, row)
|
|
xlsx_file = create_xlsx_file(rows)
|
|
files.append((f"{project_id}.xlsx", xlsx_file))
|
|
|
|
|
|
@shared_task
|
|
def issue_export_task(provider, workspace_id, project_ids, token_id, multiple, slug):
|
|
try:
|
|
exporter_instance = ExporterHistory.objects.get(token=token_id)
|
|
exporter_instance.status = "processing"
|
|
exporter_instance.save(update_fields=["status"])
|
|
|
|
workspace_issues = (
|
|
(
|
|
Issue.objects.filter(
|
|
workspace__id=workspace_id,
|
|
project_id__in=project_ids,
|
|
project__project_projectmember__member=exporter_instance.initiated_by_id,
|
|
)
|
|
.select_related("project", "workspace", "state", "parent", "created_by")
|
|
.prefetch_related(
|
|
"assignees", "labels", "issue_cycle__cycle", "issue_module__module"
|
|
)
|
|
.values(
|
|
"id",
|
|
"project__identifier",
|
|
"project__name",
|
|
"project__id",
|
|
"sequence_id",
|
|
"name",
|
|
"description_stripped",
|
|
"priority",
|
|
"state__name",
|
|
"created_at",
|
|
"updated_at",
|
|
"completed_at",
|
|
"archived_at",
|
|
"issue_cycle__cycle__name",
|
|
"issue_cycle__cycle__start_date",
|
|
"issue_cycle__cycle__end_date",
|
|
"issue_module__module__name",
|
|
"issue_module__module__start_date",
|
|
"issue_module__module__target_date",
|
|
"created_by__first_name",
|
|
"created_by__last_name",
|
|
"assignees__first_name",
|
|
"assignees__last_name",
|
|
"labels__name",
|
|
)
|
|
)
|
|
.order_by("project__identifier", "sequence_id")
|
|
.distinct()
|
|
)
|
|
# CSV header
|
|
header = [
|
|
"ID",
|
|
"Project",
|
|
"Name",
|
|
"Description",
|
|
"State",
|
|
"Priority",
|
|
"Created By",
|
|
"Assignee",
|
|
"Labels",
|
|
"Cycle Name",
|
|
"Cycle Start Date",
|
|
"Cycle End Date",
|
|
"Module Name",
|
|
"Module Start Date",
|
|
"Module Target Date",
|
|
"Created At",
|
|
"Updated At",
|
|
"Completed At",
|
|
"Archived At",
|
|
]
|
|
|
|
EXPORTER_MAPPER = {
|
|
"csv": generate_csv,
|
|
"json": generate_json,
|
|
"xlsx": generate_xlsx,
|
|
}
|
|
|
|
files = []
|
|
if multiple:
|
|
for project_id in project_ids:
|
|
issues = workspace_issues.filter(project__id=project_id)
|
|
exporter = EXPORTER_MAPPER.get(provider)
|
|
if exporter is not None:
|
|
exporter(
|
|
header,
|
|
project_id,
|
|
issues,
|
|
files,
|
|
)
|
|
|
|
else:
|
|
exporter = EXPORTER_MAPPER.get(provider)
|
|
if exporter is not None:
|
|
exporter(
|
|
header,
|
|
workspace_id,
|
|
workspace_issues,
|
|
files,
|
|
)
|
|
|
|
zip_buffer = create_zip_file(files)
|
|
upload_to_s3(zip_buffer, workspace_id, token_id, slug)
|
|
|
|
except Exception as e:
|
|
exporter_instance = ExporterHistory.objects.get(token=token_id)
|
|
exporter_instance.status = "failed"
|
|
exporter_instance.reason = str(e)
|
|
exporter_instance.save(update_fields=["status", "reason"])
|
|
# Print logs if in DEBUG mode
|
|
if settings.DEBUG:
|
|
print(e)
|
|
capture_exception(e)
|
|
return
|