James Ennis pushed to branch jennis/refactor_artifact_log at BuildStream / buildstream
Commits:
-
9cd6aa1d
by James Ennis at 2019-01-24T11:54:02Z
-
ae534290
by James Ennis at 2019-01-24T11:55:14Z
-
1183fdfe
by James Ennis at 2019-01-24T11:55:14Z
-
86c96245
by James Ennis at 2019-01-24T11:55:14Z
-
c84d530f
by James Ennis at 2019-01-24T11:55:14Z
-
67fdb4eb
by James Ennis at 2019-01-24T11:56:27Z
-
01c66e37
by James Ennis at 2019-01-24T11:56:27Z
-
3dd57c1b
by James Ennis at 2019-01-24T14:09:05Z
-
0d456aa1
by James Ennis at 2019-01-24T14:12:42Z
10 changed files:
- buildstream/_artifactcache.py
- + buildstream/_artifactelement.py
- buildstream/_cas/cascache.py
- buildstream/_exceptions.py
- buildstream/_frontend/cli.py
- buildstream/_project.py
- buildstream/_stream.py
- buildstream/element.py
- tests/artifactcache/pull.py
- tests/artifactcache/push.py
Changes:
... | ... | @@ -19,7 +19,6 @@ |
19 | 19 |
|
20 | 20 |
import multiprocessing
|
21 | 21 |
import os
|
22 |
-import string
|
|
23 | 22 |
from collections.abc import Mapping
|
24 | 23 |
|
25 | 24 |
from .types import _KeyStrength
|
... | ... | @@ -77,37 +76,6 @@ class ArtifactCache(): |
77 | 76 |
|
78 | 77 |
self._calculate_cache_quota()
|
79 | 78 |
|
80 |
- # get_artifact_fullname()
|
|
81 |
- #
|
|
82 |
- # Generate a full name for an artifact, including the
|
|
83 |
- # project namespace, element name and cache key.
|
|
84 |
- #
|
|
85 |
- # This can also be used as a relative path safely, and
|
|
86 |
- # will normalize parts of the element name such that only
|
|
87 |
- # digits, letters and some select characters are allowed.
|
|
88 |
- #
|
|
89 |
- # Args:
|
|
90 |
- # element (Element): The Element object
|
|
91 |
- # key (str): The element's cache key
|
|
92 |
- #
|
|
93 |
- # Returns:
|
|
94 |
- # (str): The relative path for the artifact
|
|
95 |
- #
|
|
96 |
- def get_artifact_fullname(self, element, key):
|
|
97 |
- project = element._get_project()
|
|
98 |
- |
|
99 |
- # Normalize ostree ref unsupported chars
|
|
100 |
- valid_chars = string.digits + string.ascii_letters + '-._'
|
|
101 |
- element_name = ''.join([
|
|
102 |
- x if x in valid_chars else '_'
|
|
103 |
- for x in element.normal_name
|
|
104 |
- ])
|
|
105 |
- |
|
106 |
- assert key is not None
|
|
107 |
- |
|
108 |
- # assume project and element names are not allowed to contain slashes
|
|
109 |
- return '{0}/{1}/{2}'.format(project.name, element_name, key)
|
|
110 |
- |
|
111 | 79 |
# setup_remotes():
|
112 | 80 |
#
|
113 | 81 |
# Sets up which remotes to use
|
... | ... | @@ -206,7 +174,7 @@ class ArtifactCache(): |
206 | 174 |
for key in (strong_key, weak_key):
|
207 | 175 |
if key:
|
208 | 176 |
try:
|
209 |
- ref = self.get_artifact_fullname(element, key)
|
|
177 |
+ ref = element.get_artifact_name(key)
|
|
210 | 178 |
|
211 | 179 |
self.cas.update_mtime(ref)
|
212 | 180 |
except CASError:
|
... | ... | @@ -417,7 +385,7 @@ class ArtifactCache(): |
417 | 385 |
# Returns: True if the artifact is in the cache, False otherwise
|
418 | 386 |
#
|
419 | 387 |
def contains(self, element, key):
|
420 |
- ref = self.get_artifact_fullname(element, key)
|
|
388 |
+ ref = element.get_artifact_name(key)
|
|
421 | 389 |
|
422 | 390 |
return self.cas.contains(ref)
|
423 | 391 |
|
... | ... | @@ -434,7 +402,7 @@ class ArtifactCache(): |
434 | 402 |
# Returns: True if the subdir exists & is populated in the cache, False otherwise
|
435 | 403 |
#
|
436 | 404 |
def contains_subdir_artifact(self, element, key, subdir):
|
437 |
- ref = self.get_artifact_fullname(element, key)
|
|
405 |
+ ref = element.get_artifact_name(key)
|
|
438 | 406 |
return self.cas.contains_subdir_artifact(ref, subdir)
|
439 | 407 |
|
440 | 408 |
# list_artifacts():
|
... | ... | @@ -442,11 +410,10 @@ class ArtifactCache(): |
442 | 410 |
# List artifacts in this cache in LRU order.
|
443 | 411 |
#
|
444 | 412 |
# Returns:
|
445 |
- # ([str]) - A list of artifact names as generated by
|
|
446 |
- # `ArtifactCache.get_artifact_fullname` in LRU order
|
|
413 |
+ # ([str]) - A list of artifact names as generated in LRU order
|
|
447 | 414 |
#
|
448 |
- def list_artifacts(self):
|
|
449 |
- return self.cas.list_refs()
|
|
415 |
+ def list_artifacts(self, *, glob=None):
|
|
416 |
+ return self.cas.list_refs(glob=glob)
|
|
450 | 417 |
|
451 | 418 |
# remove():
|
452 | 419 |
#
|
... | ... | @@ -455,8 +422,7 @@ class ArtifactCache(): |
455 | 422 |
#
|
456 | 423 |
# Args:
|
457 | 424 |
# ref (artifact_name): The name of the artifact to remove (as
|
458 |
- # generated by
|
|
459 |
- # `ArtifactCache.get_artifact_fullname`)
|
|
425 |
+ # generated by `Element.get_artifact_name`)
|
|
460 | 426 |
#
|
461 | 427 |
# Returns:
|
462 | 428 |
# (int|None) The amount of space pruned from the repository in
|
... | ... | @@ -503,7 +469,7 @@ class ArtifactCache(): |
503 | 469 |
# Returns: path to extracted artifact
|
504 | 470 |
#
|
505 | 471 |
def extract(self, element, key, subdir=None):
|
506 |
- ref = self.get_artifact_fullname(element, key)
|
|
472 |
+ ref = element.get_artifact_name(key)
|
|
507 | 473 |
|
508 | 474 |
path = os.path.join(self.extractdir, element._get_project().name, element.normal_name)
|
509 | 475 |
|
... | ... | @@ -519,7 +485,7 @@ class ArtifactCache(): |
519 | 485 |
# keys (list): The cache keys to use
|
520 | 486 |
#
|
521 | 487 |
def commit(self, element, content, keys):
|
522 |
- refs = [self.get_artifact_fullname(element, key) for key in keys]
|
|
488 |
+ refs = [element.get_artifact_name(key) for key in keys]
|
|
523 | 489 |
|
524 | 490 |
self.cas.commit(refs, content)
|
525 | 491 |
|
... | ... | @@ -535,8 +501,8 @@ class ArtifactCache(): |
535 | 501 |
# subdir (str): A subdirectory to limit the comparison to
|
536 | 502 |
#
|
537 | 503 |
def diff(self, element, key_a, key_b, *, subdir=None):
|
538 |
- ref_a = self.get_artifact_fullname(element, key_a)
|
|
539 |
- ref_b = self.get_artifact_fullname(element, key_b)
|
|
504 |
+ ref_a = element.get_artifact_name(key_a)
|
|
505 |
+ ref_b = element.get_artifact_name(key_b)
|
|
540 | 506 |
|
541 | 507 |
return self.cas.diff(ref_a, ref_b, subdir=subdir)
|
542 | 508 |
|
... | ... | @@ -597,7 +563,7 @@ class ArtifactCache(): |
597 | 563 |
# (ArtifactError): if there was an error
|
598 | 564 |
#
|
599 | 565 |
def push(self, element, keys):
|
600 |
- refs = [self.get_artifact_fullname(element, key) for key in list(keys)]
|
|
566 |
+ refs = [element.get_artifact_name(key) for key in list(keys)]
|
|
601 | 567 |
|
602 | 568 |
project = element._get_project()
|
603 | 569 |
|
... | ... | @@ -635,7 +601,7 @@ class ArtifactCache(): |
635 | 601 |
# (bool): True if pull was successful, False if artifact was not available
|
636 | 602 |
#
|
637 | 603 |
def pull(self, element, key, *, progress=None, subdir=None, excluded_subdirs=None):
|
638 |
- ref = self.get_artifact_fullname(element, key)
|
|
604 |
+ ref = element.get_artifact_name(key)
|
|
639 | 605 |
|
640 | 606 |
project = element._get_project()
|
641 | 607 |
|
... | ... | @@ -747,8 +713,8 @@ class ArtifactCache(): |
747 | 713 |
# newkey (str): A new cache key for the artifact
|
748 | 714 |
#
|
749 | 715 |
def link_key(self, element, oldkey, newkey):
|
750 |
- oldref = self.get_artifact_fullname(element, oldkey)
|
|
751 |
- newref = self.get_artifact_fullname(element, newkey)
|
|
716 |
+ oldref = element.get_artifact_name(oldkey)
|
|
717 |
+ newref = element.get_artifact_name(newkey)
|
|
752 | 718 |
|
753 | 719 |
self.cas.link_ref(oldref, newref)
|
754 | 720 |
|
1 |
+#
|
|
2 |
+# Copyright (C) 2019 Bloomberg Finance LP
|
|
3 |
+#
|
|
4 |
+# This program is free software; you can redistribute it and/or
|
|
5 |
+# modify it under the terms of the GNU Lesser General Public
|
|
6 |
+# License as published by the Free Software Foundation; either
|
|
7 |
+# version 2 of the License, or (at your option) any later version.
|
|
8 |
+#
|
|
9 |
+# This library is distributed in the hope that it will be useful,
|
|
10 |
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
11 |
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
12 |
+# Lesser General Public License for more details.
|
|
13 |
+#
|
|
14 |
+# You should have received a copy of the GNU Lesser General Public
|
|
15 |
+# License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
16 |
+#
|
|
17 |
+# Authors:
|
|
18 |
+# James Ennis <james ennis codethink co uk>
|
|
19 |
+from ._exceptions import ArtifactElementError
|
|
20 |
+ |
|
21 |
+ |
|
22 |
+# ArtifactElement()
|
|
23 |
+#
|
|
24 |
+# Object to be used for directly processing an artifact
|
|
25 |
+#
|
|
26 |
+# Args:
|
|
27 |
+# context (Context): The Context object
|
|
28 |
+# ref (str): The artifact ref
|
|
29 |
+#
|
|
30 |
+class ArtifactElement():
|
|
31 |
+ def __init__(self, context, ref):
|
|
32 |
+ try:
|
|
33 |
+ project_name, element, key = ref.split('/', 2)
|
|
34 |
+ except ValueError:
|
|
35 |
+ raise ArtifactElementError("Artifact: {} is not of the expected format".format(ref))
|
|
36 |
+ |
|
37 |
+ self.ref = ref
|
|
38 |
+ self.project_name = project_name
|
|
39 |
+ self.element = element
|
|
40 |
+ self.key = key
|
|
41 |
+ |
|
42 |
+ self._context = context
|
|
43 |
+ self._artifacts = context.artifactcache
|
|
44 |
+ self._cas = context.artifactcache.cas # is this bad...?
|
|
45 |
+ |
|
46 |
+ def get_artifact_name(self):
|
|
47 |
+ return self.ref
|
|
48 |
+ |
|
49 |
+ def _cached(self):
|
|
50 |
+ return self._cas.contains(self.ref)
|
... | ... | @@ -475,14 +475,18 @@ class CASCache(): |
475 | 475 |
# Returns:
|
476 | 476 |
# (list) - A list of refs in LRM order
|
477 | 477 |
#
|
478 |
- def list_refs(self):
|
|
478 |
+ def list_refs(self, *, glob=None):
|
|
479 | 479 |
# string of: /path/to/repo/refs/heads
|
480 | 480 |
ref_heads = os.path.join(self.casdir, 'refs', 'heads')
|
481 | 481 |
|
482 |
+ if glob is not None and '/' in glob:
|
|
483 |
+ ref_heads_with_glob = os.path.join(ref_heads, os.path.dirname(glob))
|
|
484 |
+ |
|
482 | 485 |
refs = []
|
483 | 486 |
mtimes = []
|
484 | 487 |
|
485 |
- for root, _, files in os.walk(ref_heads):
|
|
488 |
+ path = ref_heads_with_glob if glob else ref_heads
|
|
489 |
+ for root, _, files in os.walk(path):
|
|
486 | 490 |
for filename in files:
|
487 | 491 |
ref_path = os.path.join(root, filename)
|
488 | 492 |
refs.append(os.path.relpath(ref_path, ref_heads))
|
... | ... | @@ -344,3 +344,12 @@ class AppError(BstError): |
344 | 344 |
#
|
345 | 345 |
class SkipJob(Exception):
|
346 | 346 |
pass
|
347 |
+ |
|
348 |
+ |
|
349 |
+# ArtifactElementError
|
|
350 |
+#
|
|
351 |
+# Raised when errors are encountered by artifact elements
|
|
352 |
+#
|
|
353 |
+class ArtifactElementError(BstError):
|
|
354 |
+ def __init__(self, message, *, detail=None, reason=None, temporary=False):
|
|
355 |
+ super().__init__(message, detail=detail, domain=ErrorDomain.ELEMENT, reason=reason, temporary=True)
|
1 | 1 |
import os
|
2 | 2 |
import sys
|
3 | 3 |
from contextlib import ExitStack
|
4 |
-from fnmatch import fnmatch
|
|
5 | 4 |
from functools import partial
|
6 | 5 |
from tempfile import TemporaryDirectory
|
7 | 6 |
|
... | ... | @@ -859,38 +858,6 @@ def workspace_list(app): |
859 | 858 |
#############################################################
|
860 | 859 |
# Artifact Commands #
|
861 | 860 |
#############################################################
|
862 |
-def _classify_artifacts(names, cas, project_directory):
|
|
863 |
- element_targets = []
|
|
864 |
- artifact_refs = []
|
|
865 |
- element_globs = []
|
|
866 |
- artifact_globs = []
|
|
867 |
- |
|
868 |
- for name in names:
|
|
869 |
- if name.endswith('.bst'):
|
|
870 |
- if any(c in "*?[" for c in name):
|
|
871 |
- element_globs.append(name)
|
|
872 |
- else:
|
|
873 |
- element_targets.append(name)
|
|
874 |
- else:
|
|
875 |
- if any(c in "*?[" for c in name):
|
|
876 |
- artifact_globs.append(name)
|
|
877 |
- else:
|
|
878 |
- artifact_refs.append(name)
|
|
879 |
- |
|
880 |
- if element_globs:
|
|
881 |
- for dirpath, _, filenames in os.walk(project_directory):
|
|
882 |
- for filename in filenames:
|
|
883 |
- element_path = os.path.join(dirpath, filename).lstrip(project_directory).lstrip('/')
|
|
884 |
- if any(fnmatch(element_path, glob) for glob in element_globs):
|
|
885 |
- element_targets.append(element_path)
|
|
886 |
- |
|
887 |
- if artifact_globs:
|
|
888 |
- artifact_refs.extend(ref for ref in cas.list_refs()
|
|
889 |
- if any(fnmatch(ref, glob) for glob in artifact_globs))
|
|
890 |
- |
|
891 |
- return element_targets, artifact_refs
|
|
892 |
- |
|
893 |
- |
|
894 | 861 |
@cli.group(short_help="Manipulate cached artifacts")
|
895 | 862 |
def artifact():
|
896 | 863 |
"""Manipulate cached artifacts"""
|
... | ... | @@ -1045,53 +1012,31 @@ def artifact_push(app, elements, deps, remote): |
1045 | 1012 |
@click.pass_obj
|
1046 | 1013 |
def artifact_log(app, artifacts):
|
1047 | 1014 |
"""Show logs of all artifacts"""
|
1048 |
- from .._exceptions import CASError
|
|
1049 |
- from .._message import MessageType
|
|
1050 |
- from .._pipeline import PipelineSelection
|
|
1051 |
- from ..storage._casbaseddirectory import CasBasedDirectory
|
|
1052 |
- |
|
1053 |
- with ExitStack() as stack:
|
|
1054 |
- stack.enter_context(app.initialized())
|
|
1055 |
- cache = app.context.artifactcache
|
|
1015 |
+ # Guess the element if we're in a workspace
|
|
1016 |
+ if not artifacts:
|
|
1017 |
+ guessed_target = app.context.guess_element()
|
|
1018 |
+ if guessed_target:
|
|
1019 |
+ artifacts = [guessed_target]
|
|
1056 | 1020 |
|
1057 |
- elements, artifacts = _classify_artifacts(artifacts, cache.cas,
|
|
1058 |
- app.project.directory)
|
|
1059 |
- |
|
1060 |
- vdirs = []
|
|
1061 |
- extractdirs = []
|
|
1062 |
- if artifacts:
|
|
1063 |
- for ref in artifacts:
|
|
1064 |
- try:
|
|
1065 |
- cache_id = cache.cas.resolve_ref(ref, update_mtime=True)
|
|
1066 |
- vdir = CasBasedDirectory(cache.cas, cache_id)
|
|
1067 |
- vdirs.append(vdir)
|
|
1068 |
- except CASError as e:
|
|
1069 |
- app._message(MessageType.WARN, "Artifact {} is not cached".format(ref), detail=str(e))
|
|
1070 |
- continue
|
|
1071 |
- if elements:
|
|
1072 |
- elements = app.stream.load_selection(elements, selection=PipelineSelection.NONE)
|
|
1073 |
- for element in elements:
|
|
1074 |
- if not element._cached():
|
|
1075 |
- app._message(MessageType.WARN, "Element {} is not cached".format(element))
|
|
1076 |
- continue
|
|
1077 |
- ref = cache.get_artifact_fullname(element, element._get_cache_key())
|
|
1078 |
- cache_id = cache.cas.resolve_ref(ref, update_mtime=True)
|
|
1079 |
- vdir = CasBasedDirectory(cache.cas, cache_id)
|
|
1080 |
- vdirs.append(vdir)
|
|
1081 |
- |
|
1082 |
- for vdir in vdirs:
|
|
1083 |
- # NOTE: If reading the logs feels unresponsive, here would be a good place to provide progress information.
|
|
1084 |
- logsdir = vdir.descend(["logs"])
|
|
1085 |
- td = stack.enter_context(TemporaryDirectory())
|
|
1086 |
- logsdir.export_files(td, can_link=True)
|
|
1087 |
- extractdirs.append(td)
|
|
1088 |
- |
|
1089 |
- for extractdir in extractdirs:
|
|
1090 |
- for log in (os.path.join(extractdir, log) for log in os.listdir(extractdir)):
|
|
1091 |
- # NOTE: Should click gain the ability to pass files to the pager this can be optimised.
|
|
1092 |
- with open(log) as f:
|
|
1093 |
- data = f.read()
|
|
1094 |
- click.echo_via_pager(data)
|
|
1021 |
+ with app.initialized():
|
|
1022 |
+ vdirs = app.stream.artifact_log(artifacts)
|
|
1023 |
+ |
|
1024 |
+ with ExitStack() as stack:
|
|
1025 |
+ extractdirs = []
|
|
1026 |
+ for vdir in vdirs:
|
|
1027 |
+ # NOTE: If reading the logs feels unresponsive, here would be a good place
|
|
1028 |
+ # to provide progress information.
|
|
1029 |
+ logsdir = vdir.descend(["logs"])
|
|
1030 |
+ td = stack.enter_context(TemporaryDirectory())
|
|
1031 |
+ logsdir.export_files(td, can_link=True)
|
|
1032 |
+ extractdirs.append(td)
|
|
1033 |
+ |
|
1034 |
+ for extractdir in extractdirs:
|
|
1035 |
+ for log in (os.path.join(extractdir, log) for log in os.listdir(extractdir)):
|
|
1036 |
+ # NOTE: Should click gain the ability to pass files to the pager this can be optimised.
|
|
1037 |
+ with open(log) as f:
|
|
1038 |
+ data = f.read()
|
|
1039 |
+ click.echo_via_pager(data)
|
|
1095 | 1040 |
|
1096 | 1041 |
|
1097 | 1042 |
##################################################################
|
... | ... | @@ -26,6 +26,7 @@ from . import utils |
26 | 26 |
from . import _cachekey
|
27 | 27 |
from . import _site
|
28 | 28 |
from . import _yaml
|
29 |
+from ._artifactelement import ArtifactElement
|
|
29 | 30 |
from ._profile import Topics, profile_start, profile_end
|
30 | 31 |
from ._exceptions import LoadError, LoadErrorReason
|
31 | 32 |
from ._options import OptionPool
|
... | ... | @@ -252,6 +253,19 @@ class Project(): |
252 | 253 |
else:
|
253 | 254 |
return self.config.element_factory.create(self._context, self, meta)
|
254 | 255 |
|
256 |
+ # create_artifact_element()
|
|
257 |
+ #
|
|
258 |
+ # Instantiate and return an ArtifactElement
|
|
259 |
+ #
|
|
260 |
+ # Args:
|
|
261 |
+ # ref (str): A string of the artifact ref
|
|
262 |
+ #
|
|
263 |
+ # Returns:
|
|
264 |
+ # (ArtifactElement): A newly created ArtifactElement object of the appropriate kind
|
|
265 |
+ #
|
|
266 |
+ def create_artifact_element(self, ref):
|
|
267 |
+ return ArtifactElement(self._context, ref)
|
|
268 |
+ |
|
255 | 269 |
# create_source()
|
256 | 270 |
#
|
257 | 271 |
# Instantiate and return a Source
|
... | ... | @@ -440,6 +440,51 @@ class Stream(): |
440 | 440 |
raise StreamError("Error while staging dependencies into a sandbox"
|
441 | 441 |
": '{}'".format(e), detail=e.detail, reason=e.reason) from e
|
442 | 442 |
|
443 |
+ # artifact_log()
|
|
444 |
+ #
|
|
445 |
+ # Show the full log of an artifact
|
|
446 |
+ #
|
|
447 |
+ # Args:
|
|
448 |
+ # targets (str): Targets to view the logs of
|
|
449 |
+ #
|
|
450 |
+ # Returns:
|
|
451 |
+ # vdirs (list): A list of CasBasedDirectory objects
|
|
452 |
+ #
|
|
453 |
+ def artifact_log(self, targets):
|
|
454 |
+ from .storage._casbaseddirectory import CasBasedDirectory # is this bad?
|
|
455 |
+ |
|
456 |
+ cas = self._artifacts.cas # Is this bad?
|
|
457 |
+ |
|
458 |
+ # Distinguish the artifacts from the elements
|
|
459 |
+ elements, artifacts = self._classify_artifacts(targets)
|
|
460 |
+ |
|
461 |
+ # Obtain Element objects
|
|
462 |
+ if elements:
|
|
463 |
+ elements = self.load_selection(elements, selection=PipelineSelection.NONE)
|
|
464 |
+ |
|
465 |
+ # Obtain ArtifactElement objects
|
|
466 |
+ artifact_elements = []
|
|
467 |
+ if artifacts:
|
|
468 |
+ for ref in artifacts:
|
|
469 |
+ artifact_element = self._project.create_artifact_element(ref)
|
|
470 |
+ artifact_elements.append(artifact_element)
|
|
471 |
+ |
|
472 |
+ # Concatenate the lists
|
|
473 |
+ objects = elements + artifact_elements
|
|
474 |
+ |
|
475 |
+ vdirs = []
|
|
476 |
+ for obj in objects:
|
|
477 |
+ ref = obj.get_artifact_name()
|
|
478 |
+ if not obj._cached():
|
|
479 |
+ self._message(MessageType.WARN, "{} is not cached".format(ref))
|
|
480 |
+ continue
|
|
481 |
+ |
|
482 |
+ cache_id = cas.resolve_ref(ref, update_mtime=True)
|
|
483 |
+ vdir = CasBasedDirectory(cas, cache_id)
|
|
484 |
+ vdirs.append(vdir)
|
|
485 |
+ |
|
486 |
+ return vdirs
|
|
487 |
+ |
|
443 | 488 |
# source_checkout()
|
444 | 489 |
#
|
445 | 490 |
# Checkout sources of the target element to the specified location
|
... | ... | @@ -1277,44 +1322,47 @@ class Stream(): |
1277 | 1322 |
|
1278 | 1323 |
# _classify_artifacts()
|
1279 | 1324 |
#
|
1280 |
- # Split up a list of tagets into element names and artifact refs
|
|
1325 |
+ # Split up a list of targets into element names and artifact refs
|
|
1281 | 1326 |
#
|
1282 | 1327 |
# Args:
|
1283 |
- # names (list): A list of targets
|
|
1284 |
- # cas (CASCache): The CASCache object
|
|
1285 |
- # project_directory (str): Absolute path to the project
|
|
1328 |
+ # targets (list): A list of targets
|
|
1286 | 1329 |
#
|
1287 | 1330 |
# Returns:
|
1288 | 1331 |
# (list): element names present in the targets
|
1289 | 1332 |
# (list): artifact refs present in the targets
|
1290 | 1333 |
#
|
1291 |
- def _classify_artifacts(names, cas, project_directory):
|
|
1334 |
+ def _classify_artifacts(self, targets):
|
|
1292 | 1335 |
element_targets = []
|
1293 | 1336 |
artifact_refs = []
|
1294 | 1337 |
element_globs = []
|
1295 | 1338 |
artifact_globs = []
|
1296 | 1339 |
|
1297 |
- for name in names:
|
|
1298 |
- if name.endswith('.bst'):
|
|
1299 |
- if any(c in "*?[" for c in name):
|
|
1300 |
- element_globs.append(name)
|
|
1340 |
+ for target in targets:
|
|
1341 |
+ if target.endswith('.bst'):
|
|
1342 |
+ if any(c in "*?[" for c in target):
|
|
1343 |
+ element_globs.append(target)
|
|
1301 | 1344 |
else:
|
1302 |
- element_targets.append(name)
|
|
1345 |
+ element_targets.append(target)
|
|
1303 | 1346 |
else:
|
1304 |
- if any(c in "*?[" for c in name):
|
|
1305 |
- artifact_globs.append(name)
|
|
1347 |
+ if any(c in "*?[" for c in target):
|
|
1348 |
+ artifact_globs.append(target)
|
|
1306 | 1349 |
else:
|
1307 |
- artifact_refs.append(name)
|
|
1350 |
+ artifact_refs.append(target)
|
|
1308 | 1351 |
|
1309 | 1352 |
if element_globs:
|
1310 |
- for dirpath, _, filenames in os.walk(project_directory):
|
|
1353 |
+ for dirpath, _, filenames in os.walk(self._project.element_path):
|
|
1311 | 1354 |
for filename in filenames:
|
1312 |
- element_path = os.path.join(dirpath, filename).lstrip(project_directory).lstrip('/')
|
|
1355 |
+ element_path = os.path.join(dirpath, filename).lstrip(self._project.element_path).lstrip('/')
|
|
1313 | 1356 |
if any(fnmatch(element_path, glob) for glob in element_globs):
|
1314 | 1357 |
element_targets.append(element_path)
|
1315 | 1358 |
|
1316 | 1359 |
if artifact_globs:
|
1317 |
- artifact_refs.extend(ref for ref in cas.list_refs()
|
|
1318 |
- if any(fnmatch(ref, glob) for glob in artifact_globs))
|
|
1360 |
+ for glob in artifact_globs:
|
|
1361 |
+ refs = self._artifacts.list_artifacts(glob=glob)
|
|
1362 |
+ for ref in refs:
|
|
1363 |
+ if fnmatch(ref, glob):
|
|
1364 |
+ artifact_refs.append(ref)
|
|
1365 |
+ if not artifact_refs:
|
|
1366 |
+ self._message(MessageType.WARN, "No artifacts found for globs: {}".format(', '.join(artifact_globs)))
|
|
1319 | 1367 |
|
1320 | 1368 |
return element_targets, artifact_refs
|
... | ... | @@ -82,6 +82,7 @@ import contextlib |
82 | 82 |
from contextlib import contextmanager
|
83 | 83 |
import tempfile
|
84 | 84 |
import shutil
|
85 |
+import string
|
|
85 | 86 |
|
86 | 87 |
from . import _yaml
|
87 | 88 |
from ._variables import Variables
|
... | ... | @@ -577,6 +578,37 @@ class Element(Plugin): |
577 | 578 |
self.__assert_cached()
|
578 | 579 |
return self.__compute_splits(include, exclude, orphans)
|
579 | 580 |
|
581 |
+ def get_artifact_name(self, key=None):
|
|
582 |
+ """Compute and return this element's full artifact name
|
|
583 |
+ |
|
584 |
+ Generate a full name for an artifact, including the project
|
|
585 |
+ namespace, element name and cache key.
|
|
586 |
+ |
|
587 |
+ This can also be used as a relative path safely, and
|
|
588 |
+ will normalize parts of the element name such that only
|
|
589 |
+ digits, letters and some select characters are allowed.
|
|
590 |
+ |
|
591 |
+ Args:
|
|
592 |
+ key (str): The element's cache key. Defaults to None
|
|
593 |
+ |
|
594 |
+ Returns:
|
|
595 |
+ (str): The relative path for the artifact
|
|
596 |
+ """
|
|
597 |
+ project = self._get_project()
|
|
598 |
+ if key is None:
|
|
599 |
+ key = self._get_cache_key()
|
|
600 |
+ |
|
601 |
+ assert key is not None
|
|
602 |
+ |
|
603 |
+ valid_chars = string.digits + string.ascii_letters + '-._'
|
|
604 |
+ element_name = ''.join([
|
|
605 |
+ x if x in valid_chars else '_'
|
|
606 |
+ for x in self.normal_name
|
|
607 |
+ ])
|
|
608 |
+ |
|
609 |
+ # assume project and element names are not allowed to contain slashes
|
|
610 |
+ return '{0}/{1}/{2}'.format(project.name, element_name, key)
|
|
611 |
+ |
|
580 | 612 |
def stage_artifact(self, sandbox, *, path=None, include=None, exclude=None, orphans=True, update_mtimes=None):
|
581 | 613 |
"""Stage this element's output artifact in the sandbox
|
582 | 614 |
|
... | ... | @@ -210,7 +210,7 @@ def test_pull_tree(cli, tmpdir, datafiles): |
210 | 210 |
assert artifactcache.contains(element, element_key)
|
211 | 211 |
|
212 | 212 |
# Retrieve the Directory object from the cached artifact
|
213 |
- artifact_ref = artifactcache.get_artifact_fullname(element, element_key)
|
|
213 |
+ artifact_ref = element.get_artifact_name(element_key)
|
|
214 | 214 |
artifact_digest = cas.resolve_ref(artifact_ref)
|
215 | 215 |
|
216 | 216 |
queue = multiprocessing.Queue()
|
... | ... | @@ -190,7 +190,7 @@ def test_push_directory(cli, tmpdir, datafiles): |
190 | 190 |
assert artifactcache.has_push_remotes(element=element)
|
191 | 191 |
|
192 | 192 |
# Recreate the CasBasedDirectory object from the cached artifact
|
193 |
- artifact_ref = artifactcache.get_artifact_fullname(element, element_key)
|
|
193 |
+ artifact_ref = element.get_artifact_name(element_key)
|
|
194 | 194 |
artifact_digest = cas.resolve_ref(artifact_ref)
|
195 | 195 |
|
196 | 196 |
queue = multiprocessing.Queue()
|