appium 2.0.0-beta.21 → 2.0.0-beta.25

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.
Files changed (93) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +1 -2
  3. package/build/check-npm-pack-files.js +23 -0
  4. package/build/commands-yml/parse.js +319 -0
  5. package/build/commands-yml/validator.js +130 -0
  6. package/build/index.js +19 -0
  7. package/build/lib/appium.js +22 -7
  8. package/build/lib/cli/args.js +13 -15
  9. package/build/lib/cli/npm.js +27 -16
  10. package/build/lib/cli/parser.js +7 -3
  11. package/build/lib/config-file.js +4 -7
  12. package/build/lib/config.js +57 -48
  13. package/build/lib/extension-config.js +1 -1
  14. package/build/lib/main.js +28 -28
  15. package/build/lib/plugin-config.js +2 -2
  16. package/build/lib/plugins.js +4 -2
  17. package/build/lib/schema/appium-config-schema.js +3 -2
  18. package/build/lib/schema/arg-spec.js +5 -3
  19. package/build/lib/schema/cli-args.js +25 -16
  20. package/build/lib/schema/keywords.js +14 -4
  21. package/build/lib/schema/schema.js +86 -9
  22. package/build/lib/utils.js +16 -36
  23. package/build/postinstall.js +90 -0
  24. package/build/test/cli/cli-e2e-specs.js +221 -0
  25. package/build/test/cli/cli-helpers.js +86 -0
  26. package/build/test/cli/cli-specs.js +71 -0
  27. package/build/test/cli/fixtures/test-driver/package.json +27 -0
  28. package/build/test/cli/schema-args-specs.js +48 -0
  29. package/build/test/cli/schema-e2e-specs.js +47 -0
  30. package/build/test/config-e2e-specs.js +112 -0
  31. package/build/test/config-file-e2e-specs.js +191 -0
  32. package/build/test/config-file-specs.js +281 -0
  33. package/build/test/config-specs.js +258 -0
  34. package/build/test/driver-e2e-specs.js +435 -0
  35. package/build/test/driver-specs.js +386 -0
  36. package/build/test/ext-config-io-specs.js +181 -0
  37. package/build/test/extension-config-specs.js +365 -0
  38. package/build/test/fixtures/allow-feat.txt +5 -0
  39. package/build/test/fixtures/caps.json +3 -0
  40. package/build/test/fixtures/config/allow-insecure.txt +3 -0
  41. package/build/test/fixtures/config/appium.config.bad-nodeconfig.json +5 -0
  42. package/build/test/fixtures/config/appium.config.bad.json +32 -0
  43. package/build/test/fixtures/config/appium.config.ext-good.json +9 -0
  44. package/build/test/fixtures/config/appium.config.ext-unknown-props.json +10 -0
  45. package/build/test/fixtures/config/appium.config.good.js +40 -0
  46. package/build/test/fixtures/config/appium.config.good.json +33 -0
  47. package/build/test/fixtures/config/appium.config.good.yaml +30 -0
  48. package/build/test/fixtures/config/appium.config.invalid.json +31 -0
  49. package/build/test/fixtures/config/appium.config.security-array.json +5 -0
  50. package/build/test/fixtures/config/appium.config.security-delimited.json +5 -0
  51. package/build/test/fixtures/config/appium.config.security-path.json +5 -0
  52. package/build/test/fixtures/config/driver-fake.config.json +8 -0
  53. package/build/test/fixtures/config/nodeconfig.json +3 -0
  54. package/build/test/fixtures/config/plugin-fake.config.json +0 -0
  55. package/build/test/fixtures/default-args.js +35 -0
  56. package/build/test/fixtures/deny-feat.txt +5 -0
  57. package/build/test/fixtures/driver.schema.js +20 -0
  58. package/build/test/fixtures/extensions.yaml +27 -0
  59. package/build/test/fixtures/flattened-schema.js +532 -0
  60. package/build/test/fixtures/plugin.schema.js +20 -0
  61. package/build/test/fixtures/schema-with-extensions.js +28 -0
  62. package/build/test/grid-register-specs.js +74 -0
  63. package/build/test/helpers.js +75 -0
  64. package/build/test/logger-specs.js +76 -0
  65. package/build/test/npm-specs.js +20 -0
  66. package/build/test/parser-specs.js +319 -0
  67. package/build/test/plugin-e2e-specs.js +316 -0
  68. package/build/test/schema/arg-spec-specs.js +70 -0
  69. package/build/test/schema/cli-args-specs.js +408 -0
  70. package/build/test/schema/schema-specs.js +407 -0
  71. package/build/test/utils-specs.js +288 -0
  72. package/index.js +11 -0
  73. package/lib/appium-config.schema.json +2 -1
  74. package/lib/appium.js +51 -8
  75. package/lib/cli/args.js +17 -14
  76. package/lib/cli/npm.js +68 -6
  77. package/lib/cli/parser.js +5 -2
  78. package/lib/config-file.js +9 -12
  79. package/lib/config.js +104 -56
  80. package/lib/extension-config.js +1 -1
  81. package/lib/main.js +94 -40
  82. package/lib/plugin-config.js +1 -1
  83. package/lib/plugins.js +2 -0
  84. package/lib/schema/appium-config-schema.js +1 -0
  85. package/lib/schema/arg-spec.js +13 -3
  86. package/lib/schema/cli-args.js +22 -34
  87. package/lib/schema/keywords.js +20 -4
  88. package/lib/schema/schema.js +150 -24
  89. package/lib/utils.js +28 -29
  90. package/package.json +9 -14
  91. package/types/types.d.ts +5 -0
  92. package/build/lib/cli/argparse-actions.js +0 -104
  93. package/lib/cli/argparse-actions.js +0 -77
@@ -0,0 +1,191 @@
1
+ "use strict";
2
+
3
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
+
5
+ require("source-map-support/register");
6
+
7
+ var _configFile = require("../lib/config-file");
8
+
9
+ var _schema = require("../lib/schema/schema");
10
+
11
+ var _driverSchema = _interopRequireDefault(require("./fixtures/driver.schema.js"));
12
+
13
+ var _helpers = require("./helpers");
14
+
15
+ describe('config file behavior', function () {
16
+ const GOOD_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.good.json');
17
+ const BAD_NODECONFIG_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.bad-nodeconfig.json');
18
+ const BAD_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.bad.json');
19
+ const INVALID_JSON_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.invalid.json');
20
+ const SECURITY_ARRAY_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.security-array.json');
21
+ const SECURITY_DELIMITED_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.security-delimited.json');
22
+ const SECURITY_PATH_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.security-path.json');
23
+ const UNKNOWN_PROPS_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.ext-unknown-props.json');
24
+ const EXT_PROPS_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.ext-good.json');
25
+ beforeEach(function () {
26
+ (0, _schema.finalizeSchema)();
27
+ });
28
+ afterEach(function () {
29
+ (0, _schema.resetSchema)();
30
+ });
31
+ describe('when provided a path to a config file', function () {
32
+ describe('when the config file is valid per the schema', function () {
33
+ it('should return a valid config object', async function () {
34
+ const result = await (0, _configFile.readConfigFile)(GOOD_FILEPATH);
35
+ result.should.deep.equal({
36
+ config: require(GOOD_FILEPATH),
37
+ filepath: GOOD_FILEPATH,
38
+ errors: []
39
+ });
40
+ });
41
+ describe('server.nodeconfig behavior', function () {
42
+ describe('when a string', function () {
43
+ it('should return errors', async function () {
44
+ const result = await (0, _configFile.readConfigFile)(BAD_NODECONFIG_FILEPATH);
45
+ result.should.have.nested.property('errors[0].instancePath', '/server/nodeconfig');
46
+ });
47
+ });
48
+ describe('when an object', function () {
49
+ it('should return a valid config object', async function () {
50
+ const result = await (0, _configFile.readConfigFile)(GOOD_FILEPATH);
51
+ result.should.have.property('errors').that.is.empty;
52
+ });
53
+ });
54
+ });
55
+ describe('server.allow-insecure behavior', function () {
56
+ describe('when a string path', function () {
57
+ it('should return errors', async function () {
58
+ const result = await (0, _configFile.readConfigFile)(SECURITY_PATH_FILEPATH);
59
+ result.should.have.nested.property('errors[0].instancePath', '/server/allow-insecure');
60
+ });
61
+ });
62
+ describe('when a comma-delimited string', function () {
63
+ it('should return errors', async function () {
64
+ const result = await (0, _configFile.readConfigFile)(SECURITY_DELIMITED_FILEPATH);
65
+ result.should.have.nested.property('errors[0].instancePath', '/server/allow-insecure');
66
+ });
67
+ });
68
+ describe('when an array', function () {
69
+ it('should return a valid config object', async function () {
70
+ const result = await (0, _configFile.readConfigFile)(SECURITY_ARRAY_FILEPATH);
71
+ result.should.deep.equal({
72
+ config: require(SECURITY_ARRAY_FILEPATH),
73
+ filepath: SECURITY_ARRAY_FILEPATH,
74
+ errors: []
75
+ });
76
+ });
77
+ });
78
+ });
79
+ });
80
+ describe('when the config file is invalid per the schema', function () {
81
+ describe('without extensions', function () {
82
+ it('should return an object containing errors', async function () {
83
+ const result = await (0, _configFile.readConfigFile)(BAD_FILEPATH);
84
+ result.should.have.deep.property('config', (0, _configFile.normalizeConfig)(require(BAD_FILEPATH)));
85
+ result.should.have.property('filepath', BAD_FILEPATH);
86
+ result.should.have.deep.property('errors').that.contains.members([{
87
+ instancePath: '',
88
+ schemaPath: '#/additionalProperties',
89
+ keyword: 'additionalProperties',
90
+ params: {
91
+ additionalProperty: 'appium-home'
92
+ },
93
+ message: 'must NOT have additional properties',
94
+ isIdentifierLocation: true
95
+ }, {
96
+ instancePath: '/server/allow-cors',
97
+ schemaPath: '#/properties/server/properties/allow-cors/type',
98
+ keyword: 'type',
99
+ params: {
100
+ type: 'boolean'
101
+ },
102
+ message: 'must be boolean'
103
+ }, {
104
+ instancePath: '/server/allow-insecure',
105
+ schemaPath: '#/properties/server/properties/allow-insecure/type',
106
+ keyword: 'type',
107
+ params: {
108
+ type: 'array'
109
+ },
110
+ message: 'must be array'
111
+ }, {
112
+ instancePath: '/server/callback-port',
113
+ schemaPath: '#/properties/server/properties/callback-port/maximum',
114
+ keyword: 'maximum',
115
+ params: {
116
+ comparison: '<=',
117
+ limit: 65535
118
+ },
119
+ message: 'must be <= 65535'
120
+ }, {
121
+ instancePath: '/server/log-level',
122
+ schemaPath: '#/properties/server/properties/log-level/enum',
123
+ keyword: 'enum',
124
+ params: {
125
+ allowedValues: ['info', 'info:debug', 'info:info', 'info:warn', 'info:error', 'warn', 'warn:debug', 'warn:info', 'warn:warn', 'warn:error', 'error', 'error:debug', 'error:info', 'error:warn', 'error:error', 'debug', 'debug:debug', 'debug:info', 'debug:warn', 'debug:error']
126
+ },
127
+ message: 'must be equal to one of the allowed values'
128
+ }, {
129
+ instancePath: '/server/log-no-colors',
130
+ schemaPath: '#/properties/server/properties/log-no-colors/type',
131
+ keyword: 'type',
132
+ params: {
133
+ type: 'boolean'
134
+ },
135
+ message: 'must be boolean'
136
+ }, {
137
+ instancePath: '/server/port',
138
+ schemaPath: '#/properties/server/properties/port/type',
139
+ keyword: 'type',
140
+ params: {
141
+ type: 'integer'
142
+ },
143
+ message: 'must be integer'
144
+ }]).and.lengthOf(7);
145
+ result.should.have.property('reason').that.is.a.string;
146
+ });
147
+ });
148
+ describe('with extensions', function () {
149
+ let result;
150
+ beforeEach(function () {
151
+ (0, _schema.resetSchema)();
152
+ (0, _schema.registerSchema)('driver', 'fake', _driverSchema.default);
153
+ (0, _schema.finalizeSchema)();
154
+ });
155
+ describe('when provided a config file with unknown properties', function () {
156
+ beforeEach(async function () {
157
+ result = await (0, _configFile.readConfigFile)(UNKNOWN_PROPS_FILEPATH);
158
+ });
159
+ it('should return an object containing errors', function () {
160
+ result.should.have.deep.property('errors', [{
161
+ instancePath: '/server/driver/fake',
162
+ schemaPath: 'driver-fake.json/additionalProperties',
163
+ keyword: 'additionalProperties',
164
+ params: {
165
+ additionalProperty: 'bubb'
166
+ },
167
+ message: 'must NOT have additional properties',
168
+ isIdentifierLocation: true
169
+ }]);
170
+ });
171
+ });
172
+ describe('when provided a config file with valid properties', function () {
173
+ beforeEach(async function () {
174
+ result = await (0, _configFile.readConfigFile)(EXT_PROPS_FILEPATH);
175
+ });
176
+ it('should return an object containing no errors', function () {
177
+ result.should.have.deep.property('errors', []);
178
+ });
179
+ });
180
+ });
181
+ });
182
+ describe('when the config file is invalid JSON', function () {
183
+ it('should reject with a user-friendly error message', async function () {
184
+ await (0, _configFile.readConfigFile)(INVALID_JSON_FILEPATH).should.be.rejectedWith(new RegExp(`${INVALID_JSON_FILEPATH} is invalid`));
185
+ });
186
+ });
187
+ });
188
+ });require('source-map-support').install();
189
+
190
+
191
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["test/config-file-e2e-specs.js"],"names":["describe","GOOD_FILEPATH","BAD_NODECONFIG_FILEPATH","BAD_FILEPATH","INVALID_JSON_FILEPATH","SECURITY_ARRAY_FILEPATH","SECURITY_DELIMITED_FILEPATH","SECURITY_PATH_FILEPATH","UNKNOWN_PROPS_FILEPATH","EXT_PROPS_FILEPATH","beforeEach","afterEach","it","result","should","deep","equal","config","require","filepath","errors","have","nested","property","that","is","empty","contains","members","instancePath","schemaPath","keyword","params","additionalProperty","message","isIdentifierLocation","type","comparison","limit","allowedValues","and","lengthOf","a","string","extSchema","be","rejectedWith","RegExp"],"mappings":"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","sourcesContent":["// @ts-check\n\nimport { readConfigFile, normalizeConfig } from '../lib/config-file';\nimport { finalizeSchema, registerSchema, resetSchema } from '../lib/schema/schema';\nimport extSchema from './fixtures/driver.schema.js';\nimport { resolveFixture } from './helpers';\n\ndescribe('config file behavior', function () {\n  const GOOD_FILEPATH = resolveFixture('config', 'appium.config.good.json');\n  const BAD_NODECONFIG_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.bad-nodeconfig.json',\n  );\n  const BAD_FILEPATH = resolveFixture('config', 'appium.config.bad.json');\n  const INVALID_JSON_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.invalid.json',\n  );\n  const SECURITY_ARRAY_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.security-array.json',\n  );\n  const SECURITY_DELIMITED_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.security-delimited.json',\n  );\n  const SECURITY_PATH_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.security-path.json',\n  );\n  const UNKNOWN_PROPS_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.ext-unknown-props.json',\n  );\n  const EXT_PROPS_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.ext-good.json',\n  );\n\n  beforeEach(function () {\n    finalizeSchema();\n  });\n\n  afterEach(function () {\n    resetSchema();\n  });\n\n\n  describe('when provided a path to a config file', function () {\n    describe('when the config file is valid per the schema', function () {\n      it('should return a valid config object', async function () {\n        const result = await readConfigFile(GOOD_FILEPATH);\n        result.should.deep.equal({\n          config: require(GOOD_FILEPATH),\n          filepath: GOOD_FILEPATH,\n          errors: [],\n        });\n      });\n\n      describe('server.nodeconfig behavior', function () {\n        describe('when a string', function () {\n          it('should return errors', async function () {\n            const result = await readConfigFile(BAD_NODECONFIG_FILEPATH);\n            result.should.have.nested.property(\n              'errors[0].instancePath',\n              '/server/nodeconfig',\n            );\n          });\n        });\n\n        describe('when an object', function () {\n          it('should return a valid config object', async function () {\n            const result = await readConfigFile(GOOD_FILEPATH);\n            result.should.have.property('errors').that.is.empty;\n          });\n        });\n      });\n\n      describe('server.allow-insecure behavior', function () {\n        describe('when a string path', function () {\n          it('should return errors', async function () {\n            const result = await readConfigFile(SECURITY_PATH_FILEPATH);\n            result.should.have.nested.property(\n              'errors[0].instancePath',\n              '/server/allow-insecure',\n            );\n          });\n        });\n\n        describe('when a comma-delimited string', function () {\n          it('should return errors', async function () {\n            const result = await readConfigFile(SECURITY_DELIMITED_FILEPATH);\n            result.should.have.nested.property(\n              'errors[0].instancePath',\n              '/server/allow-insecure',\n            );\n          });\n        });\n\n        describe('when an array', function () {\n          it('should return a valid config object', async function () {\n            const result = await readConfigFile(SECURITY_ARRAY_FILEPATH);\n            result.should.deep.equal({\n              config: require(SECURITY_ARRAY_FILEPATH),\n              filepath: SECURITY_ARRAY_FILEPATH,\n              errors: [],\n            });\n          });\n        });\n      });\n    });\n\n    describe('when the config file is invalid per the schema', function () {\n      describe('without extensions', function () {\n        it('should return an object containing errors', async function () {\n          const result = await readConfigFile(BAD_FILEPATH);\n          result.should.have.deep.property('config', normalizeConfig(require(BAD_FILEPATH)));\n          result.should.have.property('filepath', BAD_FILEPATH);\n          result.should.have.deep.property('errors').that.contains.members([\n            {\n              instancePath: '',\n              schemaPath: '#/additionalProperties',\n              keyword: 'additionalProperties',\n              params: {\n                additionalProperty: 'appium-home',\n              },\n              message: 'must NOT have additional properties',\n              isIdentifierLocation: true,\n            },\n            {\n              instancePath: '/server/allow-cors',\n              schemaPath: '#/properties/server/properties/allow-cors/type',\n              keyword: 'type',\n              params: {\n                type: 'boolean',\n              },\n              message: 'must be boolean',\n            },\n            {\n              instancePath: '/server/allow-insecure',\n              schemaPath: '#/properties/server/properties/allow-insecure/type',\n              keyword: 'type',\n              params: {\n                type: 'array'\n              },\n              message: 'must be array',\n            },\n            {\n              instancePath: '/server/callback-port',\n              schemaPath:\n                '#/properties/server/properties/callback-port/maximum',\n              keyword: 'maximum',\n              params: {\n                comparison: '<=',\n                limit: 65535,\n              },\n              message: 'must be <= 65535',\n            },\n            {\n              instancePath: '/server/log-level',\n              schemaPath: '#/properties/server/properties/log-level/enum',\n              keyword: 'enum',\n              params: {\n                allowedValues: [\n                  'info',\n                  'info:debug',\n                  'info:info',\n                  'info:warn',\n                  'info:error',\n                  'warn',\n                  'warn:debug',\n                  'warn:info',\n                  'warn:warn',\n                  'warn:error',\n                  'error',\n                  'error:debug',\n                  'error:info',\n                  'error:warn',\n                  'error:error',\n                  'debug',\n                  'debug:debug',\n                  'debug:info',\n                  'debug:warn',\n                  'debug:error',\n                ],\n              },\n              message: 'must be equal to one of the allowed values',\n            },\n            {\n              instancePath: '/server/log-no-colors',\n              schemaPath: '#/properties/server/properties/log-no-colors/type',\n              keyword: 'type',\n              params: {\n                type: 'boolean',\n              },\n              message: 'must be boolean',\n            },\n            {\n              instancePath: '/server/port',\n              schemaPath: '#/properties/server/properties/port/type',\n              keyword: 'type',\n              params: {\n                type: 'integer',\n              },\n              message: 'must be integer',\n            },\n          ]).and.lengthOf(7);\n\n          result.should.have.property('reason').that.is.a.string;\n        });\n      });\n\n      describe('with extensions', function () {\n        /** @type {import('../lib/config-file').ReadConfigFileResult} */\n        let result;\n\n        beforeEach(function () {\n          resetSchema();\n          registerSchema('driver', 'fake', extSchema);\n          finalizeSchema();\n        });\n\n        describe('when provided a config file with unknown properties', function () {\n          beforeEach(async function () {\n            result = await readConfigFile(UNKNOWN_PROPS_FILEPATH);\n          });\n          it('should return an object containing errors', function () {\n            result.should.have.deep.property('errors', [\n              {\n                instancePath: '/server/driver/fake',\n                schemaPath:\n                  'driver-fake.json/additionalProperties',\n                keyword: 'additionalProperties',\n                params: {additionalProperty: 'bubb'},\n                message: 'must NOT have additional properties',\n                isIdentifierLocation: true,\n              },\n            ]);\n          });\n        });\n\n        describe('when provided a config file with valid properties', function () {\n          beforeEach(async function () {\n            result = await readConfigFile(EXT_PROPS_FILEPATH);\n          });\n          it('should return an object containing no errors', function () {\n            result.should.have.deep.property('errors', []);\n          });\n        });\n      });\n    });\n\n    describe('when the config file is invalid JSON', function () {\n      it('should reject with a user-friendly error message', async function () {\n        await readConfigFile(INVALID_JSON_FILEPATH).should.be.rejectedWith(\n          new RegExp(`${INVALID_JSON_FILEPATH} is invalid`),\n        );\n      });\n    });\n  });\n});\n"],"file":"test/config-file-e2e-specs.js","sourceRoot":"../.."}
@@ -0,0 +1,281 @@
1
+ "use strict";
2
+
3
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
+
5
+ require("source-map-support/register");
6
+
7
+ var _fs = _interopRequireDefault(require("fs"));
8
+
9
+ var _sinon = _interopRequireDefault(require("sinon"));
10
+
11
+ var _yaml = _interopRequireDefault(require("yaml"));
12
+
13
+ var schema = _interopRequireWildcard(require("../lib/schema/schema"));
14
+
15
+ var _helpers = require("./helpers");
16
+
17
+ 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); }
18
+
19
+ function _interopRequireWildcard(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 = {}; 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; }
20
+
21
+ const expect = chai.expect;
22
+ describe('config-file', function () {
23
+ const GOOD_YAML_CONFIG_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.good.yaml');
24
+ const GOOD_JSON_CONFIG_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.good.json');
25
+ const GOOD_JS_CONFIG_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.good.js');
26
+
27
+ const GOOD_YAML_CONFIG = _yaml.default.parse(_fs.default.readFileSync(GOOD_YAML_CONFIG_FILEPATH, 'utf8'));
28
+
29
+ const GOOD_JSON_CONFIG = require(GOOD_JSON_CONFIG_FILEPATH);
30
+
31
+ const BAD_JSON_CONFIG_FILEPATH = (0, _helpers.resolveFixture)('config', 'appium.config.bad.json');
32
+
33
+ const BAD_JSON_CONFIG = require(BAD_JSON_CONFIG_FILEPATH);
34
+
35
+ let sandbox;
36
+ let readConfigFile;
37
+ let lc;
38
+ let configFileModule;
39
+ let mocks;
40
+ before(function () {
41
+ schema.finalizeSchema();
42
+ });
43
+ beforeEach(function () {
44
+ sandbox = _sinon.default.createSandbox();
45
+ const load = sandbox.stub().resolves({
46
+ config: GOOD_JSON_CONFIG,
47
+ filepath: GOOD_JSON_CONFIG_FILEPATH
48
+ });
49
+ load.withArgs(GOOD_YAML_CONFIG_FILEPATH).resolves({
50
+ config: GOOD_YAML_CONFIG,
51
+ filepath: GOOD_YAML_CONFIG_FILEPATH
52
+ });
53
+ load.withArgs(BAD_JSON_CONFIG_FILEPATH).resolves({
54
+ config: BAD_JSON_CONFIG,
55
+ filepath: BAD_JSON_CONFIG_FILEPATH
56
+ });
57
+ const search = sandbox.stub().resolves({
58
+ config: GOOD_JSON_CONFIG,
59
+ filepath: GOOD_JSON_CONFIG_FILEPATH
60
+ });
61
+ lc = {
62
+ load,
63
+ search
64
+ };
65
+ mocks = {
66
+ lilconfig: {
67
+ lilconfig: sandbox.stub().returns(lc)
68
+ },
69
+ '@sidvind/better-ajv-errors': sandbox.stub().returns('')
70
+ };
71
+ configFileModule = _helpers.rewiremock.proxy(() => require('../lib/config-file'), mocks);
72
+ readConfigFile = configFileModule.readConfigFile;
73
+ sandbox.spy(schema, 'validate');
74
+ });
75
+ afterEach(function () {
76
+ sandbox.restore();
77
+ });
78
+ describe('readConfigFile()', function () {
79
+ let result;
80
+ it('should support yaml', async function () {
81
+ const {
82
+ config
83
+ } = await readConfigFile(GOOD_YAML_CONFIG_FILEPATH);
84
+ expect(config).to.eql(GOOD_JSON_CONFIG);
85
+ expect(schema.validate).to.have.been.calledOnce;
86
+ });
87
+ it('should support json', async function () {
88
+ const {
89
+ config
90
+ } = await readConfigFile(GOOD_JSON_CONFIG_FILEPATH);
91
+ expect(config).to.eql(GOOD_JSON_CONFIG);
92
+ expect(schema.validate).to.have.been.calledOnce;
93
+ });
94
+ it('should support js', async function () {
95
+ const {
96
+ config
97
+ } = await readConfigFile(GOOD_JS_CONFIG_FILEPATH);
98
+ expect(config).to.eql(GOOD_JSON_CONFIG);
99
+ expect(schema.validate).to.have.been.calledOnce;
100
+ });
101
+ describe('when no filepath provided', function () {
102
+ beforeEach(async function () {
103
+ result = await readConfigFile();
104
+ });
105
+ it('should search for a config file', function () {
106
+ expect(lc.search).to.have.been.calledOnce;
107
+ expect(schema.validate).to.have.been.calledOnce;
108
+ });
109
+ it('should not try to load a config file directly', function () {
110
+ expect(lc.load).to.not.have.been.called;
111
+ });
112
+ describe('when no config file is found', function () {
113
+ beforeEach(async function () {
114
+ lc.search.resolves();
115
+ schema.validate.resetHistory();
116
+ result = await readConfigFile();
117
+ });
118
+ it('should resolve with an empty object', function () {
119
+ expect(result).to.be.an('object').that.is.empty;
120
+ expect(schema.validate).not.to.have.been.called;
121
+ });
122
+ });
123
+ describe('when a config file is found', function () {
124
+ describe('when the config file is empty', function () {
125
+ beforeEach(async function () {
126
+ lc.search.resolves({
127
+ isEmpty: true,
128
+ filepath: '/path/to/file.json',
129
+ config: {}
130
+ });
131
+ result = await readConfigFile();
132
+ });
133
+ it('should resolve with an object with an `isEmpty` property', function () {
134
+ expect(result).to.have.property('isEmpty', true);
135
+ });
136
+ });
137
+ describe('when the config file is not empty', function () {
138
+ it('should validate the config against a schema', function () {
139
+ expect(schema.validate).to.have.been.calledOnceWith(GOOD_JSON_CONFIG);
140
+ });
141
+ describe('when the config file is valid', function () {
142
+ beforeEach(async function () {
143
+ result = await readConfigFile();
144
+ });
145
+ it('should resolve with an object having `config` property and empty array of errors', function () {
146
+ expect(result).to.deep.equal({
147
+ config: GOOD_JSON_CONFIG,
148
+ errors: [],
149
+ filepath: GOOD_JSON_CONFIG_FILEPATH
150
+ });
151
+ });
152
+ });
153
+ describe('when the config file is invalid', function () {
154
+ beforeEach(function () {
155
+ lc.search.resolves({
156
+ config: {
157
+ foo: 'bar'
158
+ },
159
+ filepath: '/path/to/file.json'
160
+ });
161
+ });
162
+ beforeEach(async function () {
163
+ result = await readConfigFile();
164
+ });
165
+ it('should resolve with an object having a nonempty array of errors', function () {
166
+ expect(result).to.have.property('errors').that.is.not.empty;
167
+ });
168
+ });
169
+ });
170
+ });
171
+ });
172
+ describe('when filepath provided', function () {
173
+ beforeEach(async function () {
174
+ result = await readConfigFile('appium.json');
175
+ });
176
+ it('should not attempt to find a config file', function () {
177
+ expect(lc.search).to.not.have.been.called;
178
+ });
179
+ it('should try to load a config file directly', function () {
180
+ expect(lc.load).to.have.been.calledOnce;
181
+ });
182
+ describe('when no config file exists at path', function () {
183
+ beforeEach(function () {
184
+ lc.load.rejects(Object.assign(new Error(), {
185
+ code: 'ENOENT'
186
+ }));
187
+ });
188
+ it('should reject with user-friendly message', async function () {
189
+ await expect(readConfigFile('appium.json')).to.be.rejectedWith(/not found at user-provided path/);
190
+ });
191
+ });
192
+ describe('when the config file is invalid JSON', function () {
193
+ beforeEach(function () {
194
+ lc.load.rejects(new SyntaxError());
195
+ });
196
+ it('should reject with user-friendly message', async function () {
197
+ await expect(readConfigFile('appium.json')).to.be.rejectedWith(/Config file at user-provided path appium.json is invalid/);
198
+ });
199
+ });
200
+ describe('when something else is wrong with loading the config file', function () {
201
+ beforeEach(function () {
202
+ lc.load.rejects(new Error('guru meditation'));
203
+ });
204
+ it('should pass error through', async function () {
205
+ await expect(readConfigFile('appium.json')).to.be.rejectedWith(/guru meditation/);
206
+ });
207
+ });
208
+ describe('when a config file is found', function () {
209
+ describe('when the config file is empty', function () {
210
+ beforeEach(async function () {
211
+ lc.search.resolves({
212
+ isEmpty: true,
213
+ filepath: '/path/to/file.json',
214
+ config: {}
215
+ });
216
+ result = await readConfigFile();
217
+ });
218
+ it('should resolve with an object with an `isEmpty` property', function () {
219
+ expect(result).to.have.property('isEmpty', true);
220
+ });
221
+ });
222
+ describe('when the config file is not empty', function () {
223
+ it('should validate the config against a schema', function () {
224
+ expect(schema.validate).to.have.been.calledOnceWith(GOOD_JSON_CONFIG);
225
+ });
226
+ describe('when the config file is valid', function () {
227
+ beforeEach(async function () {
228
+ result = await readConfigFile();
229
+ });
230
+ it('should resolve with an object having `config` property and empty array of errors', function () {
231
+ expect(result).to.deep.equal({
232
+ errors: [],
233
+ config: GOOD_JSON_CONFIG,
234
+ filepath: GOOD_JSON_CONFIG_FILEPATH
235
+ });
236
+ });
237
+ });
238
+ describe('when the config file is invalid', function () {
239
+ beforeEach(async function () {
240
+ result = await readConfigFile(BAD_JSON_CONFIG_FILEPATH);
241
+ });
242
+ it('should resolve with an object having a nonempty array of errors', function () {
243
+ expect(result).to.have.property('errors').that.is.not.empty;
244
+ });
245
+ });
246
+ });
247
+ });
248
+ });
249
+ });
250
+ describe('formatErrors()', function () {
251
+ describe('when provided `errors` as an empty array', function () {
252
+ it('should throw', function () {
253
+ expect(() => configFileModule.formatErrors([])).to.throw(TypeError, 'Array of errors must be non-empty');
254
+ });
255
+ });
256
+ describe('when provided `errors` as `undefined`', function () {
257
+ it('should throw', function () {
258
+ expect(() => configFileModule.formatErrors()).to.throw(TypeError, 'Array of errors must be non-empty');
259
+ });
260
+ });
261
+ describe('when provided `errors` as a non-empty array', function () {
262
+ it('should return a string', function () {
263
+ expect(configFileModule.formatErrors([{}])).to.be.a('string');
264
+ });
265
+ });
266
+ describe('when `opts.json` is a string', function () {
267
+ it('should call `betterAjvErrors()` with option `json: opts.json`', function () {
268
+ configFileModule.formatErrors([{}], {}, {
269
+ json: '{"foo": "bar"}'
270
+ });
271
+ expect(mocks['@sidvind/better-ajv-errors']).to.have.been.calledWith(schema.getSchema(), {}, [{}], {
272
+ format: 'cli',
273
+ json: '{"foo": "bar"}'
274
+ });
275
+ });
276
+ });
277
+ });
278
+ });require('source-map-support').install();
279
+
280
+
281
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["test/config-file-specs.js"],"names":["expect","chai","describe","GOOD_YAML_CONFIG_FILEPATH","GOOD_JSON_CONFIG_FILEPATH","GOOD_JS_CONFIG_FILEPATH","GOOD_YAML_CONFIG","YAML","parse","fs","readFileSync","GOOD_JSON_CONFIG","require","BAD_JSON_CONFIG_FILEPATH","BAD_JSON_CONFIG","sandbox","readConfigFile","lc","configFileModule","mocks","before","schema","finalizeSchema","beforeEach","sinon","createSandbox","load","stub","resolves","config","filepath","withArgs","search","lilconfig","returns","rewiremock","proxy","spy","afterEach","restore","result","it","to","eql","validate","have","been","calledOnce","not","called","resetHistory","be","an","that","is","empty","isEmpty","property","calledOnceWith","deep","equal","errors","foo","rejects","Object","assign","Error","code","rejectedWith","SyntaxError","formatErrors","throw","TypeError","a","json","calledWith","getSchema","format"],"mappings":"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","sourcesContent":["// @ts-check\n\nimport fs from 'fs';\nimport sinon from 'sinon';\nimport YAML from 'yaml';\nimport * as schema from '../lib/schema/schema';\nimport { resolveFixture, rewiremock } from './helpers';\n\nconst expect = chai.expect;\n\ndescribe('config-file', function () {\n  const GOOD_YAML_CONFIG_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.good.yaml',\n  );\n  const GOOD_JSON_CONFIG_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.good.json',\n  );\n  const GOOD_JS_CONFIG_FILEPATH = resolveFixture('config', 'appium.config.good.js');\n  const GOOD_YAML_CONFIG = YAML.parse(\n    fs.readFileSync(GOOD_YAML_CONFIG_FILEPATH, 'utf8'),\n  );\n  const GOOD_JSON_CONFIG = require(GOOD_JSON_CONFIG_FILEPATH);\n  const BAD_JSON_CONFIG_FILEPATH = resolveFixture(\n    'config',\n    'appium.config.bad.json',\n  );\n  const BAD_JSON_CONFIG = require(BAD_JSON_CONFIG_FILEPATH);\n\n  /**\n   * @type {import('sinon').SinonSandbox}\n   */\n  let sandbox;\n\n  /**\n   * `readConfigFile()` from an isolated `config-file` module\n   * @type {typeof import('../lib/config-file').readConfigFile}\n   */\n  let readConfigFile;\n  /**\n   * Mock instance of `lilconfig` containing stubs for\n   * `lilconfig#search()` and `lilconfig#load`.\n   * Not _actually_ a `SinonStubbedInstance`, but duck-typed.\n   * @type {import('sinon').SinonStubbedInstance<ReturnType<import('lilconfig').lilconfig>>}\n   */\n  let lc;\n\n  /**\n   * @type {typeof import('../lib/config-file')}\n   */\n  let configFileModule;\n\n  let mocks;\n\n  before(function () {\n    // generally called via the CLI parser, this needs to be done manually in tests.\n    // we don't need to do this before _each_ test, because we're not changing the schema.\n    // if we did change the schema, this would need to be in `beforeEach()` and `afterEach()`\n    // would need to call `schema.reset()`.\n    schema.finalizeSchema();\n  });\n\n  beforeEach(function () {\n    sandbox = sinon.createSandbox();\n\n    // we have to manually type this (and `search()`) because we'd only get the real type\n    // when stubbing an object prop; e.g., `stub(lilconfig, 'load')`\n    const load = /** @type {AsyncSearcherLoadStub} */ (\n      sandbox.stub().resolves({\n        config: GOOD_JSON_CONFIG,\n        filepath: GOOD_JSON_CONFIG_FILEPATH,\n      })\n    );\n    load.withArgs(GOOD_YAML_CONFIG_FILEPATH).resolves({\n      config: GOOD_YAML_CONFIG,\n      filepath: GOOD_YAML_CONFIG_FILEPATH,\n    });\n    load.withArgs(BAD_JSON_CONFIG_FILEPATH).resolves({\n      config: BAD_JSON_CONFIG,\n      filepath: BAD_JSON_CONFIG_FILEPATH,\n    });\n\n    const search = /** @type {AsyncSearcherLoadStub} */ (\n      sandbox.stub().resolves({\n        config: GOOD_JSON_CONFIG,\n        filepath: GOOD_JSON_CONFIG_FILEPATH,\n      })\n    );\n\n    lc = {\n      load,\n      search,\n    };\n\n    mocks = {\n      lilconfig: {\n        lilconfig: sandbox.stub().returns(lc),\n      },\n      '@sidvind/better-ajv-errors': sandbox.stub().returns(''),\n    };\n\n    // loads the `config-file` module using the lilconfig mock.\n    // we only mock lilconfig because it'd otherwise be a pain in the rear to test\n    // searching for config files, and it increases the likelihood that we'd load the wrong file.\n    configFileModule = rewiremock.proxy(() => require('../lib/config-file'), mocks);\n    readConfigFile = configFileModule.readConfigFile;\n\n    // just want to be extra-sure `validate()` happens\n    sandbox.spy(schema, 'validate');\n  });\n\n  afterEach(function () {\n    sandbox.restore();\n  });\n\n  describe('readConfigFile()', function () {\n    /**\n     * @type {import('../lib/config-file').ReadConfigFileResult}\n     */\n    let result;\n\n    it('should support yaml', async function () {\n      const {config} = await readConfigFile(GOOD_YAML_CONFIG_FILEPATH);\n      expect(config).to.eql(GOOD_JSON_CONFIG);\n      expect(schema.validate).to.have.been.calledOnce;\n    });\n\n    it('should support json', async function () {\n      const {config} = await readConfigFile(GOOD_JSON_CONFIG_FILEPATH);\n      expect(config).to.eql(GOOD_JSON_CONFIG);\n      expect(schema.validate).to.have.been.calledOnce;\n    });\n\n    it('should support js', async function () {\n      const {config} = await readConfigFile(GOOD_JS_CONFIG_FILEPATH);\n      expect(config).to.eql(GOOD_JSON_CONFIG);\n      expect(schema.validate).to.have.been.calledOnce;\n    });\n\n    describe('when no filepath provided', function () {\n      beforeEach(async function () {\n        result = await readConfigFile();\n      });\n\n      it('should search for a config file', function () {\n        expect(lc.search).to.have.been.calledOnce;\n        expect(schema.validate).to.have.been.calledOnce;\n      });\n\n      it('should not try to load a config file directly', function () {\n        expect(lc.load).to.not.have.been.called;\n      });\n\n      describe('when no config file is found', function () {\n        beforeEach(async function () {\n          lc.search.resolves();\n          /** @type {import('sinon').SinonSpiedMember<typeof schema.validate>} */(schema.validate).resetHistory();\n          result = await readConfigFile();\n        });\n\n        it('should resolve with an empty object', function () {\n          expect(result).to.be.an('object').that.is.empty;\n          expect(schema.validate).not.to.have.been.called;\n        });\n      });\n\n      describe('when a config file is found', function () {\n        describe('when the config file is empty', function () {\n          beforeEach(async function () {\n            lc.search.resolves({\n              isEmpty: true,\n              filepath: '/path/to/file.json',\n              config: {},\n            });\n\n            result = await readConfigFile();\n          });\n\n          it('should resolve with an object with an `isEmpty` property', function () {\n            expect(result).to.have.property('isEmpty', true);\n          });\n        });\n\n        describe('when the config file is not empty', function () {\n          it('should validate the config against a schema', function () {\n            expect(schema.validate).to.have.been.calledOnceWith(\n              GOOD_JSON_CONFIG,\n            );\n          });\n\n          describe('when the config file is valid', function () {\n            beforeEach(async function () {\n              result = await readConfigFile();\n            });\n\n            it('should resolve with an object having `config` property and empty array of errors', function () {\n              expect(result).to.deep.equal({\n                config: GOOD_JSON_CONFIG,\n                errors: [],\n                filepath: GOOD_JSON_CONFIG_FILEPATH,\n              });\n            });\n          });\n\n          describe('when the config file is invalid', function () {\n            beforeEach(function () {\n              lc.search.resolves({\n                config: {foo: 'bar'},\n                filepath: '/path/to/file.json',\n              });\n            });\n\n            beforeEach(async function () {\n              result = await readConfigFile();\n            });\n\n            it('should resolve with an object having a nonempty array of errors', function () {\n              expect(result).to.have.property('errors').that.is.not.empty;\n            });\n          });\n        });\n      });\n    });\n\n    describe('when filepath provided', function () {\n      beforeEach(async function () {\n        result = await readConfigFile('appium.json');\n      });\n\n      it('should not attempt to find a config file', function () {\n        expect(lc.search).to.not.have.been.called;\n      });\n\n      it('should try to load a config file directly', function () {\n        expect(lc.load).to.have.been.calledOnce;\n      });\n\n      describe('when no config file exists at path', function () {\n        beforeEach(function () {\n          lc.load.rejects(Object.assign(new Error(), {code: 'ENOENT'}));\n        });\n\n        it('should reject with user-friendly message', async function () {\n          await expect(readConfigFile('appium.json')).to.be.rejectedWith(\n            /not found at user-provided path/,\n          );\n        });\n      });\n\n      describe('when the config file is invalid JSON', function () {\n        beforeEach(function () {\n          lc.load.rejects(new SyntaxError());\n        });\n\n        it('should reject with user-friendly message', async function () {\n          await expect(readConfigFile('appium.json')).to.be.rejectedWith(\n            /Config file at user-provided path appium.json is invalid/,\n          );\n        });\n      });\n\n      describe('when something else is wrong with loading the config file', function () {\n        beforeEach(function () {\n          lc.load.rejects(new Error('guru meditation'));\n        });\n\n        it('should pass error through', async function () {\n          await expect(readConfigFile('appium.json')).to.be.rejectedWith(\n            /guru meditation/,\n          );\n        });\n      });\n\n      describe('when a config file is found', function () {\n        describe('when the config file is empty', function () {\n          beforeEach(async function () {\n            lc.search.resolves({\n              isEmpty: true,\n              filepath: '/path/to/file.json',\n              config: {},\n            });\n            result = await readConfigFile();\n          });\n\n          it('should resolve with an object with an `isEmpty` property', function () {\n            expect(result).to.have.property('isEmpty', true);\n          });\n        });\n\n        describe('when the config file is not empty', function () {\n          it('should validate the config against a schema', function () {\n            expect(schema.validate).to.have.been.calledOnceWith(\n              GOOD_JSON_CONFIG,\n            );\n          });\n\n          describe('when the config file is valid', function () {\n            beforeEach(async function () {\n              result = await readConfigFile();\n            });\n\n            it('should resolve with an object having `config` property and empty array of errors', function () {\n              expect(result).to.deep.equal({\n                errors: [],\n                config: GOOD_JSON_CONFIG,\n                filepath: GOOD_JSON_CONFIG_FILEPATH,\n              });\n            });\n          });\n\n          describe('when the config file is invalid', function () {\n            beforeEach(async function () {\n              result = await readConfigFile(BAD_JSON_CONFIG_FILEPATH);\n            });\n\n            it('should resolve with an object having a nonempty array of errors', function () {\n              expect(result).to.have.property('errors').that.is.not.empty;\n            });\n          });\n        });\n      });\n    });\n  });\n\n  describe('formatErrors()', function () {\n    describe('when provided `errors` as an empty array', function () {\n      it('should throw', function () {\n        expect(() => configFileModule.formatErrors([])).to.throw(\n          TypeError,\n          'Array of errors must be non-empty',\n        );\n      });\n    });\n\n    describe('when provided `errors` as `undefined`', function () {\n      it('should throw', function () {\n        // @ts-ignore\n        expect(() => configFileModule.formatErrors()).to.throw(\n          TypeError,\n          'Array of errors must be non-empty',\n        );\n      });\n    });\n\n    describe('when provided `errors` as a non-empty array', function () {\n      it('should return a string', function () {\n        // @ts-ignore\n        expect(configFileModule.formatErrors([{}])).to.be.a('string');\n      });\n    });\n\n    describe('when `opts.json` is a string', function () {\n      it('should call `betterAjvErrors()` with option `json: opts.json`', function () {\n        // @ts-ignore\n        configFileModule.formatErrors([{}], {}, {json: '{\"foo\": \"bar\"}'});\n        expect(mocks['@sidvind/better-ajv-errors']).to.have.been.calledWith(\n          schema.getSchema(),\n          {},\n          [{}],\n          {format: 'cli', json: '{\"foo\": \"bar\"}'},\n        );\n      });\n    });\n  });\n});\n\n// the following are just aliases\n\n/**\n * @typedef {import('ajv').ErrorObject} ErrorObject\n */\n\n/**\n * @typedef {import('ajv').ValidateFunction} ValidateFunction\n */\n\n/**\n * @typedef {import('sinon').SinonStubbedMember<ReturnType<import('lilconfig').lilconfig>['load']>} AsyncSearcherLoadStub\n */\n\n/**\n * @typedef {import('sinon').SinonStubbedMember<ReturnType<import('lilconfig').lilconfig>['search']>} AsyncSearcherSearchStub\n */\n"],"file":"test/config-file-specs.js","sourceRoot":"../.."}