-
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathhandler.py
152 lines (128 loc) · 5.31 KB
/
handler.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
import os
from datetime import datetime
from django.conf import settings
from constance import config
from django.db import transaction
from config.constants import X86_DEVICE_CODENAMES
from core.exceptions import UploadException
from core.models import Build, Variant, X86Type
from core.models.build.metadata import Metadata
from core.tasks import generate_checksum, mirror_build
from core.utils import (
parse_filename_with_regex,
is_version_in_target_versions,
get_required_capture_groups,
)
@transaction.atomic
def handle_chunked_build(device, chunked_file):
# Parse file name
filename_parts = parse_filename_with_regex(chunked_file.filename)
# Run validations before continuing
run_validations(chunked_file, filename_parts)
# Construct full path to save files in
target_file_full_path = os.path.join(
settings.MEDIA_ROOT, device.codename, chunked_file.filename
)
# See if the build exists from a previous failed attempt
if os.path.exists(target_file_full_path):
os.remove(target_file_full_path)
# Make sure device codename folder exists
if not os.path.exists(os.path.join(settings.MEDIA_ROOT, device.codename)):
os.mkdir(os.path.join(settings.MEDIA_ROOT, device.codename))
# Rename chunked file and move to correct folder
os.rename(chunked_file.file.path, target_file_full_path)
# Construct and save build object in database
build = Build(
device=device,
file_name=os.path.splitext(chunked_file.filename)[0],
size=os.path.getsize(target_file_full_path),
version=filename_parts["version"],
variant=Variant.objects.get(codename=filename_parts["variant"]),
build_date=datetime.strptime(filename_parts["date"], "%Y%m%d"),
zip_file="{}/{}".format(device.codename, chunked_file.filename),
enabled=True,
)
if filename_parts["codename"] in X86_DEVICE_CODENAMES:
build.x86_type = X86Type.objects.get(codename=filename_parts["x86_type"])
build.save()
# Create metadata objects for build (if any exists)
create_metadata(build, filename_parts)
# Delete unused chunked_upload file
chunked_file.delete()
# Execute background tasks
generate_checksum.delay(build.id)
mirror_build.delay(build.id)
return build.id
def run_validations(chunked_file, filename_parts):
# Check for duplicate builds
if (
Build.objects.filter(
file_name=os.path.splitext(chunked_file.filename)[0]
).count()
>= 1
):
raise UploadException(
{
"error": "duplicate_build",
"message": "The build already exists in the system!",
}
)
# Check if variant is supported
variant_codenames = [variant.codename for variant in Variant.objects.all()]
if filename_parts["variant"] not in variant_codenames:
raise UploadException(
{
"error": "unsupported_variant",
"message": "The build's variant is not supported by this server "
"instance. If you believe the variant is valid, please contact an "
"admin to change the allowed variants list.",
}
)
# If x86, check for x86 type
if filename_parts["codename"] in X86_DEVICE_CODENAMES:
x86_type_codenames = [x.codename for x in X86Type.objects.all()]
if "x86_type" not in filename_parts:
raise UploadException(
{
"error": "missing_x86_type",
"message": "The x86 type is missing from the build's file name, "
"even though the build is an x86 build.",
}
)
if filename_parts["x86_type"] not in x86_type_codenames:
raise UploadException(
{
"error": "unsupported_x86_type",
"message": "The build's x86 type is not supported by this server "
"instance. If you believe the x86 type is valid, please contact an "
"admin to change the allowed x86 types list.",
}
)
# Check if version is in allowed versions list
if not is_version_in_target_versions(
filename_parts["version"], config.SHIPPER_ALLOWED_VERSIONS_TO_UPLOAD
):
raise UploadException(
{
"error": "version_not_allowed",
"message": "The server is currently not accepting this build's"
f"version, {filename_parts['version']}. The server only accepts the "
f"following versions: {config.SHIPPER_ALLOWED_VERSIONS_TO_UPLOAD}. "
"If you believe the version should be allowed, please contact an "
"admin to adjust server settings.",
}
)
def create_metadata(build, filename_parts):
# Filter out all keys that are not part of the required groups
required_groups = get_required_capture_groups(filename_parts["codename"])
filtered_parts = {
k: v for k, v in filename_parts.items() if k not in required_groups
}
# For remaining groups, construct metadata objects associated with build
for key in filtered_parts:
metadata = Metadata(
build=build,
name=key,
value=filtered_parts[key],
)
metadata.save()