Example #1
0
File: Context.py Project: azag0/caf
 def make_targets(self, out, cache):
     for target, tasks in self.targets.items():
         if len(tasks) == 1 and None in tasks:
             relink(Path('../.caf/db')/'/'.join(tasks[None].path.parts[-4:]), out/target, relative=False)
         else:
             if not (out/target).is_dir():
                 mkdir(out/target)
             for name, task in tasks.items():
                 relink(Path('../../.caf/db')/'/'.join(task.path.parts[-4:]), out/target/name, relative=False)
Example #2
0
File: Caf.py Project: azag0/caf
def init(caf):
    """
    Initialize the Caf repository.

    Usage:
        caf init

    By default create directory in .caf/db. If 'cache' is defined in
    ~/.config/caf/conf.yaml, the repository is created there and symlinked to
    .caf/db, otherwise it is created locally.
    """
    if 'cache' in caf.conf:
        timestamp = get_timestamp()
        cache_path = Path(caf.conf['cache'])/'{}_{}'.format(Path().resolve().name, timestamp)
        mkdir(cache_path)
        relink(cache_path, caf.cache, relative=False)
    else:
        cache_path = caf.cache
        if cache_path.exists():
            error('{} exists, cannot overwrite'.format(cache_path))
        mkdir(cache_path)
    info('Initializing an empty repository at {}.'.format(cache_path))
    mkdir(caf.cellar)
    mkdir(caf.brewery)
    with open('.gitignore', 'w') as f:
        f.write('\n'.join(['.caf']))
    with open(os.devnull, 'w') as null:
        sp.call(['git', 'init'], stdout=null)
        sp.call(['git', 'add', 'caf', 'cscript.py', '.gitignore'], stdout=null)
        sp.call(['git', 'commit', '-m', 'initial commit'], stdout=null)
Example #3
0
File: Context.py Project: azag0/caf
    def build(self, path):
        """Prepare, lock and store the task.

        Check if not already locked. Touch (link in children, save chilren to
        .caf/children). Check if needed children are already sealed. Check if
        children are already locked. Prepare task. Get hashes.  Lock task with
        hashes. Check if a task has been already stored previously and if not,
        store it and relink children.
        """
        with timing('task init'):
            if not path.is_dir():
                mkdir(path)
            self.path = Path(path).resolve()
            if self.is_locked():
                warn('{} already locked'.format(self))
                return
            if not self.is_touched():
                self.touch()
            for linkname, link in self.links.items():
                if link.needed and not link.task.is_sealed():
                    warn('{}: dependency "{}" not sealed'.format(self, linkname))
                    return
            if not all(child.is_locked() for child in self.children):
                return
        with timing('prepare'):
            self.prepare()
        with timing('hash'):
            hashes = self.get_hashes()
        with timing('lock'):
            self.lock(hashes)
        myhash = get_file_hash(self.path/'.caf/lock')
        with timing('storing'):
            cellarpath = self.ctx.cellar/str_to_path(myhash)
            if cellarpath.is_dir():
                env_file = Path(self.path/'.caf/env')
                if env_file.is_file():
                    env_file.rename(cellarpath/'.caf/env')
                shutil.rmtree(str(self.path))
            else:
                info('Stored new task {}'.format(self))
                mkdir(cellarpath.parent, parents=True, exist_ok=True)
                self.path.rename(cellarpath)
            relink(cellarpath, self.path)
            self.path = cellarpath
        with timing('linking deps'):
            self.link_deps()
Example #4
0
File: Context.py Project: azag0/caf
 def store_link_text(self, text, target, label=None):
     h = hashlib.new(hashf)
     h.update(text.encode())
     texthash = h.hexdigest()
     cellarpath = self.ctx.cellar/str_to_path(texthash)
     if not cellarpath.is_file():
         if label is True:
             info('Stored new file "{}"'.format(target))
         elif label:
             info('Stored new text labeled "{}"'.format(label))
         else:
             info('Stored new text')
         mkdir(cellarpath.parent, parents=True, exist_ok=True)
         with cellarpath.open('w') as f:
             f.write(text)
         make_nonwritable(cellarpath)
     with cd(self.path):
         relink(os.path.relpath(str(cellarpath)), target)
     self.files[target] = cellarpath
Example #5
0
File: Caf.py Project: azag0/caf
def build(caf, dry: '--dry', do_init: 'init'):
    """
    Prepare tasks and targets defined in cscript.

    Usage:
        caf [init] build [--dry]

    Options:
        -n, --dry                  Dry run (do not write to disk).

    Tasks are created in .caf/db/Brewery/Latest and if their preparation does
    not depened on unfinished tasks, they are prepared and stored in
    .caf/db/Cellar based on their SHA1 hash. Targets (collections of symlinks to
    tasks) are created in ./build.
    """
    if not hasattr(caf.cscript, 'build'):
        error('cscript has to contain function build(ctx)')
    if do_init:
        init(['caf', 'init'], caf)
    ctx = Context(caf.cache/cellar, caf.top, caf.libpath)
    with timing('dependency tree'):
        caf.cscript.build(ctx)
    if not dry:
        timestamp = get_timestamp()
        mkdir(caf.brewery/timestamp)
        relink(timestamp, caf.brewery/latest, relative=False)
        with timing('build'):
            ctx.build(caf.brewery/latest)
        if caf.out.is_dir():
            shutil.rmtree(str(caf.out))
        mkdir(caf.out)
        with timing('targets'):
            ctx.make_targets(caf.out, caf.cache)
        if hasattr(caf.cscript, 'json'):
            warn('Make sure json is not printing dictionaries in features')
    with open(os.devnull, 'w') as null:
        sp.call(['git', 'add', '--all', 'build'], stdout=null)
        sp.call(['git', 'commit', '-a', '-m', '#build'], stdout=null)
Example #6
0
File: Context.py Project: azag0/caf
 def store_link_file(self, source, target=None):
     try:
         text = source.getvalue()
     except AttributeError:
         pass
     else:
         assert target
         self.store_link_text(text, target)
         return
     if not target:
         target = source
     if Path(source).is_dir():
         (self.path/target).mkdir()
         return
     filehash = get_file_hash(Path(source))
     cellarpath = self.ctx.cellar/str_to_path(filehash)
     if not cellarpath.is_file():
         info('Stored new file "{}"'.format(source))
         mkdir(cellarpath.parent, parents=True, exist_ok=True)
         shutil.copy(str(source), str(cellarpath))
         make_nonwritable(cellarpath)
     with cd(self.path):
         relink(os.path.relpath(str(cellarpath)), target)
     self.files[str(target)] = cellarpath
Example #7
0
File: Context.py Project: azag0/caf
 def touch(self):
     mkdir(self.path/'.caf')
     with (self.path/'.caf/children').open('w') as f:
         json.dump(list(self.links), f, sort_keys=True)
     self.link_deps()