mirror of
				https://github.com/actions/setup-go.git
				synced 2025-11-04 04:23:42 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			290 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			290 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
'use strict';
 | 
						|
 | 
						|
Object.defineProperty(exports, '__esModule', {
 | 
						|
  value: true
 | 
						|
});
 | 
						|
exports.default = void 0;
 | 
						|
 | 
						|
function _path() {
 | 
						|
  const data = _interopRequireDefault(require('path'));
 | 
						|
 | 
						|
  _path = function _path() {
 | 
						|
    return data;
 | 
						|
  };
 | 
						|
 | 
						|
  return data;
 | 
						|
}
 | 
						|
 | 
						|
function _stream() {
 | 
						|
  const data = require('stream');
 | 
						|
 | 
						|
  _stream = function _stream() {
 | 
						|
    return data;
 | 
						|
  };
 | 
						|
 | 
						|
  return data;
 | 
						|
}
 | 
						|
 | 
						|
function _worker_threads() {
 | 
						|
  const data = require('worker_threads');
 | 
						|
 | 
						|
  _worker_threads = function _worker_threads() {
 | 
						|
    return data;
 | 
						|
  };
 | 
						|
 | 
						|
  return data;
 | 
						|
}
 | 
						|
 | 
						|
function _mergeStream() {
 | 
						|
  const data = _interopRequireDefault(require('merge-stream'));
 | 
						|
 | 
						|
  _mergeStream = function _mergeStream() {
 | 
						|
    return data;
 | 
						|
  };
 | 
						|
 | 
						|
  return data;
 | 
						|
}
 | 
						|
 | 
						|
var _types = require('../types');
 | 
						|
 | 
						|
function _interopRequireDefault(obj) {
 | 
						|
  return obj && obj.__esModule ? obj : {default: obj};
 | 
						|
}
 | 
						|
 | 
						|
function _objectSpread(target) {
 | 
						|
  for (var i = 1; i < arguments.length; i++) {
 | 
						|
    var source = arguments[i] != null ? arguments[i] : {};
 | 
						|
    var ownKeys = Object.keys(source);
 | 
						|
    if (typeof Object.getOwnPropertySymbols === 'function') {
 | 
						|
      ownKeys = ownKeys.concat(
 | 
						|
        Object.getOwnPropertySymbols(source).filter(function(sym) {
 | 
						|
          return Object.getOwnPropertyDescriptor(source, sym).enumerable;
 | 
						|
        })
 | 
						|
      );
 | 
						|
    }
 | 
						|
    ownKeys.forEach(function(key) {
 | 
						|
      _defineProperty(target, key, source[key]);
 | 
						|
    });
 | 
						|
  }
 | 
						|
  return target;
 | 
						|
}
 | 
						|
 | 
						|
function _defineProperty(obj, key, value) {
 | 
						|
  if (key in obj) {
 | 
						|
    Object.defineProperty(obj, key, {
 | 
						|
      value: value,
 | 
						|
      enumerable: true,
 | 
						|
      configurable: true,
 | 
						|
      writable: true
 | 
						|
    });
 | 
						|
  } else {
 | 
						|
    obj[key] = value;
 | 
						|
  }
 | 
						|
  return obj;
 | 
						|
}
 | 
						|
 | 
						|
class ExperimentalWorker {
 | 
						|
  constructor(options) {
 | 
						|
    _defineProperty(this, '_worker', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_options', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_onProcessEnd', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_request', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_retries', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_stderr', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_stdout', void 0);
 | 
						|
 | 
						|
    _defineProperty(this, '_fakeStream', void 0);
 | 
						|
 | 
						|
    this._options = options;
 | 
						|
    this._request = null;
 | 
						|
    this._stderr = null;
 | 
						|
    this._stdout = null;
 | 
						|
    this._fakeStream = null;
 | 
						|
    this.initialize();
 | 
						|
  }
 | 
						|
 | 
						|
  initialize() {
 | 
						|
    this._worker = new (_worker_threads()).Worker(
 | 
						|
      _path().default.resolve(__dirname, './threadChild.js'),
 | 
						|
      {
 | 
						|
        eval: false,
 | 
						|
        stderr: true,
 | 
						|
        stdout: true,
 | 
						|
        workerData: _objectSpread(
 | 
						|
          {
 | 
						|
            cwd: process.cwd(),
 | 
						|
            env: _objectSpread({}, process.env, {
 | 
						|
              JEST_WORKER_ID: String(this._options.workerId + 1) // 0-indexed workerId, 1-indexed JEST_WORKER_ID
 | 
						|
            }),
 | 
						|
            // Suppress --debug / --inspect flags while preserving others (like --harmony).
 | 
						|
            execArgv: process.execArgv.filter(
 | 
						|
              v => !/^--(debug|inspect)/.test(v)
 | 
						|
            ),
 | 
						|
            silent: true
 | 
						|
          },
 | 
						|
          this._options.forkOptions
 | 
						|
        )
 | 
						|
      }
 | 
						|
    );
 | 
						|
 | 
						|
    if (this._worker.stdout) {
 | 
						|
      if (!this._stdout) {
 | 
						|
        // We need to add a permanent stream to the merged stream to prevent it
 | 
						|
        // from ending when the subprocess stream ends
 | 
						|
        this._stdout = (0, _mergeStream().default)(this._getFakeStream());
 | 
						|
      }
 | 
						|
 | 
						|
      this._stdout.add(this._worker.stdout);
 | 
						|
    }
 | 
						|
 | 
						|
    if (this._worker.stderr) {
 | 
						|
      if (!this._stderr) {
 | 
						|
        // We need to add a permanent stream to the merged stream to prevent it
 | 
						|
        // from ending when the subprocess stream ends
 | 
						|
        this._stderr = (0, _mergeStream().default)(this._getFakeStream());
 | 
						|
      }
 | 
						|
 | 
						|
      this._stderr.add(this._worker.stderr);
 | 
						|
    }
 | 
						|
 | 
						|
    this._worker.on('message', this.onMessage.bind(this));
 | 
						|
 | 
						|
    this._worker.on('exit', this.onExit.bind(this));
 | 
						|
 | 
						|
    this._worker.postMessage([
 | 
						|
      _types.CHILD_MESSAGE_INITIALIZE,
 | 
						|
      false,
 | 
						|
      this._options.workerPath,
 | 
						|
      this._options.setupArgs
 | 
						|
    ]);
 | 
						|
 | 
						|
    this._retries++; // If we exceeded the amount of retries, we will emulate an error reply
 | 
						|
    // coming from the child. This avoids code duplication related with cleaning
 | 
						|
    // the queue, and scheduling the next call.
 | 
						|
 | 
						|
    if (this._retries > this._options.maxRetries) {
 | 
						|
      const error = new Error('Call retries were exceeded');
 | 
						|
      this.onMessage([
 | 
						|
        _types.PARENT_MESSAGE_CLIENT_ERROR,
 | 
						|
        error.name,
 | 
						|
        error.message,
 | 
						|
        error.stack,
 | 
						|
        {
 | 
						|
          type: 'WorkerError'
 | 
						|
        }
 | 
						|
      ]);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  _shutdown() {
 | 
						|
    // End the permanent stream so the merged stream end too
 | 
						|
    if (this._fakeStream) {
 | 
						|
      this._fakeStream.end();
 | 
						|
 | 
						|
      this._fakeStream = null;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  onMessage(response) {
 | 
						|
    let error;
 | 
						|
 | 
						|
    switch (response[0]) {
 | 
						|
      case _types.PARENT_MESSAGE_OK:
 | 
						|
        this._onProcessEnd(null, response[1]);
 | 
						|
 | 
						|
        break;
 | 
						|
 | 
						|
      case _types.PARENT_MESSAGE_CLIENT_ERROR:
 | 
						|
        error = response[4];
 | 
						|
 | 
						|
        if (error != null && typeof error === 'object') {
 | 
						|
          const extra = error; // @ts-ignore: no index
 | 
						|
 | 
						|
          const NativeCtor = global[response[1]];
 | 
						|
          const Ctor = typeof NativeCtor === 'function' ? NativeCtor : Error;
 | 
						|
          error = new Ctor(response[2]);
 | 
						|
          error.type = response[1];
 | 
						|
          error.stack = response[3];
 | 
						|
 | 
						|
          for (const key in extra) {
 | 
						|
            // @ts-ignore: no index
 | 
						|
            error[key] = extra[key];
 | 
						|
          }
 | 
						|
        }
 | 
						|
 | 
						|
        this._onProcessEnd(error, null);
 | 
						|
 | 
						|
        break;
 | 
						|
 | 
						|
      case _types.PARENT_MESSAGE_SETUP_ERROR:
 | 
						|
        error = new Error('Error when calling setup: ' + response[2]); // @ts-ignore: adding custom properties to errors.
 | 
						|
 | 
						|
        error.type = response[1];
 | 
						|
        error.stack = response[3];
 | 
						|
 | 
						|
        this._onProcessEnd(error, null);
 | 
						|
 | 
						|
        break;
 | 
						|
 | 
						|
      default:
 | 
						|
        throw new TypeError('Unexpected response from worker: ' + response[0]);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  onExit(exitCode) {
 | 
						|
    if (exitCode !== 0) {
 | 
						|
      this.initialize();
 | 
						|
 | 
						|
      if (this._request) {
 | 
						|
        this._worker.postMessage(this._request);
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      this._shutdown();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  send(request, onProcessStart, onProcessEnd) {
 | 
						|
    onProcessStart(this);
 | 
						|
 | 
						|
    this._onProcessEnd = (...args) => {
 | 
						|
      // Clean the request to avoid sending past requests to workers that fail
 | 
						|
      // while waiting for a new request (timers, unhandled rejections...)
 | 
						|
      this._request = null;
 | 
						|
      return onProcessEnd(...args);
 | 
						|
    };
 | 
						|
 | 
						|
    this._request = request;
 | 
						|
    this._retries = 0;
 | 
						|
 | 
						|
    this._worker.postMessage(request);
 | 
						|
  }
 | 
						|
 | 
						|
  getWorkerId() {
 | 
						|
    return this._options.workerId;
 | 
						|
  }
 | 
						|
 | 
						|
  getStdout() {
 | 
						|
    return this._stdout;
 | 
						|
  }
 | 
						|
 | 
						|
  getStderr() {
 | 
						|
    return this._stderr;
 | 
						|
  }
 | 
						|
 | 
						|
  _getFakeStream() {
 | 
						|
    if (!this._fakeStream) {
 | 
						|
      this._fakeStream = new (_stream()).PassThrough();
 | 
						|
    }
 | 
						|
 | 
						|
    return this._fakeStream;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
exports.default = ExperimentalWorker;
 |