UNPKG

@dashkite/tempo

Version:

Mono/polyrepo project management

527 lines (526 loc) 73.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); function _export(target, all) { for(var name in all)Object.defineProperty(target, name, { enumerable: true, get: Object.getOwnPropertyDescriptor(all, name).get }); } _export(exports, { get Repo () { return Repo; }, get Repos () { return Repos; } }); const _nodepath = /*#__PURE__*/ _interop_require_default(require("node:path")); const _nodecrypto = /*#__PURE__*/ _interop_require_default(require("node:crypto")); const _function = /*#__PURE__*/ _interop_require_wildcard(require("@dashkite/joy/function")); const _iterable = /*#__PURE__*/ _interop_require_wildcard(require("@dashkite/joy/iterable")); const _type = /*#__PURE__*/ _interop_require_wildcard(require("@dashkite/joy/type")); const _text = /*#__PURE__*/ _interop_require_wildcard(require("@dashkite/joy/text")); const _generic = require("@dashkite/joy/generic"); const _bake = require("@dashkite/bake"); const _zephyr = /*#__PURE__*/ _interop_require_default(require("@dashkite/zephyr")); const _kaiko = /*#__PURE__*/ _interop_require_default(require("@dashkite/kaiko")); const _plimit = /*#__PURE__*/ _interop_require_default(require("p-limit")); const _progress = /*#__PURE__*/ _interop_require_default(require("./progress")); const _scripts = require("./scripts"); function _interop_require_default(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); } function _interop_require_wildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = { __proto__: null }; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for(var key in obj){ if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; } var Hash, Memos, Repo, Repos, has, partition, push, remove, slice, indexOf = [].indexOf; has = function(keys) { if (!_type.isArray(keys)) { keys = [ keys ]; } return function(value) { return keys.every(function(key) { return value[key] != null; }); }; }; push = function(stack, value) { stack.unshift(value); return value; }; remove = function(list, target) { var index; if ((index = list.indexOf(target)) > -1) { return list.splice(index, 1); } }; slice = function(stack, start, skip) { return stack.slice(start, start + skip); }; partition = function*(size, list) { var i, j, results; i = 0; j = Math.ceil(list.length / size); results = []; while(i < j){ results.push((yield slice(list, i++ * size, size))); } return results; }; Hash = { md5: function(buffer) { return (0, _bake.convert)({ from: "bytes", to: "base36" }, new Uint8Array(function() { return _nodecrypto.default.createHash("md5").update(buffer).digest().buffer; }())); }, array: function(array) { return _text.truncate(8, Hash.md5(array.sort().join(","))); } }; Memos = { path: _nodepath.default.join(".tempo", "memos.json") }; Repos = { path: _nodepath.default.join(".tempo", "repos.yaml"), initialize: function() { return _zephyr.default.update(Repos.path, function(repos) { return repos != null ? repos : repos = []; }); }, load: function() { return _zephyr.default.read(Repos.path); }, get: async function(name) { var repos; repos = await Repos.load(); return repos.find(function(repo) { return repo.name === name; }); }, add: function({ organization, name }) { return _zephyr.default.update(Repos.path, function(repos) { repos.push({ organization, name }); return repos; }); }, remove: function({ organization, name }) { return _zephyr.default.update(Repos.path, function(repos) { var repo; repo = repos.find(function(repo) { return repo.organization === organization && repo.name === name; }); remove(repos, repo); return repos; }); }, tag: async function(repos, tags) { var k, len, repo, results; results = []; for(k = 0, len = repos.length; k < len; k++){ repo = repos[k]; results.push(await Repo.tag(repo, tags)); } return results; }, untag: async function(repos, tags) { var k, len, repo, results; results = []; for(k = 0, len = repos.length; k < len; k++){ repo = repos[k]; results.push(await Repo.untag(repo, tags)); } return results; }, find: function({ find }) { find = (0, _generic.generic)({ name: "Repos.find", default: function() { return Repos.load(); } }); (0, _generic.generic)(find, has("repos"), function({ repos, ...options }) { return _function.flow([ function() { return Repos.find(options); }, _iterable.select(function(repo) { var ref; return ref = repo.name, indexOf.call(repos, ref) >= 0; }) ])(); }); (0, _generic.generic)(find, has("include"), async function({ include, ...options }) { var repos; repos = await _zephyr.default.read(include); return Repos.find({ repos, ...options }); }); (0, _generic.generic)(find, has([ "repos", "include" ]), function({ repos, include }) { return _function.flow([ Repos.find({ include: repos }), async function(result) { return result.concat(await Repos.find({ include })); } ])(); }); (0, _generic.generic)(find, has("tags"), function({ tags, ...options }) { return _function.flow([ function() { return Repos.find(options); }, _iterable.select(function(repo) { return repo.tags != null && tags.some(function(tag) { return indexOf.call(repo.tags, tag) >= 0; }); }) ])(); }); (0, _generic.generic)(find, has("exclude"), async function({ exclude, ...options }) { exclude = await _zephyr.default.read(exclude); return _function.flow([ function() { return Repos.find(options); }, _iterable.select(function(repo) { var ref; return !(ref = repo.name, indexOf.call(exclude, ref) >= 0); }) ])(); }); return find; }({}), run: function({ run }) { run = (0, _generic.generic)({ name: "Repos.run" }); (0, _generic.generic)(run, _type.isObject, async function({ repos, command, key, batch, retry }) { var before, count, done, failed, failures, found, group, groups, hash, history, index, k, l, len, len1, len2, len3, limiter, m, memos, mulligan, n, name1, pending, progress, repo, retries, succeeded; if (batch == null) { batch = 6; // max parallel builds } if (retry == null) { retry = true; } if (retry) { memos = await _zephyr.default.read(Memos.path); if (memos == null) { memos = {}; } groups = memos[key]; } // default the trivial group if (groups == null) { groups = [ repos.map(function({ name }) { return name; }) ]; } // check for missing repos // add to first group if we find any for(k = 0, len = repos.length; k < len; k++){ repo = repos[k]; found = groups.find(function(group) { var ref; return ref = repo.name, indexOf.call(group, ref) >= 0; }) != null; if (!found) { push(groups[0], repo.name); } } // remove repos that are not in the target repos list groups = groups.map(function(group) { return group.filter(function(name) { return repos.find(function(repo) { return repo.name === name; }) != null; }); // remove empty groups since they will halt the run loop }).filter(function(group) { return group.length !== 0; }); failed = void 0; hash = void 0; retries = retry ? 6 : 0; history = []; // initialize failures lookup failures = {}; for(l = 0, len1 = repos.length; l < len1; l++){ repo = repos[l]; failures[repo.name] = 0; } _kaiko.default.info({ console: true, message: `Running [ ${_text.elide(40, "...", command)} ]`, command: command }); limiter = (0, _plimit.default)(batch); done = function() { var ref; return failed != null && (failed.length === 0 || (ref = hash = Hash.array(failed), indexOf.call(history, ref) >= 0)); }; count = 0; while(!done()){ if (hash != null) { history.push(hash); } if (failed != null) { groups.push(failed); } index = 0; succeeded = 0; before = -1; if (typeof progress !== "undefined" && progress !== null) { progress.stop(); } console.log(`Attempt #${++count}`); progress = _progress.default.make({ count: repos.length }); progress.start(); mulligan = true; while((group = groups[index]) != null && (succeeded > before || mulligan)){ if (succeeded === before) { mulligan = false; } before = succeeded; failed = []; pending = function() { var len2, m, results; results = []; for(m = 0, len2 = group.length; m < len2; m++){ repo = group[m]; results.push(function(repo) { return limiter(async function() { var error, result; _kaiko.default.debug({ repo, command }); if (failures[repo] <= retries) { try { result = await _scripts.Script.run(command, { cwd: repo }); _kaiko.default.debug({ repo, result }); succeeded++; return progress.increment(); } catch (error1) { error = error1; _kaiko.default.error({ repo: repo, message: error.message, error: error }); return push(failed, repo); } } else { _kaiko.default.error({ repo: repo, failures: failures[repo], retries: retries, message: "Too many failures" }); return push(failed, repo); } }); }(repo)); } return results; }(); await Promise.all(pending); // demote failures if (succeeded > before && failed.length > 0) { if (groups[name1 = index + 1] == null) { groups[name1] = []; } for(m = 0, len2 = failed.length; m < len2; m++){ repo = failed[m]; _kaiko.default.debug({ message: "demoting repo", repo }); failures[repo]++; remove(group, repo); push(groups[index + 1], repo); } } index++; } } progress.stop(); for(n = 0, len3 = failed.length; n < len3; n++){ repo = failed[n]; _kaiko.default.error({ console: true, repo: repo, message: "failed" }); } _kaiko.default.info({ console: true, message: `succeeded: ${succeeded}, failed: ${repos.length - succeeded}` }); memos[key] = groups; return _zephyr.default.write(".tempo/memos.json", memos); }); (0, _generic.generic)(run, has("serial"), function({ serial, ...options }) { return Repos.run({ batch: 1, ...options }); }); (0, _generic.generic)(run, has([ "command", "args" ]), function({ command, args, ...options }) { if (_type.isObject(command)) { ({ command } = command); options = { ...options, ...command.options }; } return Repos.run({ command: _scripts.Script.expand(command, args), ...options }); }); (0, _generic.generic)(run, has("script"), async function({ script, ...options }) { return Repos.run({ command: await _scripts.Scripts.find(script), ...options }); }); return run; }({}) }; Repo = { parse: function(specifier) { var name, organization; [organization, name] = specifier.split("/"); return { organization, name }; }, same: function(a, b) { return a.organization === b.organization && a.name === b.name; }, save: function(repo) { return _zephyr.default.update(Repos.path, function(repos) { var _repo, k, len, results; results = []; for(k = 0, len = repos.length; k < len; k++){ _repo = repos[k]; if (Repo.same(_repo, repo)) { results.push(repo); } else { results.push(_repo); } } return results; }); }, tag: function(repo, tags) { if (repo.tags == null) { repo.tags = []; } repo.tags = Array.from(new Set([ ...repo.tags, ...tags ])); return Repo.save(repo); }, untag: function(repo, tags) { repo.tags = function() { var k, len, ref, results, tag; ref = repo.tags; results = []; for(k = 0, len = ref.length; k < len; k++){ tag = ref[k]; if (!(indexOf.call(tags, tag) >= 0)) { results.push(tag); } } return results; }(); return Repo.save(repo); }, changed: async function(name) { try { // returns non-zero status if there are changes in the repo await _scripts.Script.run("git diff-index --quiet HEAD", { cwd: name }); return false; } catch (error1) { return true; } } }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"","sourceRoot":"","sources":["/@dashkite/tempo/src/helpers/repos.coffee"],"names":[],"mappings":"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","sourcesContent":["import Path from \"node:path\"\nimport Crypto from \"node:crypto\"\nimport * as Fn from \"@dashkite/joy/function\"\nimport * as It from \"@dashkite/joy/iterable\"\nimport * as Type from \"@dashkite/joy/type\"\nimport * as Text from \"@dashkite/joy/text\"\nimport { generic } from \"@dashkite/joy/generic\"\nimport { convert } from \"@dashkite/bake\"\nimport Zephyr from \"@dashkite/zephyr\"\nimport log from \"@dashkite/kaiko\"\nimport pLimit from \"p-limit\"\nimport Progress from \"./progress\"\nimport { Scripts, Script } from \"./scripts\"\n\nhas = ( keys ) -> \n  if !( Type.isArray keys )\n    keys = [ keys ]\n  ( value ) -> keys.every ( key ) -> value[ key ]?\n\npush = ( stack, value ) -> stack.unshift value ; value\n\nremove = ( list, target ) ->\n  if ( index = list.indexOf target ) > -1\n    list.splice index, 1\n\nslice = ( stack, start, skip ) ->\n  stack.slice start, start + skip\n\npartition = ( size, list ) ->\n  i = 0\n  j = Math.ceil list.length / size\n  while i < j\n    yield slice list, ( i++ * size ), size\n\n\nHash =\n\n  md5: ( buffer ) ->\n    convert from: \"bytes\", to: \"base36\",\n      new Uint8Array do ->\n        Crypto\n          .createHash \"md5\"\n          .update buffer\n          .digest()\n          .buffer\n\n  array: ( array ) ->\n    Text.truncate 8, \n      Hash.md5 array.sort().join \",\"\n\nMemos =\n\n  path: Path.join \".tempo\", \"memos.json\"\n\nRepos =\n  \n  path: Path.join \".tempo\", \"repos.yaml\"\n\n  initialize: ->\n    Zephyr.update Repos.path, ( repos ) -> repos ?= []\n\n  load: -> Zephyr.read Repos.path\n\n  get: ( name ) ->\n    repos = await do Repos.load\n    repos.find ( repo ) -> repo.name == name\n\n  add: ({ organization, name }) ->\n    Zephyr.update Repos.path, ( repos ) ->\n      repos.push { organization, name }\n      repos\n\n  remove: ({ organization, name }) ->\n    Zephyr.update Repos.path, ( repos ) ->\n      repo = repos.find ( repo ) ->\n        repo.organization == organization &&\n          repo.name == name\n      remove repos, repo\n      repos\n\n  tag: ( repos, tags ) ->\n    for repo in repos\n      await Repo.tag repo, tags\n\n  untag: ( repos, tags ) ->\n    for repo in repos\n      await Repo.untag repo, tags\n\n  find: do ({ find } = {}) ->\n\n    find = generic\n      name: \"Repos.find\"\n      default: -> do Repos.load\n\n    generic find,\n      ( has \"repos\" ),\n      ({ repos, options... }) -> \n        do Fn.flow [\n          -> Repos.find options\n          It.select ( repo ) -> repo.name in repos\n        ]\n        \n    generic find,\n      ( has \"include\" ),\n      ({ include, options... }) ->\n        repos = await Zephyr.read include\n        Repos.find { repos, options... }\n\n    generic find,\n      ( has [ \"repos\", \"include\" ] ),\n      ({ repos, include }) ->\n        do Fn.flow [\n          Repos.find include: repos\n          ( result ) -> \n            result.concat await Repos.find { include }\n        ]\n\n    generic find,\n      ( has \"tags\" ),\n      ({ tags, options... }) ->\n        do Fn.flow [\n          -> Repos.find options\n          It.select ( repo ) -> \n            repo.tags? && ( tags.some ( tag ) -> tag in repo.tags )\n        ]\n\n    generic find,\n      ( has \"exclude\" ),\n      ({ exclude, options... }) ->\n        exclude = await Zephyr.read exclude\n        do Fn.flow [\n          -> Repos.find options\n          It.select ( repo ) -> !( repo.name in exclude )\n        ]\n\n    find\n\n  run: do ({ run } = {}) ->\n  \n    run = generic name: \"Repos.run\"\n\n    generic run, Type.isObject,\n      ({ repos, command, key, batch, retry }) ->\n\n        batch ?= 6 # max parallel builds\n        retry ?= true\n\n        if retry\n          memos = await Zephyr.read Memos.path\n          memos ?= {}\n          groups = memos[ key ]\n\n        # default the trivial group\n        groups ?= [( repos.map ({ name }) -> name )]\n\n        # check for missing repos\n        # add to first group if we find any\n        for repo in repos\n          found = ( groups.find ( group ) -> repo.name in group )?\n          unless found\n            push groups[ 0 ], repo.name\n        \n        groups = groups\n          # remove repos that are not in the target repos list\n          .map ( group ) ->\n            group.filter ( name ) ->\n              ( repos.find ( repo ) -> repo.name == name )?\n          # remove empty groups since they will halt the run loop\n          .filter ( group ) -> group.length != 0\n\n        failed = undefined\n        hash = undefined\n        retries = if retry then 6 else 0\n        history = []\n\n        # initialize failures lookup\n        failures = {}\n        ( failures[ repo.name ] = 0 ) for repo in repos  \n\n  \n\n        log.info \n          console: true\n          message: \"Running [ #{ Text.elide 40, \"...\", command } ]\"\n          command: command\n\n        limiter = pLimit batch\n\n        done = ->\n          failed? &&\n            (( failed.length == 0 ) ||\n              (( hash = Hash.array failed ) in history ))\n\n        count = 0\n        while !done()\n\n          ( history.push hash ) if hash?\n          ( groups.push failed ) if failed?\n\n          index = 0\n          succeeded = 0\n          before = -1\n\n          progress.stop() if progress?\n          console.log \"Attempt ##{ ++count }\"\n          progress = Progress.make count: repos.length\n          progress.start()\n\n          mulligan = true\n          while ( group = groups[ index ])? && (( succeeded > before ) || mulligan)\n            mulligan = false if succeeded == before\n            before = succeeded\n            failed = []\n\n            pending = \n              for repo in group\n                do ( repo ) ->\n                  limiter ->\n                    log.debug { repo, command }\n                    if failures[ repo ] <= retries\n                      try\n                        result = await Script.run command, cwd: repo\n                        log.debug { repo, result }\n                        succeeded++\n                        progress.increment()\n                      catch error\n                        log.error\n                          repo: repo\n                          message: error.message\n                          error: error\n                        push failed, repo\n                    else\n                      log.error\n                        repo: repo \n                        failures: failures[ repo ]\n                        retries: retries\n                        message: \"Too many failures\"\n                      push failed, repo\n\n            await Promise.all pending\n\n            # demote failures\n            if ( succeeded > before ) && ( failed.length > 0 )\n              groups[ index + 1 ] ?= []\n              for repo in failed\n                log.debug {\n                  message: \"demoting repo\"\n                  repo\n                }\n                failures[ repo ]++\n                remove group, repo\n                push groups[ index + 1 ], repo\n\n            index++\n\n        progress.stop()\n\n        for repo in failed\n          log.error\n            console:true\n            repo: repo\n            message: \"failed\"\n\n\n        log.info \n          console: true\n          message: \"succeeded: #{ succeeded },\n            failed: #{ repos.length - succeeded }\"\n\n        memos[ key ] = groups\n        Zephyr.write \".tempo/memos.json\", memos\n\n    generic run, \n      ( has \"serial\" ),\n      ({ serial, options... }) ->\n        Repos.run { batch: 1, options... }\n\n    generic run, \n      ( has [ \"command\", \"args\"  ]),\n      ({ command, args, options... }) ->\n        if Type.isObject command\n          { command } = command\n          options = { options..., command.options... }\n        Repos.run {\n          command: Script.expand command, args\n          options...\n        }\n\n    generic run, \n      ( has \"script\" ),\n      ({ script, options... }) ->\n        Repos.run {\n          command: await Scripts.find script\n          options...\n        }\n\n    run\n\nRepo =\n\n  parse: ( specifier ) ->\n    [ organization, name ] = specifier.split \"/\"\n    { organization, name }\n\n  same: ( a, b ) ->\n    a.organization == b.organization && a.name == b.name\n\n  save: ( repo ) ->\n    Zephyr.update Repos.path, ( repos ) ->\n      for _repo in repos      \n        if Repo.same _repo, repo\n          repo\n        else\n          _repo\n\n  tag: ( repo, tags ) ->\n    repo.tags ?= []\n    repo.tags = Array.from new Set [ repo.tags..., tags... ]\n    Repo.save repo\n\n  untag: ( repo, tags ) ->\n    repo.tags = do ->\n      tag for tag in repo.tags when !( tag in tags )\n    Repo.save repo\n\n  changed: ( name ) ->\n    try\n      # returns non-zero status if there are changes in the repo\n      await Script.run \"git diff-index --quiet HEAD\", cwd: name\n      false\n    catch\n      true\n\nexport { Repos, Repo }"]} //# sourceURL=/@dashkite/tempo/src/helpers/repos.coffee //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiIiwic291cmNlcyI6WyIvQGRhc2hraXRlL3RlbXBvL3NyYy9oZWxwZXJzL3JlcG9zLmNvZmZlZSJdLCJzb3VyY2VSb290IjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFBhdGggZnJvbSBcIm5vZGU6cGF0aFwiXG5pbXBvcnQgQ3J5cHRvIGZyb20gXCJub2RlOmNyeXB0b1wiXG5pbXBvcnQgKiBhcyBGbiBmcm9tIFwiQGRhc2hraXRlL2pveS9mdW5jdGlvblwiXG5pbXBvcnQgKiBhcyBJdCBmcm9tIFwiQGRhc2hraXRlL2pveS9pdGVyYWJsZVwiXG5pbXBvcnQgKiBhcyBUeXBlIGZyb20gXCJAZGFzaGtpdGUvam95L3R5cGVcIlxuaW1wb3J0ICogYXMgVGV4dCBmcm9tIFwiQGRhc2hraXRlL2pveS90ZXh0XCJcbmltcG9ydCB7IGdlbmVyaWMgfSBmcm9tIFwiQGRhc2hraXRlL2pveS9nZW5lcmljXCJcbmltcG9ydCB7IGNvbnZlcnQgfSBmcm9tIFwiQGRhc2hraXRlL2Jha2VcIlxuaW1wb3J0IFplcGh5ciBmcm9tIFwiQGRhc2hraXRlL3plcGh5clwiXG5pbXBvcnQgbG9nIGZyb20gXCJAZGFzaGtpdGUva2Fpa29cIlxuaW1wb3J0IHBMaW1pdCBmcm9tIFwicC1saW1pdFwiXG5pbXBvcnQgUHJvZ3Jlc3MgZnJvbSBcIi4vcHJvZ3Jlc3NcIlxuaW1wb3J0IHsgU2NyaXB0cywgU2NyaXB0IH0gZnJvbSBcIi4vc2NyaXB0c1wiXG5cbmhhcyA9ICgga2V5cyApIC0+IFxuICBpZiAhKCBUeXBlLmlzQXJyYXkga2V5cyApXG4gICAga2V5cyA9IFsga2V5cyBdXG4gICggdmFsdWUgKSAtPiBrZXlzLmV2ZXJ5ICgga2V5ICkgLT4gdmFsdWVbIGtleSBdP1xuXG5wdXNoID0gKCBzdGFjaywgdmFsdWUgKSAtPiBzdGFjay51bnNoaWZ0IHZhbHVlIDsgdmFsdWVcblxucmVtb3ZlID0gKCBsaXN0LCB0YXJnZXQgKSAtPlxuICBpZiAoIGluZGV4ID0gbGlzdC5pbmRleE9mIHRhcmdldCApID4gLTFcbiAgICBsaXN0LnNwbGljZSBpbmRleCwgMVxuXG5zbGljZSA9ICggc3RhY2ssIHN0YXJ0LCBza2lwICkgLT5cbiAgc3RhY2suc2xpY2Ugc3RhcnQsIHN0YXJ0ICsgc2tpcFxuXG5wYXJ0aXRpb24gPSAoIHNpemUsIGxpc3QgKSAtPlxuICBpID0gMFxuICBqID0gTWF0aC5jZWlsIGxpc3QubGVuZ3RoIC8gc2l6ZVxuICB3aGlsZSBpIDwgalxuICAgIHlpZWxkIHNsaWNlIGxpc3QsICggaSsrICogc2l6ZSApLCBzaXplXG5cblxuSGFzaCA9XG5cbiAgbWQ1OiAoIGJ1ZmZlciApIC0+XG4gICAgY29udmVydCBmcm9tOiBcImJ5dGVzXCIsIHRvOiBcImJhc2UzNlwiLFxuICAgICAgbmV3IFVpbnQ4QXJyYXkgZG8gLT5cbiAgICAgICAgQ3J5cHRvXG4gICAgICAgICAgLmNyZWF0ZUhhc2ggXCJtZDVcIlxuICAgICAgICAgIC51cGRhdGUgYnVmZmVyXG4gICAgICAgICAgLmRpZ2VzdCgpXG4gICAgICAgICAgLmJ1ZmZlclxuXG4gIGFycmF5OiAoIGFycmF5ICkgLT5cbiAgICBUZXh0LnRydW5jYXRlIDgsIFxuICAgICAgSGFzaC5tZDUgYXJyYXkuc29ydCgpLmpvaW4gXCIsXCJcblxuTWVtb3MgPVxuXG4gIHBhdGg6IFBhdGguam9pbiBcIi50ZW1wb1wiLCBcIm1lbW9zLmpzb25cIlxuXG5SZXBvcyA9XG4gIFxuICBwYXRoOiBQYXRoLmpvaW4gXCIudGVtcG9cIiwgXCJyZXBvcy55YW1sXCJcblxuICBpbml0aWFsaXplOiAtPlxuICAgIFplcGh5ci51cGRhdGUgUmVwb3MucGF0aCwgKCByZXBvcyApIC0+IHJlcG9zID89IFtdXG5cbiAgbG9hZDogLT4gWmVwaHlyLnJlYWQgUmVwb3MucGF0aFxuXG4gIGdldDogKCBuYW1lICkgLT5cbiAgICByZXBvcyA9IGF3YWl0IGRvIFJlcG9zLmxvYWRcbiAgICByZXBvcy5maW5kICggcmVwbyApIC0+IHJlcG8ubmFtZSA9PSBuYW1lXG5cbiAgYWRkOiAoeyBvcmdhbml6YXRpb24sIG5hbWUgfSkgLT5cbiAgICBaZXBoeXIudXBkYXRlIFJlcG9zLnBhdGgsICggcmVwb3MgKSAtPlxuICAgICAgcmVwb3MucHVzaCB7IG9yZ2FuaXphdGlvbiwgbmFtZSB9XG4gICAgICByZXBvc1xuXG4gIHJlbW92ZTogKHsgb3JnYW5pemF0aW9uLCBuYW1lIH0pIC0+XG4gICAgWmVwaHlyLnVwZGF0ZSBSZXBvcy5wYXRoLCAoIHJlcG9zICkgLT5cbiAgICAgIHJlcG8gPSByZXBvcy5maW5kICggcmVwbyApIC0+XG4gICAgICAgIHJlcG8ub3JnYW5pemF0aW9uID09IG9yZ2FuaXphdGlvbiAmJlxuICAgICAgICAgIHJlcG8ubmFtZSA9PSBuYW1lXG4gICAgICByZW1vdmUgcmVwb3MsIHJlcG9cbiAgICAgIHJlcG9zXG5cbiAgdGFnOiAoIHJlcG9zLCB0YWdzICkgLT5cbiAgICBmb3IgcmVwbyBpbiByZXBvc1xuICAgICAgYXdhaXQgUmVwby50YWcgcmVwbywgdGFnc1xuXG4gIHVudGFnOiAoIHJlcG9zLCB0YWdzICkgLT5cbiAgICBmb3IgcmVwbyBpbiByZXBvc1xuICAgICAgYXdhaXQgUmVwby51bnRhZyByZXBvLCB0YWdzXG5cbiAgZmluZDogZG8gKHsgZmluZCB9ID0ge30pIC0+XG5cbiAgICBmaW5kID0gZ2VuZXJpY1xuICAgICAgbmFtZTogXCJSZXBvcy5maW5kXCJcbiAgICAgIGRlZmF1bHQ6IC0+IGRvIFJlcG9zLmxvYWRcblxuICAgIGdlbmVyaWMgZmluZCxcbiAgICAgICggaGFzIFwicmVwb3NcIiApLFxuICAgICAgKHsgcmVwb3MsIG9wdGlvbnMuLi4gfSkgLT4gXG4gICAgICAgIGRvIEZuLmZsb3cgW1xuICAgICAgICAgIC0+IFJlcG9zLmZpbmQgb3B0aW9uc1xuICAgICAgICAgIEl0LnNlbGVjdCAoIHJlcG8gKSAtPiByZXBvLm5hbWUgaW4gcmVwb3NcbiAgICAgICAgXVxuICAgICAgICBcbiAgICBnZW5lcmljIGZpbmQsXG4gICAgICAoIGhhcyBcImluY2x1ZGVcIiApLFxuICAgICAgKHsgaW5jbHVkZSwgb3B0aW9ucy4uLiB9KSAtPlxuICAgICAgICByZXBvcyA9IGF3YWl0IFplcGh5ci5yZWFkIGluY2x1ZGVcbiAgICAgICAgUmVwb3MuZmluZCB7IHJlcG9zLCBvcHRpb25zLi4uIH1cblxuICAgIGdlbmVyaWMgZmluZCxcbiAgICAgICggaGFzIFsgXCJyZXBvc1wiLCBcImluY2x1ZGVcIiBdICksXG4gICAgICAoeyByZXBvcywgaW5jbHVkZSB9KSAtPlxuICAgICAgICBkbyBGbi5mbG93IFtcbiAgICAgICAgICBSZXBvcy5maW5kIGluY2x1ZGU6IHJlcG9zXG4gICAgICAgICAgKCByZXN1bHQgKSAtPiBcbiAgICAgICAgICAgIHJlc3VsdC5jb25jYXQgYXdhaXQgUmVwb3MuZmluZCB7IGluY2x1ZGUgfVxuICAgICAgICBdXG5cbiAgICBnZW5lcmljIGZpbmQsXG4gICAgICAoIGhhcyBcInRhZ3NcIiApLFxuICAgICAgKHsgdGFncywgb3B0aW9ucy4uLiB9KSAtPlxuICAgICAgICBkbyBGbi5mbG93IFtcbiAgICAgICAgICAtPiBSZXBvcy5maW5kIG9wdGlvbnNcbiAgICAgICAgICBJdC5zZWxlY3QgKCByZXBvICkgLT4gXG4gICAgICAgICAgICByZXBvLnRhZ3M/ICYmICggdGFncy5zb21lICggdGFnICkgLT4gdGFnIGluIHJlcG8udGFncyApXG4gICAgICAgIF1cblxuICAgIGdlbmVyaWMgZmluZCxcbiAgI