@mrgrain/cdk-esbuild 4.0.0-alpha.2 → 4.0.0-alpha.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,12 +1,31 @@
1
1
  "use strict";
2
- /* eslint-disable import/no-extraneous-dependencies */
3
2
  Object.defineProperty(exports, "__esModule", { value: true });
4
3
  exports.transformSync = exports.formatMessagesSync = exports.buildSync = void 0;
4
+ const which = require("which");
5
+ /* eslint-disable import/no-extraneous-dependencies */
5
6
  function esbuild() {
6
- // eslint-disable-next-line @typescript-eslint/no-require-imports
7
- return require('esbuild');
7
+ try {
8
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
9
+ return require('esbuild');
10
+ }
11
+ catch (_one) {
12
+ /**
13
+ * esbuild is not installed. We fallback to use a (hopefully available)
14
+ * global binary and our bundled polyfill.
15
+ */
16
+ try {
17
+ if (!process.env?.ESBUILD_BINARY_PATH) {
18
+ process.env.ESBUILD_BINARY_PATH = which.sync('esbuild');
19
+ }
20
+ }
21
+ catch (_two) {
22
+ throw new Error('The esbuild binary could not be found in path. Please follow the instructions to ensure esbuild is correctly installed.');
23
+ }
24
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
25
+ return require('./esbuild-polyfill');
26
+ }
8
27
  }
9
28
  exports.buildSync = esbuild().buildSync;
10
29
  exports.formatMessagesSync = esbuild().formatMessagesSync;
11
30
  exports.transformSync = esbuild().transformSync;
12
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXNidWlsZC13cmFwcGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2VzYnVpbGQtd3JhcHBlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsc0RBQXNEOzs7QUFFdEQsU0FBUyxPQUFPO0lBQ2QsaUVBQWlFO0lBQ2pFLE9BQU8sT0FBTyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0FBQzVCLENBQUM7QUFFWSxRQUFBLFNBQVMsR0FBRyxPQUFPLEVBQUUsQ0FBQyxTQUFTLENBQUM7QUFDaEMsUUFBQSxrQkFBa0IsR0FBRyxPQUFPLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztBQUNsRCxRQUFBLGFBQWEsR0FBRyxPQUFPLEVBQUUsQ0FBQyxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBpbXBvcnQvbm8tZXh0cmFuZW91cy1kZXBlbmRlbmNpZXMgKi9cblxuZnVuY3Rpb24gZXNidWlsZCgpIHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHNcbiAgcmV0dXJuIHJlcXVpcmUoJ2VzYnVpbGQnKTtcbn1cblxuZXhwb3J0IGNvbnN0IGJ1aWxkU3luYyA9IGVzYnVpbGQoKS5idWlsZFN5bmM7XG5leHBvcnQgY29uc3QgZm9ybWF0TWVzc2FnZXNTeW5jID0gZXNidWlsZCgpLmZvcm1hdE1lc3NhZ2VzU3luYztcbmV4cG9ydCBjb25zdCB0cmFuc2Zvcm1TeW5jID0gZXNidWlsZCgpLnRyYW5zZm9ybVN5bmM7XG4iXX0=
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXNidWlsZC13cmFwcGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2VzYnVpbGQtd3JhcHBlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwrQkFBK0I7QUFFL0Isc0RBQXNEO0FBRXRELFNBQVMsT0FBTztJQUNkLElBQUk7UUFDRixpRUFBaUU7UUFDakUsT0FBTyxPQUFPLENBQUMsU0FBUyxDQUFDLENBQUM7S0FDM0I7SUFBQyxPQUFPLElBQUksRUFBRTtRQUNiOzs7V0FHRztRQUNILElBQUk7WUFDRixJQUFJLENBQUMsT0FBTyxDQUFDLEdBQUcsRUFBRSxtQkFBbUIsRUFBRTtnQkFDckMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxtQkFBbUIsR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO2FBQ3pEO1NBQ0Y7UUFBQyxPQUFPLElBQUksRUFBRTtZQUNiLE1BQU0sSUFBSSxLQUFLLENBQUMseUhBQXlILENBQUMsQ0FBQztTQUM1STtRQUVELGlFQUFpRTtRQUNqRSxPQUFPLE9BQU8sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO0tBQ3RDO0FBQ0gsQ0FBQztBQUVZLFFBQUEsU0FBUyxHQUFHLE9BQU8sRUFBRSxDQUFDLFNBQVMsQ0FBQztBQUNoQyxRQUFBLGtCQUFrQixHQUFHLE9BQU8sRUFBRSxDQUFDLGtCQUFrQixDQUFDO0FBQ2xELFFBQUEsYUFBYSxHQUFHLE9BQU8sRUFBRSxDQUFDLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHdoaWNoIGZyb20gJ3doaWNoJztcblxuLyogZXNsaW50LWRpc2FibGUgaW1wb3J0L25vLWV4dHJhbmVvdXMtZGVwZW5kZW5jaWVzICovXG5cbmZ1bmN0aW9uIGVzYnVpbGQoKSB7XG4gIHRyeSB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHNcbiAgICByZXR1cm4gcmVxdWlyZSgnZXNidWlsZCcpO1xuICB9IGNhdGNoIChfb25lKSB7XG4gICAgLyoqXG4gICAgICogZXNidWlsZCBpcyBub3QgaW5zdGFsbGVkLiBXZSBmYWxsYmFjayB0byB1c2UgYSAoaG9wZWZ1bGx5IGF2YWlsYWJsZSlcbiAgICAgKiBnbG9iYWwgYmluYXJ5IGFuZCBvdXIgYnVuZGxlZCBwb2x5ZmlsbC5cbiAgICAgKi9cbiAgICB0cnkge1xuICAgICAgaWYgKCFwcm9jZXNzLmVudj8uRVNCVUlMRF9CSU5BUllfUEFUSCkge1xuICAgICAgICBwcm9jZXNzLmVudi5FU0JVSUxEX0JJTkFSWV9QQVRIID0gd2hpY2guc3luYygnZXNidWlsZCcpO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKF90d28pIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignVGhlIGVzYnVpbGQgYmluYXJ5IGNvdWxkIG5vdCBiZSBmb3VuZCBpbiBwYXRoLiBQbGVhc2UgZm9sbG93IHRoZSBpbnN0cnVjdGlvbnMgdG8gZW5zdXJlIGVzYnVpbGQgaXMgY29ycmVjdGx5IGluc3RhbGxlZC4nKTtcbiAgICB9XG5cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0c1xuICAgIHJldHVybiByZXF1aXJlKCcuL2VzYnVpbGQtcG9seWZpbGwnKTtcbiAgfVxufVxuXG5leHBvcnQgY29uc3QgYnVpbGRTeW5jID0gZXNidWlsZCgpLmJ1aWxkU3luYztcbmV4cG9ydCBjb25zdCBmb3JtYXRNZXNzYWdlc1N5bmMgPSBlc2J1aWxkKCkuZm9ybWF0TWVzc2FnZXNTeW5jO1xuZXhwb3J0IGNvbnN0IHRyYW5zZm9ybVN5bmMgPSBlc2J1aWxkKCkudHJhbnNmb3JtU3luYztcbiJdfQ==
@@ -70,7 +70,7 @@ class InlineJavaScriptCode extends BaseInlineCode {
70
70
  }
71
71
  exports.InlineJavaScriptCode = InlineJavaScriptCode;
72
72
  _a = JSII_RTTI_SYMBOL_1;
73
- InlineJavaScriptCode[_a] = { fqn: "@mrgrain/cdk-esbuild.InlineJavaScriptCode", version: "4.0.0-alpha.2" };
73
+ InlineJavaScriptCode[_a] = { fqn: "@mrgrain/cdk-esbuild.InlineJavaScriptCode", version: "4.0.0-alpha.5" };
74
74
  /**
75
75
  * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.
76
76
  *
@@ -101,7 +101,7 @@ class InlineJsxCode extends BaseInlineCode {
101
101
  }
102
102
  exports.InlineJsxCode = InlineJsxCode;
103
103
  _b = JSII_RTTI_SYMBOL_1;
104
- InlineJsxCode[_b] = { fqn: "@mrgrain/cdk-esbuild.InlineJsxCode", version: "4.0.0-alpha.2" };
104
+ InlineJsxCode[_b] = { fqn: "@mrgrain/cdk-esbuild.InlineJsxCode", version: "4.0.0-alpha.5" };
105
105
  /**
106
106
  * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.
107
107
  *
@@ -132,7 +132,7 @@ class InlineTypeScriptCode extends BaseInlineCode {
132
132
  }
133
133
  exports.InlineTypeScriptCode = InlineTypeScriptCode;
134
134
  _c = JSII_RTTI_SYMBOL_1;
135
- InlineTypeScriptCode[_c] = { fqn: "@mrgrain/cdk-esbuild.InlineTypeScriptCode", version: "4.0.0-alpha.2" };
135
+ InlineTypeScriptCode[_c] = { fqn: "@mrgrain/cdk-esbuild.InlineTypeScriptCode", version: "4.0.0-alpha.5" };
136
136
  /**
137
137
  * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.
138
138
  *
@@ -163,5 +163,5 @@ class InlineTsxCode extends BaseInlineCode {
163
163
  }
164
164
  exports.InlineTsxCode = InlineTsxCode;
165
165
  _d = JSII_RTTI_SYMBOL_1;
166
- InlineTsxCode[_d] = { fqn: "@mrgrain/cdk-esbuild.InlineTsxCode", version: "4.0.0-alpha.2" };
166
+ InlineTsxCode[_d] = { fqn: "@mrgrain/cdk-esbuild.InlineTsxCode", version: "4.0.0-alpha.5" };
167
167
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inline-code.js","sourceRoot":"","sources":["../src/inline-code.ts"],"names":[],"mappings":";;;;;AAAA,uDAAoD;AAEpD,uDAAkD;AAClD,qDAAsD;AA4BtD,MAAe,cAAe,SAAQ,uBAAU;IAC9C,YACE,IAAY,EACZ,KAAuB;QAGvB,MAAM,EAAE,WAAW,GAAG,+BAAa,EAAE,gBAAgB,GAAG,EAAE,EAAE,GAAG,KAAK,CAAC;QAErE,IAAI;YACF,MAAM,eAAe,GAAG,WAAW,CAAC,IAAI,EAAE;gBACxC,GAAG,gBAAgB;aACpB,CAAC,CAAC;YACH,mCAAkB,CAAC,eAAe,EAAE,EAAE,MAAM,EAAE,YAAY,EAAE,CAAC,CAAC;YAE9D,KAAK,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;SAC7B;QAAC,OAAO,KAAK,EAAE;YACd,mCAAkB,CAAC,KAAyB,EAAE,EAAE,MAAM,EAAE,YAAY,EAAE,CAAC,CAAC;YAExE,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;SACnD;IACH,CAAC;CACF;AAED,SAAS,0BAA0B,CAAC,MAAW;IAC7C,OAAO,kBAAkB,IAAI,MAAM,IAAI,aAAa,IAAI,MAAM,CAAC;AACjE,CAAC;AAED,SAAS,gBAAgB,CAAC,MAAc,EAAE,KAA2C;IACnF,IAAI,CAAC,KAAK,EAAE;QACV,OAAO,EAAE,gBAAgB,EAAE,EAAE,MAAM,EAAE,EAAE,CAAC;KACzC;IAED,IAAI,CAAC,0BAA0B,CAAC,KAAK,CAAC,EAAG;QACvC,OAAO,EAAE,gBAAgB,EAAE,EAAE,MAAM,EAAE,GAAG,KAAK,EAAE,EAAE,CAAC;KACnD;IAED,OAAO;QACL,GAAG,KAAK;QACR,gBAAgB,EAAE;YAChB,MAAM;YACN,GAAG,KAAK,CAAC,gBAAgB;SAC1B;KACF,CAAC;AACJ,CAAC;AAGD;;;;GAIG;AACH,MAAa,oBAAqB,SAAQ,cAAc;IACtD;IACE;;;;OAIG;IACH,IAAY;IACZ;;;;;;;;;;OAUG;IACH,KAA2C;QAG3C,KAAK,CAAC,IAAI,EAAE,gBAAgB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;IAC7C,CAAC;;AAvBH,oDAwBC;;;AAED;;;;GAIG;AACH,MAAa,aAAc,SAAQ,cAAc;IAC/C;IACE;;;;OAIG;IACH,IAAY;IACZ;;;;;;;;;;OAUG;IACH,KAA2C;QAE3C,KAAK,CAAC,IAAI,EAAE,gBAAgB,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC;IAC9C,CAAC;;AAtBH,sCAuBC;;;AAED;;;;GAIG;AACH,MAAa,oBAAqB,SAAQ,cAAc;IACtD;IACE;;;;OAIG;IACH,IAAY;IACZ;;;;;;;;;;OAUG;IACH,KAA2C;QAE3C,KAAK,CAAC,IAAI,EAAE,gBAAgB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;IAC7C,CAAC;;AAtBH,oDAuBC;;;AAED;;;;GAIG;AACH,MAAa,aAAc,SAAQ,cAAc;IAC/C;IACE;;;;OAIG;IACH,IAAY;IACZ;;;;;;;;;;OAUG;IACH,KAA2C;QAE3C,KAAK,CAAC,IAAI,EAAE,gBAAgB,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC;IAC9C,CAAC;;AAtBH,sCAuBC","sourcesContent":["import { InlineCode } from 'aws-cdk-lib/aws-lambda';\nimport { TransformOptions, Loader, TransformFailure } from './esbuild-types';\nimport { transformSync } from './esbuild-wrapper';\nimport { printBuildMessages } from './formatMessages';\n\n/**\n * @stability experimental\n */\nexport interface TransformerProps {\n  /**\n   * Transform options passed on to esbuild. Please refer to the esbuild Transform API docs for details.\n   *\n   * @see https://esbuild.github.io/api/#transform-api\n   * @stability stable\n   */\n  readonly transformOptions?: TransformOptions;\n\n  /**\n   * Escape hatch to provide the bundler with a custom transform function.\n   * The function will receive the computed options from the bundler. It can use with these options as it wishes, however a TransformResult must be returned to integrate with CDK.\n   * Must throw a `TransformFailure` on failure to correctly inform the bundler.\n   *\n   * @stability experimental\n   * @type esbuild.transformSync\n   * @returns esbuild.TransformResult\n   * @throws esbuild.TransformFailure\n   * @default esbuild.transformSync\n   */\n  readonly transformFn?: any;\n}\n\nabstract class BaseInlineCode extends InlineCode {\n  public constructor(\n    code: string,\n    props: TransformerProps,\n  ) {\n\n    const { transformFn = transformSync, transformOptions = {} } = props;\n\n    try {\n      const transformedCode = transformFn(code, {\n        ...transformOptions,\n      });\n      printBuildMessages(transformedCode, { prefix: 'Transform ' });\n\n      super(transformedCode.code);\n    } catch (error) {\n      printBuildMessages(error as TransformFailure, { prefix: 'Transform ' });\n\n      throw new Error('Failed to transform InlineCode');\n    }\n  }\n}\n\nfunction instanceOfTransformerProps(object: any): object is TransformerProps {\n  return 'transformOptions' in object || 'transformFn' in object;\n}\n\nfunction transformerProps(loader: Loader, props?: TransformerProps | TransformOptions): TransformerProps {\n  if (!props) {\n    return { transformOptions: { loader } };\n  }\n\n  if (!instanceOfTransformerProps(props) ) {\n    return { transformOptions: { loader, ...props } };\n  }\n\n  return {\n    ...props,\n    transformOptions: {\n      loader,\n      ...props.transformOptions,\n    },\n  };\n}\n\n\n/**\n * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.\n *\n * @stability experimental\n */\nexport class InlineJavaScriptCode extends BaseInlineCode {\n  public constructor(\n    /**\n     * The inline code to be transformed.\n     *\n     * @stability experimental\n     */\n    code: string,\n    /**\n     * Support for `TransformOptions` is deprecated. Please provide `TransformerProps`!\n     *\n     * Props to change the behaviour of the transformer.\n     *\n     * Default values for `props.transformOptions`:\n     * - `loader='js'`\n     *\n     * @see https://esbuild.github.io/api/#transform-api\n     * @stability experimental\n     */\n    props?: TransformerProps | TransformOptions,\n  ) {\n\n    super(code, transformerProps('js', props));\n  }\n}\n\n/**\n * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.\n *\n * @stability experimental\n */\nexport class InlineJsxCode extends BaseInlineCode {\n  public constructor(\n    /**\n     * The inline code to be transformed.\n     *\n     * @stability experimental\n     */\n    code: string,\n    /**\n     * Support for `TransformOptions` is deprecated. Please provide `TransformerProps`!\n     *\n     * Props to change the behaviour of the transformer.\n     *\n     * Default values for `transformOptions`:\n     * - `loader='jsx'`\n     *\n     * @see https://esbuild.github.io/api/#transform-api\n     * @stability experimental\n     */\n    props?: TransformerProps | TransformOptions,\n  ) {\n    super(code, transformerProps('jsx', props));\n  }\n}\n\n/**\n * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.\n *\n * @stability experimental\n */\nexport class InlineTypeScriptCode extends BaseInlineCode {\n  public constructor(\n    /**\n     * The inline code to be transformed.\n     *\n     * @stability experimental\n     */\n    code: string,\n    /**\n     * Support for `TransformOptions` is deprecated. Please provide `TransformerProps`!\n     *\n     * Props to change the behaviour of the transformer.\n     *\n     * Default values for `transformOptions`:\n     * - `loader='ts'`\n     *\n     * @see https://esbuild.github.io/api/#transform-api\n     * @stability experimental\n     */\n    props?: TransformerProps | TransformOptions,\n  ) {\n    super(code, transformerProps('ts', props));\n  }\n}\n\n/**\n * An implementation of `lambda.InlineCode` using the esbuild Transform API. Inline function code is limited to 4 KiB after transformation.\n *\n * @stability experimental\n */\nexport class InlineTsxCode extends BaseInlineCode {\n  public constructor(\n    /**\n     * The inline code to be transformed.\n     *\n     * @stability experimental\n     */\n    code: string,\n    /**\n     * Support for `TransformOptions` is deprecated. Please provide `TransformerProps`!\n     *\n     * Props to change the behaviour of the transformer.\n     *\n     * Default values for `transformOptions`:\n     * - `loader='tsx'`\n     *\n     * @see https://esbuild.github.io/api/#transform-api\n     * @stability experimental\n     */\n    props?: TransformerProps | TransformOptions,\n  ) {\n    super(code, transformerProps('tsx', props));\n  }\n}\n"]}
package/lib/source.js CHANGED
@@ -55,7 +55,7 @@ class JavaScriptSource extends Source {
55
55
  }
56
56
  exports.JavaScriptSource = JavaScriptSource;
57
57
  _a = JSII_RTTI_SYMBOL_1;
58
- JavaScriptSource[_a] = { fqn: "@mrgrain/cdk-esbuild.JavaScriptSource", version: "4.0.0-alpha.2" };
58
+ JavaScriptSource[_a] = { fqn: "@mrgrain/cdk-esbuild.JavaScriptSource", version: "4.0.0-alpha.5" };
59
59
  class TypeScriptSource extends Source {
60
60
  constructor(entryPoints, props = {}) {
61
61
  super(entryPoints, props);
@@ -64,5 +64,5 @@ class TypeScriptSource extends Source {
64
64
  }
65
65
  exports.TypeScriptSource = TypeScriptSource;
66
66
  _b = JSII_RTTI_SYMBOL_1;
67
- TypeScriptSource[_b] = { fqn: "@mrgrain/cdk-esbuild.TypeScriptSource", version: "4.0.0-alpha.2" };
67
+ TypeScriptSource[_b] = { fqn: "@mrgrain/cdk-esbuild.TypeScriptSource", version: "4.0.0-alpha.5" };
68
68
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic291cmNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3NvdXJjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUFvQztBQU9wQyxtQ0FBdUY7QUFJeEIsQ0FBQztBQUNELENBQUM7QUFFaEUsTUFBZSxNQUFNO0lBY25COzs7O09BSUc7SUFDSCxZQUFZLFdBQXdCLEVBQUUsS0FBWTtRQUNoRCxNQUFNLGNBQWMsR0FBMEI7WUFDNUMsUUFBUSxFQUFFLFNBQVM7U0FDcEIsQ0FBQztRQUVGLElBQUksQ0FBQyxLQUFLLEdBQUc7WUFDWCxXQUFXO1lBQ1gsR0FBRyxLQUFLO1lBQ1IsWUFBWSxFQUFFO2dCQUNaLEdBQUcsY0FBYztnQkFDakIsR0FBRyxLQUFLLENBQUMsWUFBWTthQUN0QjtTQUNGLENBQUM7SUFDSixDQUFDO0lBRUQsSUFBSSxDQUFDLEtBQWdCLEVBQUUsT0FBaUM7UUFDdEQscUZBQXFGO1FBQ3JGLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFO1lBQ2YsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLElBQUksQ0FBQyxVQUFVLENBQzlCLEtBQUssRUFDTCxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksRUFDckIsSUFBSSxDQUFDLEtBQUssQ0FDWCxDQUFDO1NBQ0g7YUFBTSxJQUFJLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUNuRCxNQUFNLElBQUksS0FBSyxDQUNiLG1EQUNFLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxTQUN2QixLQUFLLEdBQUcsOENBQThDLENBQ3ZELENBQUM7U0FDSDtRQUVELElBQUksQ0FBQyxPQUFPLEVBQUU7WUFDWixNQUFNLElBQUksS0FBSyxDQUNiLFlBQVksSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLDRCQUE0QixDQUM5RCxDQUFDO1NBQ0g7UUFFRCx3RUFBd0U7UUFDeEUsc0VBQXNFO1FBQ3RFLDZEQUE2RDtRQUM3RCxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxTQUFTLENBQUMsT0FBTyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBRWpELE9BQU87WUFDTCxNQUFNLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNO1lBQ3pCLFlBQVksRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVc7U0FDckMsQ0FBQztJQUNKLENBQUM7Q0FDRjtBQUVELE1BQWEsZ0JBQWlCLFNBQVEsTUFHckM7SUFHQyxZQUNFLFdBQXdCLEVBQ3hCLFFBQStCLEVBQUU7UUFFakMsS0FBSyxDQUFDLFdBQVcsRUFBRSxLQUFLLENBQUMsQ0FBQztRQU41QixlQUFVLEdBQUcsdUJBQWUsQ0FBQztJQU83QixDQUFDOztBQVhILDRDQVlDOzs7QUFFRCxNQUFhLGdCQUFpQixTQUFRLE1BR3JDO0lBR0MsWUFDRSxXQUF3QixFQUN4QixRQUErQixFQUFFO1FBRWpDLEtBQUssQ0FBQyxXQUFXLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFONUIsZUFBVSxHQUFHLHVCQUFlLENBQUM7SUFPN0IsQ0FBQzs7QUFYSCw0Q0FZQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFN0YWNrIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHtcbiAgRGVwbG95bWVudFNvdXJjZUNvbnRleHQsXG4gIElTb3VyY2UsXG4gIFNvdXJjZUNvbmZpZyxcbn0gZnJvbSAnYXdzLWNkay1saWIvYXdzLXMzLWRlcGxveW1lbnQnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5pbXBvcnQgeyBBc3NldEJhc2VQcm9wcywgQXNzZXRQcm9wcywgSmF2YVNjcmlwdEFzc2V0LCBUeXBlU2NyaXB0QXNzZXQgfSBmcm9tICcuL2Fzc2V0JztcbmltcG9ydCB7IEVudHJ5UG9pbnRzIH0gZnJvbSAnLi9idW5kbGVyJztcbmltcG9ydCB7IEJ1aWxkT3B0aW9ucyB9IGZyb20gJy4vZXNidWlsZC10eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgSmF2YVNjcmlwdFNvdXJjZVByb3BzIGV4dGVuZHMgQXNzZXRCYXNlUHJvcHN7fTtcbmV4cG9ydCBpbnRlcmZhY2UgVHlwZVNjcmlwdFNvdXJjZVByb3BzIGV4dGVuZHMgQXNzZXRCYXNlUHJvcHN7fTtcblxuYWJzdHJhY3QgY2xhc3MgU291cmNlPFxuICBQcm9wcyBleHRlbmRzIEphdmFTY3JpcHRTb3VyY2VQcm9wcyB8IFR5cGVTY3JpcHRTb3VyY2VQcm9wcyxcbiAgQXNzZXQgZXh0ZW5kcyBKYXZhU2NyaXB0QXNzZXQgfCBUeXBlU2NyaXB0QXNzZXQsXG4+IGltcGxlbWVudHMgSVNvdXJjZSB7XG4gIHByb3RlY3RlZCByZWFkb25seSBhYnN0cmFjdCBhc3NldENsYXNzOiBuZXcgKFxuICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogQXNzZXRQcm9wcyxcbiAgKSA9PiBBc3NldDtcblxuICBwcm90ZWN0ZWQgcHJvcHM6IEFzc2V0UHJvcHM7XG5cbiAgcHJvdGVjdGVkIGFzc2V0ITogQXNzZXQ7XG5cbiAgLyoqXG4gICAqXG4gICAqIEBwYXJhbSBlbnRyeVBvaW50cyAtIFJlbGF0aXZlIHBhdGggdG8gdGhlIHNvdXJjZSBjb2RlLiBVc2UgYHByb3BzLmJ1aWxkT3B0aW9ucy5hYnNXb3JraW5nRGlyYCBpZiBhbiBhYnNvbHV0ZSBwYXRoIGlzIHJlcXVpcmVkLlxuICAgKiBAcGFyYW0gcHJvcHMgLSBTb3VyY2UgcHJvcGVydGllcy5cbiAgICovXG4gIGNvbnN0cnVjdG9yKGVudHJ5UG9pbnRzOiBFbnRyeVBvaW50cywgcHJvcHM6IFByb3BzKSB7XG4gICAgY29uc3QgZGVmYXVsdE9wdGlvbnM6IFBhcnRpYWw8QnVpbGRPcHRpb25zPiA9IHtcbiAgICAgIHBsYXRmb3JtOiAnYnJvd3NlcicsXG4gICAgfTtcblxuICAgIHRoaXMucHJvcHMgPSB7XG4gICAgICBlbnRyeVBvaW50cyxcbiAgICAgIC4uLnByb3BzLFxuICAgICAgYnVpbGRPcHRpb25zOiB7XG4gICAgICAgIC4uLmRlZmF1bHRPcHRpb25zLFxuICAgICAgICAuLi5wcm9wcy5idWlsZE9wdGlvbnMsXG4gICAgICB9LFxuICAgIH07XG4gIH1cblxuICBiaW5kKHNjb3BlOiBDb25zdHJ1Y3QsIGNvbnRleHQ/OiBEZXBsb3ltZW50U291cmNlQ29udGV4dCk6IFNvdXJjZUNvbmZpZyB7XG4gICAgLy8gSWYgdGhlIHNhbWUgQXNzZXRDb2RlIGlzIHVzZWQgbXVsdGlwbGUgdGltZXMsIHJldGFpbiBvbmx5IHRoZSBmaXJzdCBpbnN0YW50aWF0aW9uLlxuICAgIGlmICghdGhpcy5hc3NldCkge1xuICAgICAgdGhpcy5hc3NldCA9IG5ldyB0aGlzLmFzc2V0Q2xhc3MoXG4gICAgICAgIHNjb3BlLFxuICAgICAgICB0aGlzLmNvbnN0cnVjdG9yLm5hbWUsXG4gICAgICAgIHRoaXMucHJvcHMsXG4gICAgICApO1xuICAgIH0gZWxzZSBpZiAoU3RhY2sub2YodGhpcy5hc3NldCkgIT09IFN0YWNrLm9mKHNjb3BlKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgQXNzZXQgaXMgYWxyZWFkeSBhc3NvY2lhdGVkIHdpdGggYW5vdGhlciBzdGFjayAnJHtcbiAgICAgICAgICBTdGFjay5vZih0aGlzLmFzc2V0KS5zdGFja05hbWVcbiAgICAgICAgfScuIGAgKyAnQ3JlYXRlIGEgbmV3IEFzc2V0IGluc3RhbmNlIGZvciBldmVyeSBzdGFjay4nLFxuICAgICAgKTtcbiAgICB9XG5cbiAgICBpZiAoIWNvbnRleHQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgYFRvIHVzZSBhICR7dGhpcy5jb25zdHJ1Y3Rvci5uYW1lfSwgY29udGV4dCBtdXN0IGJlIHByb3ZpZGVkYCxcbiAgICAgICk7XG4gICAgfVxuXG4gICAgLy8gd2UgZ2l2ZSBwZXJtaXNzaW9ucyBvbiBhbGwgZmlsZXMgaW4gdGhlIGJ1Y2tldCBzaW5jZSB3ZSBkb24ndCB3YW50IHRvXG4gICAgLy8gYWNjaWRlbnRhbGx5IHJldm9rZSBwZXJtaXNzaW9uIG9uIG9sZCB2ZXJzaW9ucyB3aGVuIGRlcGxveWluZyBhIG5ld1xuICAgIC8vIHZlcnNpb24gKGZvciBleGFtcGxlLCB3aGVuIHVzaW5nIExhbWJkYSB0cmFmZmljIHNoaWZ0aW5nKS5cbiAgICB0aGlzLmFzc2V0LmJ1Y2tldC5ncmFudFJlYWQoY29udGV4dC5oYW5kbGVyUm9sZSk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgYnVja2V0OiB0aGlzLmFzc2V0LmJ1Y2tldCxcbiAgICAgIHppcE9iamVjdEtleTogdGhpcy5hc3NldC5zM09iamVjdEtleSxcbiAgICB9O1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBKYXZhU2NyaXB0U291cmNlIGV4dGVuZHMgU291cmNlPFxuSmF2YVNjcmlwdFNvdXJjZVByb3BzLFxuSmF2YVNjcmlwdEFzc2V0XG4+IHtcbiAgYXNzZXRDbGFzcyA9IEphdmFTY3JpcHRBc3NldDtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBlbnRyeVBvaW50czogRW50cnlQb2ludHMsXG4gICAgcHJvcHM6IEphdmFTY3JpcHRTb3VyY2VQcm9wcyA9IHt9LFxuICApIHtcbiAgICBzdXBlcihlbnRyeVBvaW50cywgcHJvcHMpO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBUeXBlU2NyaXB0U291cmNlIGV4dGVuZHMgU291cmNlPFxuVHlwZVNjcmlwdFNvdXJjZVByb3BzLFxuVHlwZVNjcmlwdEFzc2V0XG4+IHtcbiAgYXNzZXRDbGFzcyA9IFR5cGVTY3JpcHRBc3NldDtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBlbnRyeVBvaW50czogRW50cnlQb2ludHMsXG4gICAgcHJvcHM6IFR5cGVTY3JpcHRTb3VyY2VQcm9wcyA9IHt9LFxuICApIHtcbiAgICBzdXBlcihlbnRyeVBvaW50cywgcHJvcHMpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ .nyc_output/
2
+ coverage/
@@ -0,0 +1,15 @@
1
+ The ISC License
2
+
3
+ Copyright (c) Isaac Z. Schlueter and Contributors
4
+
5
+ Permission to use, copy, modify, and/or distribute this software for any
6
+ purpose with or without fee is hereby granted, provided that the above
7
+ copyright notice and this permission notice appear in all copies.
8
+
9
+ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
15
+ IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
@@ -0,0 +1,51 @@
1
+ # isexe
2
+
3
+ Minimal module to check if a file is executable, and a normal file.
4
+
5
+ Uses `fs.stat` and tests against the `PATHEXT` environment variable on
6
+ Windows.
7
+
8
+ ## USAGE
9
+
10
+ ```javascript
11
+ var isexe = require('isexe')
12
+ isexe('some-file-name', function (err, isExe) {
13
+ if (err) {
14
+ console.error('probably file does not exist or something', err)
15
+ } else if (isExe) {
16
+ console.error('this thing can be run')
17
+ } else {
18
+ console.error('cannot be run')
19
+ }
20
+ })
21
+
22
+ // same thing but synchronous, throws errors
23
+ var isExe = isexe.sync('some-file-name')
24
+
25
+ // treat errors as just "not executable"
26
+ isexe('maybe-missing-file', { ignoreErrors: true }, callback)
27
+ var isExe = isexe.sync('maybe-missing-file', { ignoreErrors: true })
28
+ ```
29
+
30
+ ## API
31
+
32
+ ### `isexe(path, [options], [callback])`
33
+
34
+ Check if the path is executable. If no callback provided, and a
35
+ global `Promise` object is available, then a Promise will be returned.
36
+
37
+ Will raise whatever errors may be raised by `fs.stat`, unless
38
+ `options.ignoreErrors` is set to true.
39
+
40
+ ### `isexe.sync(path, [options])`
41
+
42
+ Same as `isexe` but returns the value and throws any errors raised.
43
+
44
+ ### Options
45
+
46
+ * `ignoreErrors` Treat all errors as "no, this is not executable", but
47
+ don't raise them.
48
+ * `uid` Number to use as the user id
49
+ * `gid` Number to use as the group id
50
+ * `pathExt` List of path extensions to use instead of `PATHEXT`
51
+ environment variable on Windows.
@@ -0,0 +1,57 @@
1
+ var fs = require('fs')
2
+ var core
3
+ if (process.platform === 'win32' || global.TESTING_WINDOWS) {
4
+ core = require('./windows.js')
5
+ } else {
6
+ core = require('./mode.js')
7
+ }
8
+
9
+ module.exports = isexe
10
+ isexe.sync = sync
11
+
12
+ function isexe (path, options, cb) {
13
+ if (typeof options === 'function') {
14
+ cb = options
15
+ options = {}
16
+ }
17
+
18
+ if (!cb) {
19
+ if (typeof Promise !== 'function') {
20
+ throw new TypeError('callback not provided')
21
+ }
22
+
23
+ return new Promise(function (resolve, reject) {
24
+ isexe(path, options || {}, function (er, is) {
25
+ if (er) {
26
+ reject(er)
27
+ } else {
28
+ resolve(is)
29
+ }
30
+ })
31
+ })
32
+ }
33
+
34
+ core(path, options || {}, function (er, is) {
35
+ // ignore EACCES because that just means we aren't allowed to run it
36
+ if (er) {
37
+ if (er.code === 'EACCES' || options && options.ignoreErrors) {
38
+ er = null
39
+ is = false
40
+ }
41
+ }
42
+ cb(er, is)
43
+ })
44
+ }
45
+
46
+ function sync (path, options) {
47
+ // my kingdom for a filtered catch
48
+ try {
49
+ return core.sync(path, options || {})
50
+ } catch (er) {
51
+ if (options && options.ignoreErrors || er.code === 'EACCES') {
52
+ return false
53
+ } else {
54
+ throw er
55
+ }
56
+ }
57
+ }
@@ -0,0 +1,41 @@
1
+ module.exports = isexe
2
+ isexe.sync = sync
3
+
4
+ var fs = require('fs')
5
+
6
+ function isexe (path, options, cb) {
7
+ fs.stat(path, function (er, stat) {
8
+ cb(er, er ? false : checkStat(stat, options))
9
+ })
10
+ }
11
+
12
+ function sync (path, options) {
13
+ return checkStat(fs.statSync(path), options)
14
+ }
15
+
16
+ function checkStat (stat, options) {
17
+ return stat.isFile() && checkMode(stat, options)
18
+ }
19
+
20
+ function checkMode (stat, options) {
21
+ var mod = stat.mode
22
+ var uid = stat.uid
23
+ var gid = stat.gid
24
+
25
+ var myUid = options.uid !== undefined ?
26
+ options.uid : process.getuid && process.getuid()
27
+ var myGid = options.gid !== undefined ?
28
+ options.gid : process.getgid && process.getgid()
29
+
30
+ var u = parseInt('100', 8)
31
+ var g = parseInt('010', 8)
32
+ var o = parseInt('001', 8)
33
+ var ug = u | g
34
+
35
+ var ret = (mod & o) ||
36
+ (mod & g) && gid === myGid ||
37
+ (mod & u) && uid === myUid ||
38
+ (mod & ug) && myUid === 0
39
+
40
+ return ret
41
+ }
@@ -0,0 +1,35 @@
1
+ {
2
+ "name": "isexe",
3
+ "version": "2.0.0",
4
+ "description": "Minimal module to check if a file is executable.",
5
+ "main": "index.js",
6
+ "directories": {
7
+ "test": "test"
8
+ },
9
+ "devDependencies": {
10
+ "mkdirp": "^0.5.1",
11
+ "rimraf": "^2.5.0",
12
+ "tap": "^10.3.0"
13
+ },
14
+ "scripts": {
15
+ "test": "tap test/*.js --100",
16
+ "preversion": "npm test",
17
+ "postversion": "npm publish",
18
+ "postpublish": "git push origin --all; git push origin --tags"
19
+ },
20
+ "author": "Isaac Z. Schlueter <i@izs.me> (http://blog.izs.me/)",
21
+ "license": "ISC",
22
+ "repository": {
23
+ "type": "git",
24
+ "url": "git+https://github.com/isaacs/isexe.git"
25
+ },
26
+ "keywords": [],
27
+ "bugs": {
28
+ "url": "https://github.com/isaacs/isexe/issues"
29
+ },
30
+ "homepage": "https://github.com/isaacs/isexe#readme"
31
+
32
+ ,"_resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz"
33
+ ,"_integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw=="
34
+ ,"_from": "isexe@2.0.0"
35
+ }
@@ -0,0 +1,221 @@
1
+ var t = require('tap')
2
+ var fs = require('fs')
3
+ var path = require('path')
4
+ var fixture = path.resolve(__dirname, 'fixtures')
5
+ var meow = fixture + '/meow.cat'
6
+ var mine = fixture + '/mine.cat'
7
+ var ours = fixture + '/ours.cat'
8
+ var fail = fixture + '/fail.false'
9
+ var noent = fixture + '/enoent.exe'
10
+ var mkdirp = require('mkdirp')
11
+ var rimraf = require('rimraf')
12
+
13
+ var isWindows = process.platform === 'win32'
14
+ var hasAccess = typeof fs.access === 'function'
15
+ var winSkip = isWindows && 'windows'
16
+ var accessSkip = !hasAccess && 'no fs.access function'
17
+ var hasPromise = typeof Promise === 'function'
18
+ var promiseSkip = !hasPromise && 'no global Promise'
19
+
20
+ function reset () {
21
+ delete require.cache[require.resolve('../')]
22
+ return require('../')
23
+ }
24
+
25
+ t.test('setup fixtures', function (t) {
26
+ rimraf.sync(fixture)
27
+ mkdirp.sync(fixture)
28
+ fs.writeFileSync(meow, '#!/usr/bin/env cat\nmeow\n')
29
+ fs.chmodSync(meow, parseInt('0755', 8))
30
+ fs.writeFileSync(fail, '#!/usr/bin/env false\n')
31
+ fs.chmodSync(fail, parseInt('0644', 8))
32
+ fs.writeFileSync(mine, '#!/usr/bin/env cat\nmine\n')
33
+ fs.chmodSync(mine, parseInt('0744', 8))
34
+ fs.writeFileSync(ours, '#!/usr/bin/env cat\nours\n')
35
+ fs.chmodSync(ours, parseInt('0754', 8))
36
+ t.end()
37
+ })
38
+
39
+ t.test('promise', { skip: promiseSkip }, function (t) {
40
+ var isexe = reset()
41
+ t.test('meow async', function (t) {
42
+ isexe(meow).then(function (is) {
43
+ t.ok(is)
44
+ t.end()
45
+ })
46
+ })
47
+ t.test('fail async', function (t) {
48
+ isexe(fail).then(function (is) {
49
+ t.notOk(is)
50
+ t.end()
51
+ })
52
+ })
53
+ t.test('noent async', function (t) {
54
+ isexe(noent).catch(function (er) {
55
+ t.ok(er)
56
+ t.end()
57
+ })
58
+ })
59
+ t.test('noent ignore async', function (t) {
60
+ isexe(noent, { ignoreErrors: true }).then(function (is) {
61
+ t.notOk(is)
62
+ t.end()
63
+ })
64
+ })
65
+ t.end()
66
+ })
67
+
68
+ t.test('no promise', function (t) {
69
+ global.Promise = null
70
+ var isexe = reset()
71
+ t.throws('try to meow a promise', function () {
72
+ isexe(meow)
73
+ })
74
+ t.end()
75
+ })
76
+
77
+ t.test('access', { skip: accessSkip || winSkip }, function (t) {
78
+ runTest(t)
79
+ })
80
+
81
+ t.test('mode', { skip: winSkip }, function (t) {
82
+ delete fs.access
83
+ delete fs.accessSync
84
+ var isexe = reset()
85
+ t.ok(isexe.sync(ours, { uid: 0, gid: 0 }))
86
+ t.ok(isexe.sync(mine, { uid: 0, gid: 0 }))
87
+ runTest(t)
88
+ })
89
+
90
+ t.test('windows', function (t) {
91
+ global.TESTING_WINDOWS = true
92
+ var pathExt = '.EXE;.CAT;.CMD;.COM'
93
+ t.test('pathExt option', function (t) {
94
+ runTest(t, { pathExt: '.EXE;.CAT;.CMD;.COM' })
95
+ })
96
+ t.test('pathExt env', function (t) {
97
+ process.env.PATHEXT = pathExt
98
+ runTest(t)
99
+ })
100
+ t.test('no pathExt', function (t) {
101
+ // with a pathExt of '', any filename is fine.
102
+ // so the "fail" one would still pass.
103
+ runTest(t, { pathExt: '', skipFail: true })
104
+ })
105
+ t.test('pathext with empty entry', function (t) {
106
+ // with a pathExt of '', any filename is fine.
107
+ // so the "fail" one would still pass.
108
+ runTest(t, { pathExt: ';' + pathExt, skipFail: true })
109
+ })
110
+ t.end()
111
+ })
112
+
113
+ t.test('cleanup', function (t) {
114
+ rimraf.sync(fixture)
115
+ t.end()
116
+ })
117
+
118
+ function runTest (t, options) {
119
+ var isexe = reset()
120
+
121
+ var optionsIgnore = Object.create(options || {})
122
+ optionsIgnore.ignoreErrors = true
123
+
124
+ if (!options || !options.skipFail) {
125
+ t.notOk(isexe.sync(fail, options))
126
+ }
127
+ t.notOk(isexe.sync(noent, optionsIgnore))
128
+ if (!options) {
129
+ t.ok(isexe.sync(meow))
130
+ } else {
131
+ t.ok(isexe.sync(meow, options))
132
+ }
133
+
134
+ t.ok(isexe.sync(mine, options))
135
+ t.ok(isexe.sync(ours, options))
136
+ t.throws(function () {
137
+ isexe.sync(noent, options)
138
+ })
139
+
140
+ t.test('meow async', function (t) {
141
+ if (!options) {
142
+ isexe(meow, function (er, is) {
143
+ if (er) {
144
+ throw er
145
+ }
146
+ t.ok(is)
147
+ t.end()
148
+ })
149
+ } else {
150
+ isexe(meow, options, function (er, is) {
151
+ if (er) {
152
+ throw er
153
+ }
154
+ t.ok(is)
155
+ t.end()
156
+ })
157
+ }
158
+ })
159
+
160
+ t.test('mine async', function (t) {
161
+ isexe(mine, options, function (er, is) {
162
+ if (er) {
163
+ throw er
164
+ }
165
+ t.ok(is)
166
+ t.end()
167
+ })
168
+ })
169
+
170
+ t.test('ours async', function (t) {
171
+ isexe(ours, options, function (er, is) {
172
+ if (er) {
173
+ throw er
174
+ }
175
+ t.ok(is)
176
+ t.end()
177
+ })
178
+ })
179
+
180
+ if (!options || !options.skipFail) {
181
+ t.test('fail async', function (t) {
182
+ isexe(fail, options, function (er, is) {
183
+ if (er) {
184
+ throw er
185
+ }
186
+ t.notOk(is)
187
+ t.end()
188
+ })
189
+ })
190
+ }
191
+
192
+ t.test('noent async', function (t) {
193
+ isexe(noent, options, function (er, is) {
194
+ t.ok(er)
195
+ t.notOk(is)
196
+ t.end()
197
+ })
198
+ })
199
+
200
+ t.test('noent ignore async', function (t) {
201
+ isexe(noent, optionsIgnore, function (er, is) {
202
+ if (er) {
203
+ throw er
204
+ }
205
+ t.notOk(is)
206
+ t.end()
207
+ })
208
+ })
209
+
210
+ t.test('directory is not executable', function (t) {
211
+ isexe(__dirname, options, function (er, is) {
212
+ if (er) {
213
+ throw er
214
+ }
215
+ t.notOk(is)
216
+ t.end()
217
+ })
218
+ })
219
+
220
+ t.end()
221
+ }
@@ -0,0 +1,42 @@
1
+ module.exports = isexe
2
+ isexe.sync = sync
3
+
4
+ var fs = require('fs')
5
+
6
+ function checkPathExt (path, options) {
7
+ var pathext = options.pathExt !== undefined ?
8
+ options.pathExt : process.env.PATHEXT
9
+
10
+ if (!pathext) {
11
+ return true
12
+ }
13
+
14
+ pathext = pathext.split(';')
15
+ if (pathext.indexOf('') !== -1) {
16
+ return true
17
+ }
18
+ for (var i = 0; i < pathext.length; i++) {
19
+ var p = pathext[i].toLowerCase()
20
+ if (p && path.substr(-p.length).toLowerCase() === p) {
21
+ return true
22
+ }
23
+ }
24
+ return false
25
+ }
26
+
27
+ function checkStat (stat, path, options) {
28
+ if (!stat.isSymbolicLink() && !stat.isFile()) {
29
+ return false
30
+ }
31
+ return checkPathExt(path, options)
32
+ }
33
+
34
+ function isexe (path, options, cb) {
35
+ fs.stat(path, function (er, stat) {
36
+ cb(er, er ? false : checkStat(stat, path, options))
37
+ })
38
+ }
39
+
40
+ function sync (path, options) {
41
+ return checkStat(fs.statSync(path), path, options)
42
+ }