codeceptjs 3.5.12-beta.1 → 3.5.12-beta.2

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 (213) hide show
  1. package/docs/advanced.md +351 -0
  2. package/docs/ai.md +248 -0
  3. package/docs/api.md +323 -0
  4. package/docs/basics.md +979 -0
  5. package/docs/bdd.md +539 -0
  6. package/docs/best.md +237 -0
  7. package/docs/books.md +37 -0
  8. package/docs/bootstrap.md +135 -0
  9. package/docs/build/ApiDataFactory.js +410 -0
  10. package/docs/build/Appium.js +2027 -0
  11. package/docs/build/Expect.js +422 -0
  12. package/docs/build/FileSystem.js +228 -0
  13. package/docs/build/GraphQL.js +229 -0
  14. package/docs/build/GraphQLDataFactory.js +309 -0
  15. package/docs/build/JSONResponse.js +338 -0
  16. package/docs/build/Mochawesome.js +71 -0
  17. package/docs/build/Nightmare.js +2152 -0
  18. package/docs/build/OpenAI.js +126 -0
  19. package/docs/build/Playwright.js +5110 -0
  20. package/docs/build/Protractor.js +2706 -0
  21. package/docs/build/Puppeteer.js +3905 -0
  22. package/docs/build/REST.js +344 -0
  23. package/docs/build/TestCafe.js +2125 -0
  24. package/docs/build/WebDriver.js +4240 -0
  25. package/docs/changelog.md +2572 -0
  26. package/docs/commands.md +266 -0
  27. package/docs/community-helpers.md +58 -0
  28. package/docs/configuration.md +157 -0
  29. package/docs/continuous-integration.md +22 -0
  30. package/docs/custom-helpers.md +306 -0
  31. package/docs/data.md +379 -0
  32. package/docs/detox.md +235 -0
  33. package/docs/docker.md +136 -0
  34. package/docs/email.md +183 -0
  35. package/docs/examples.md +149 -0
  36. package/docs/helpers/ApiDataFactory.md +266 -0
  37. package/docs/helpers/Appium.md +1374 -0
  38. package/docs/helpers/Detox.md +586 -0
  39. package/docs/helpers/Expect.md +275 -0
  40. package/docs/helpers/FileSystem.md +152 -0
  41. package/docs/helpers/GraphQL.md +151 -0
  42. package/docs/helpers/GraphQLDataFactory.md +226 -0
  43. package/docs/helpers/JSONResponse.md +254 -0
  44. package/docs/helpers/Mochawesome.md +8 -0
  45. package/docs/helpers/MockRequest.md +377 -0
  46. package/docs/helpers/Nightmare.md +1305 -0
  47. package/docs/helpers/OpenAI.md +70 -0
  48. package/docs/helpers/Playwright.md +2759 -0
  49. package/docs/helpers/Polly.md +44 -0
  50. package/docs/helpers/Protractor.md +1769 -0
  51. package/docs/helpers/Puppeteer-firefox.md +86 -0
  52. package/docs/helpers/Puppeteer.md +2317 -0
  53. package/docs/helpers/REST.md +218 -0
  54. package/docs/helpers/TestCafe.md +1321 -0
  55. package/docs/helpers/WebDriver.md +2547 -0
  56. package/docs/hooks.md +340 -0
  57. package/docs/index.md +111 -0
  58. package/docs/installation.md +75 -0
  59. package/docs/internal-api.md +266 -0
  60. package/docs/locators.md +339 -0
  61. package/docs/mobile-react-native-locators.md +67 -0
  62. package/docs/mobile.md +338 -0
  63. package/docs/pageobjects.md +291 -0
  64. package/docs/parallel.md +400 -0
  65. package/docs/playwright.md +632 -0
  66. package/docs/plugins.md +1259 -0
  67. package/docs/puppeteer.md +316 -0
  68. package/docs/quickstart.md +162 -0
  69. package/docs/react.md +70 -0
  70. package/docs/reports.md +392 -0
  71. package/docs/secrets.md +36 -0
  72. package/docs/shadow.md +68 -0
  73. package/docs/shared/keys.mustache +31 -0
  74. package/docs/shared/react.mustache +1 -0
  75. package/docs/testcafe.md +174 -0
  76. package/docs/translation.md +247 -0
  77. package/docs/tutorial.md +271 -0
  78. package/docs/typescript.md +180 -0
  79. package/docs/ui.md +59 -0
  80. package/docs/videos.md +28 -0
  81. package/docs/visual.md +202 -0
  82. package/docs/vue.md +143 -0
  83. package/docs/webapi/amOnPage.mustache +11 -0
  84. package/docs/webapi/appendField.mustache +11 -0
  85. package/docs/webapi/attachFile.mustache +12 -0
  86. package/docs/webapi/blur.mustache +18 -0
  87. package/docs/webapi/checkOption.mustache +13 -0
  88. package/docs/webapi/clearCookie.mustache +9 -0
  89. package/docs/webapi/clearField.mustache +9 -0
  90. package/docs/webapi/click.mustache +25 -0
  91. package/docs/webapi/clickLink.mustache +8 -0
  92. package/docs/webapi/closeCurrentTab.mustache +7 -0
  93. package/docs/webapi/closeOtherTabs.mustache +8 -0
  94. package/docs/webapi/dontSee.mustache +11 -0
  95. package/docs/webapi/dontSeeCheckboxIsChecked.mustache +10 -0
  96. package/docs/webapi/dontSeeCookie.mustache +8 -0
  97. package/docs/webapi/dontSeeCurrentUrlEquals.mustache +10 -0
  98. package/docs/webapi/dontSeeElement.mustache +8 -0
  99. package/docs/webapi/dontSeeElementInDOM.mustache +8 -0
  100. package/docs/webapi/dontSeeInCurrentUrl.mustache +4 -0
  101. package/docs/webapi/dontSeeInField.mustache +11 -0
  102. package/docs/webapi/dontSeeInSource.mustache +8 -0
  103. package/docs/webapi/dontSeeInTitle.mustache +8 -0
  104. package/docs/webapi/doubleClick.mustache +13 -0
  105. package/docs/webapi/downloadFile.mustache +12 -0
  106. package/docs/webapi/dragAndDrop.mustache +9 -0
  107. package/docs/webapi/dragSlider.mustache +11 -0
  108. package/docs/webapi/executeAsyncScript.mustache +24 -0
  109. package/docs/webapi/executeScript.mustache +26 -0
  110. package/docs/webapi/fillField.mustache +16 -0
  111. package/docs/webapi/focus.mustache +13 -0
  112. package/docs/webapi/forceClick.mustache +28 -0
  113. package/docs/webapi/forceRightClick.mustache +18 -0
  114. package/docs/webapi/grabAllWindowHandles.mustache +7 -0
  115. package/docs/webapi/grabAttributeFrom.mustache +10 -0
  116. package/docs/webapi/grabAttributeFromAll.mustache +9 -0
  117. package/docs/webapi/grabBrowserLogs.mustache +9 -0
  118. package/docs/webapi/grabCookie.mustache +11 -0
  119. package/docs/webapi/grabCssPropertyFrom.mustache +11 -0
  120. package/docs/webapi/grabCssPropertyFromAll.mustache +10 -0
  121. package/docs/webapi/grabCurrentUrl.mustache +9 -0
  122. package/docs/webapi/grabCurrentWindowHandle.mustache +6 -0
  123. package/docs/webapi/grabDataFromPerformanceTiming.mustache +20 -0
  124. package/docs/webapi/grabElementBoundingRect.mustache +20 -0
  125. package/docs/webapi/grabGeoLocation.mustache +8 -0
  126. package/docs/webapi/grabHTMLFrom.mustache +10 -0
  127. package/docs/webapi/grabHTMLFromAll.mustache +9 -0
  128. package/docs/webapi/grabNumberOfOpenTabs.mustache +8 -0
  129. package/docs/webapi/grabNumberOfVisibleElements.mustache +9 -0
  130. package/docs/webapi/grabPageScrollPosition.mustache +8 -0
  131. package/docs/webapi/grabPopupText.mustache +5 -0
  132. package/docs/webapi/grabSource.mustache +8 -0
  133. package/docs/webapi/grabTextFrom.mustache +10 -0
  134. package/docs/webapi/grabTextFromAll.mustache +9 -0
  135. package/docs/webapi/grabTitle.mustache +8 -0
  136. package/docs/webapi/grabValueFrom.mustache +9 -0
  137. package/docs/webapi/grabValueFromAll.mustache +8 -0
  138. package/docs/webapi/grabWebElement.mustache +9 -0
  139. package/docs/webapi/grabWebElements.mustache +9 -0
  140. package/docs/webapi/moveCursorTo.mustache +12 -0
  141. package/docs/webapi/openNewTab.mustache +7 -0
  142. package/docs/webapi/pressKey.mustache +12 -0
  143. package/docs/webapi/pressKeyDown.mustache +12 -0
  144. package/docs/webapi/pressKeyUp.mustache +12 -0
  145. package/docs/webapi/pressKeyWithKeyNormalization.mustache +60 -0
  146. package/docs/webapi/refreshPage.mustache +6 -0
  147. package/docs/webapi/resizeWindow.mustache +6 -0
  148. package/docs/webapi/rightClick.mustache +14 -0
  149. package/docs/webapi/saveElementScreenshot.mustache +10 -0
  150. package/docs/webapi/saveScreenshot.mustache +12 -0
  151. package/docs/webapi/say.mustache +10 -0
  152. package/docs/webapi/scrollIntoView.mustache +11 -0
  153. package/docs/webapi/scrollPageToBottom.mustache +6 -0
  154. package/docs/webapi/scrollPageToTop.mustache +6 -0
  155. package/docs/webapi/scrollTo.mustache +12 -0
  156. package/docs/webapi/see.mustache +11 -0
  157. package/docs/webapi/seeAttributesOnElements.mustache +9 -0
  158. package/docs/webapi/seeCheckboxIsChecked.mustache +10 -0
  159. package/docs/webapi/seeCookie.mustache +8 -0
  160. package/docs/webapi/seeCssPropertiesOnElements.mustache +9 -0
  161. package/docs/webapi/seeCurrentUrlEquals.mustache +11 -0
  162. package/docs/webapi/seeElement.mustache +8 -0
  163. package/docs/webapi/seeElementInDOM.mustache +8 -0
  164. package/docs/webapi/seeInCurrentUrl.mustache +8 -0
  165. package/docs/webapi/seeInField.mustache +12 -0
  166. package/docs/webapi/seeInPopup.mustache +8 -0
  167. package/docs/webapi/seeInSource.mustache +7 -0
  168. package/docs/webapi/seeInTitle.mustache +8 -0
  169. package/docs/webapi/seeNumberOfElements.mustache +11 -0
  170. package/docs/webapi/seeNumberOfVisibleElements.mustache +10 -0
  171. package/docs/webapi/seeTextEquals.mustache +9 -0
  172. package/docs/webapi/seeTitleEquals.mustache +8 -0
  173. package/docs/webapi/selectOption.mustache +21 -0
  174. package/docs/webapi/setCookie.mustache +16 -0
  175. package/docs/webapi/setGeoLocation.mustache +12 -0
  176. package/docs/webapi/switchTo.mustache +9 -0
  177. package/docs/webapi/switchToNextTab.mustache +10 -0
  178. package/docs/webapi/switchToPreviousTab.mustache +10 -0
  179. package/docs/webapi/type.mustache +21 -0
  180. package/docs/webapi/uncheckOption.mustache +13 -0
  181. package/docs/webapi/wait.mustache +8 -0
  182. package/docs/webapi/waitForClickable.mustache +11 -0
  183. package/docs/webapi/waitForDetached.mustache +10 -0
  184. package/docs/webapi/waitForElement.mustache +11 -0
  185. package/docs/webapi/waitForEnabled.mustache +6 -0
  186. package/docs/webapi/waitForFunction.mustache +17 -0
  187. package/docs/webapi/waitForInvisible.mustache +10 -0
  188. package/docs/webapi/waitForNumberOfTabs.mustache +9 -0
  189. package/docs/webapi/waitForText.mustache +13 -0
  190. package/docs/webapi/waitForValue.mustache +10 -0
  191. package/docs/webapi/waitForVisible.mustache +10 -0
  192. package/docs/webapi/waitInUrl.mustache +9 -0
  193. package/docs/webapi/waitNumberOfVisibleElements.mustache +10 -0
  194. package/docs/webapi/waitToHide.mustache +10 -0
  195. package/docs/webapi/waitUrlEquals.mustache +10 -0
  196. package/docs/webdriver.md +701 -0
  197. package/docs/wiki/Books-&-Posts.md +27 -0
  198. package/docs/wiki/Community-Helpers-&-Plugins.md +53 -0
  199. package/docs/wiki/Converting-Playwright-to-Istanbul-Coverage.md +61 -0
  200. package/docs/wiki/Examples.md +145 -0
  201. package/docs/wiki/Google-Summer-of-Code-(GSoC)-2020.md +68 -0
  202. package/docs/wiki/Home.md +16 -0
  203. package/docs/wiki/Migration-to-Appium-v2---CodeceptJS.md +83 -0
  204. package/docs/wiki/Release-Process.md +24 -0
  205. package/docs/wiki/Roadmap.md +23 -0
  206. package/docs/wiki/Tests.md +1393 -0
  207. package/docs/wiki/Upgrading-to-CodeceptJS-3.md +153 -0
  208. package/docs/wiki/Videos.md +19 -0
  209. package/lib/css2xpath/js/css_to_xpath.js +20 -0
  210. package/lib/css2xpath/js/expression.js +23 -0
  211. package/lib/css2xpath/js/renderer.js +239 -0
  212. package/lib/locator.js +15 -1
  213. package/package.json +12 -9
@@ -0,0 +1,422 @@
1
+ const chai = require('chai');
2
+ const output = require('../output');
3
+
4
+ const { expect } = chai;
5
+
6
+ chai.use(require('chai-string'));
7
+ // @ts-ignore
8
+ chai.use(require('chai-exclude'));
9
+ chai.use(require('chai-match-pattern'));
10
+
11
+ /**
12
+ * This helper allows performing assertions based on Chai.
13
+ *
14
+ * ### Examples
15
+ *
16
+ * Zero-configuration when paired with other helpers like REST, Playwright:
17
+ *
18
+ * ```js
19
+ * // inside codecept.conf.js
20
+ *{
21
+ * helpers: {
22
+ * Playwright: {...},
23
+ * Expect: {},
24
+ * }
25
+ *}
26
+ * ```
27
+ *
28
+ * ## Methods
29
+ */
30
+ class ExpectHelper {
31
+ /**
32
+ *
33
+ * @param {*} actualValue
34
+ * @param {*} expectedValue
35
+ * @param {*} [customErrorMsg]
36
+ */
37
+ expectEqual(actualValue, expectedValue, customErrorMsg = '') {
38
+ // @ts-ignore
39
+ output.step(`I expect "${JSON.stringify(actualValue)}" to equal "${JSON.stringify(expectedValue)}"`);
40
+ return expect(actualValue, customErrorMsg).to.equal(expectedValue);
41
+ }
42
+
43
+ /**
44
+ *
45
+ * @param {*} actualValue
46
+ * @param {*} expectedValue
47
+ * @param {*} [customErrorMsg]
48
+ */
49
+ expectNotEqual(actualValue, expectedValue, customErrorMsg = '') {
50
+ // @ts-ignore
51
+ output.step(`I expect "${JSON.stringify(actualValue)}" to not equal "${JSON.stringify(expectedValue)}"`);
52
+ return expect(actualValue, customErrorMsg).not.to.equal(expectedValue);
53
+ }
54
+
55
+ /**
56
+ *
57
+ * @param {*} actualValue
58
+ * @param {*} expectedValue
59
+ * @param {*} [customErrorMsg]
60
+
61
+ */
62
+ expectDeepEqual(actualValue, expectedValue, customErrorMsg = '') {
63
+ // @ts-ignore
64
+ output.step(`I expect "${JSON.stringify(actualValue)}" to deep equal "${JSON.stringify(expectedValue)}"`);
65
+ return expect(actualValue, customErrorMsg).to.deep.equal(expectedValue);
66
+ }
67
+
68
+ /**
69
+ *
70
+ * @param {*} actualValue
71
+ * @param {*} expectedValue
72
+ * @param {*} [customErrorMsg]
73
+ */
74
+ expectNotDeepEqual(actualValue, expectedValue, customErrorMsg = '') {
75
+ // @ts-ignore
76
+ output.step(`I expect "${JSON.stringify(actualValue)}" to not deep equal "${JSON.stringify(expectedValue)}"`);
77
+ return expect(actualValue, customErrorMsg).to.not.deep.equal(expectedValue);
78
+ }
79
+
80
+ /**
81
+ *
82
+ * @param {*} actualValue
83
+ * @param {*} expectedValueToContain
84
+ * @param {*} [customErrorMsg]
85
+ */
86
+ expectContain(actualValue, expectedValueToContain, customErrorMsg = '') {
87
+ // @ts-ignore
88
+ output.step(`I expect "${JSON.stringify(actualValue)}" to contain "${JSON.stringify(expectedValueToContain)}"`);
89
+ return expect(actualValue, customErrorMsg).to.contain(
90
+ expectedValueToContain,
91
+ );
92
+ }
93
+
94
+ /**
95
+ *
96
+ * @param {*} actualValue
97
+ * @param {*} expectedValueToNotContain
98
+ * @param {*} [customErrorMsg]
99
+ */
100
+ expectNotContain(
101
+ actualValue,
102
+ expectedValueToNotContain,
103
+ customErrorMsg = '',
104
+ ) {
105
+ // @ts-ignore
106
+ output.step(`I expect "${JSON.stringify(actualValue)}" to not contain "${JSON.stringify(expectedValueToNotContain)}"`);
107
+ return expect(actualValue, customErrorMsg).not.to.contain(
108
+ expectedValueToNotContain,
109
+ );
110
+ }
111
+
112
+ /**
113
+ *
114
+ * @param {*} actualValue
115
+ * @param {*} expectedValueToStartWith
116
+ * @param {*} [customErrorMsg]
117
+ */
118
+ expectStartsWith(actualValue, expectedValueToStartWith, customErrorMsg = '') {
119
+ // @ts-ignore
120
+ output.step(`I expect "${JSON.stringify(actualValue)}" to start with "${JSON.stringify(expectedValueToStartWith)}"`);
121
+ return expect(actualValue, customErrorMsg).to.startsWith(
122
+ expectedValueToStartWith,
123
+ );
124
+ }
125
+
126
+ /**
127
+ *
128
+ * @param {*} actualValue
129
+ * @param {*} expectedValueToNotStartWith
130
+ * @param {*} [customErrorMsg]
131
+ */
132
+ expectNotStartsWith(
133
+ actualValue,
134
+ expectedValueToNotStartWith,
135
+ customErrorMsg = '',
136
+ ) {
137
+ // @ts-ignore
138
+ output.step(`I expect "${JSON.stringify(actualValue)}" to not start with "${JSON.stringify(expectedValueToNotStartWith)}"`);
139
+ return expect(actualValue, customErrorMsg).not.to.startsWith(
140
+ expectedValueToNotStartWith,
141
+ );
142
+ }
143
+
144
+ /**
145
+ * @param {*} actualValue
146
+ * @param {*} expectedValueToEndWith
147
+ * @param {*} [customErrorMsg]
148
+ */
149
+ expectEndsWith(actualValue, expectedValueToEndWith, customErrorMsg = '') {
150
+ // @ts-ignore
151
+ output.step(`I expect "${JSON.stringify(actualValue)}" to end with "${JSON.stringify(expectedValueToEndWith)}"`);
152
+ return expect(actualValue, customErrorMsg).to.endsWith(
153
+ expectedValueToEndWith,
154
+ );
155
+ }
156
+
157
+ /**
158
+ * @param {*} actualValue
159
+ * @param {*} expectedValueToNotEndWith
160
+ * @param {*} [customErrorMsg]
161
+ */
162
+ expectNotEndsWith(
163
+ actualValue,
164
+ expectedValueToNotEndWith,
165
+ customErrorMsg = '',
166
+ ) {
167
+ // @ts-ignore
168
+ output.step(`I expect "${JSON.stringify(actualValue)}" to not end with "${JSON.stringify(expectedValueToNotEndWith)}"`);
169
+ return expect(actualValue, customErrorMsg).not.to.endsWith(
170
+ expectedValueToNotEndWith,
171
+ );
172
+ }
173
+
174
+ /**
175
+ * @param {*} targetData
176
+ * @param {*} jsonSchema
177
+ * @param {*} [customErrorMsg]
178
+ */
179
+ expectJsonSchema(targetData, jsonSchema, customErrorMsg = '') {
180
+ // @ts-ignore
181
+ output.step(`I expect "${JSON.stringify(targetData)}" to match this JSON schema "${JSON.stringify(jsonSchema)}"`);
182
+ chai.use(require('chai-json-schema'));
183
+ return expect(targetData, customErrorMsg).to.be.jsonSchema(jsonSchema);
184
+ }
185
+
186
+ /**
187
+ * @param {*} targetData
188
+ * @param {*} jsonSchema
189
+ * @param {*} [customErrorMsg]
190
+ * @param {*} ajvOptions Pass AJV options
191
+ */
192
+ expectJsonSchemaUsingAJV(
193
+ targetData,
194
+ jsonSchema,
195
+ customErrorMsg = '',
196
+ ajvOptions = { allErrors: true },
197
+ ) {
198
+ // @ts-ignore
199
+ output.step(`I expect "${JSON.stringify(targetData)}" to match this JSON schema using AJV "${JSON.stringify(jsonSchema)}"`);
200
+ chai.use(require('chai-json-schema-ajv').create(ajvOptions));
201
+ return expect(targetData, customErrorMsg).to.be.jsonSchema(jsonSchema);
202
+ }
203
+
204
+ /**
205
+ * @param {*} targetData
206
+ * @param {*} propertyName
207
+ * @param {*} [customErrorMsg]
208
+ */
209
+ expectHasProperty(targetData, propertyName, customErrorMsg = '') {
210
+ // @ts-ignore
211
+ output.step(`I expect "${JSON.stringify(targetData)}" to have property: "${JSON.stringify(propertyName)}"`);
212
+ return expect(targetData, customErrorMsg).to.have.property(propertyName);
213
+ }
214
+
215
+ /**
216
+ * @param {*} targetData
217
+ * @param {*} propertyName
218
+ * @param {*} [customErrorMsg]
219
+ */
220
+ expectHasAProperty(targetData, propertyName, customErrorMsg = '') {
221
+ // @ts-ignore
222
+ output.step(`I expect "${JSON.stringify(targetData)}" to have a property: "${JSON.stringify(propertyName)}"`);
223
+ return expect(targetData, customErrorMsg).to.have.a.property(propertyName);
224
+ }
225
+
226
+ /**
227
+ * @param {*} targetData
228
+ * @param {*} type
229
+ * @param {*} [customErrorMsg]
230
+ */
231
+ expectToBeA(targetData, type, customErrorMsg = '') {
232
+ // @ts-ignore
233
+ output.step(`I expect "${JSON.stringify(targetData)}" to be a "${JSON.stringify(type)}"`);
234
+ return expect(targetData, customErrorMsg).to.be.a(type);
235
+ }
236
+
237
+ /**
238
+ * @param {*} targetData
239
+ * @param {*} type
240
+ * @param {*} [customErrorMsg]
241
+ */
242
+ expectToBeAn(targetData, type, customErrorMsg = '') {
243
+ // @ts-ignore
244
+ output.step(`I expect "${JSON.stringify(targetData)}" to be an "${JSON.stringify(type)}"`);
245
+ return expect(targetData, customErrorMsg).to.be.an(type);
246
+ }
247
+
248
+ /**
249
+ * @param {*} targetData
250
+ * @param {*} regex
251
+ * @param {*} [customErrorMsg]
252
+ */
253
+ expectMatchRegex(targetData, regex, customErrorMsg = '') {
254
+ // @ts-ignore
255
+ output.step(`I expect "${JSON.stringify(targetData)}" to match the regex "${JSON.stringify(regex)}"`);
256
+ return expect(targetData, customErrorMsg).to.match(regex);
257
+ }
258
+
259
+ /**
260
+ * @param {*} targetData
261
+ * @param {*} length
262
+ * @param {*} [customErrorMsg]
263
+ */
264
+ expectLengthOf(targetData, length, customErrorMsg = '') {
265
+ // @ts-ignore
266
+ output.step(`I expect "${JSON.stringify(targetData)}" to have length of "${JSON.stringify(length)}"`);
267
+ return expect(targetData, customErrorMsg).to.have.lengthOf(length);
268
+ }
269
+
270
+ /**
271
+ * @param {*} targetData
272
+ * @param {*} [customErrorMsg]
273
+ */
274
+ expectEmpty(targetData, customErrorMsg = '') {
275
+ // @ts-ignore
276
+ output.step(`I expect "${JSON.stringify(targetData)}" to be empty`);
277
+ return expect(targetData, customErrorMsg).to.be.empty;
278
+ }
279
+
280
+ /**
281
+ * @param {*} targetData
282
+ * @param {*} [customErrorMsg]
283
+ */
284
+ expectTrue(targetData, customErrorMsg = '') {
285
+ // @ts-ignore
286
+ output.step(`I expect "${JSON.stringify(targetData)}" to be true`);
287
+ return expect(targetData, customErrorMsg).to.be.true;
288
+ }
289
+
290
+ /**
291
+ * @param {*} targetData
292
+ * @param {*} [customErrorMsg]
293
+ */
294
+ expectFalse(targetData, customErrorMsg = '') {
295
+ // @ts-ignore
296
+ output.step(`I expect "${JSON.stringify(targetData)}" to be false`);
297
+ return expect(targetData, customErrorMsg).to.be.false;
298
+ }
299
+
300
+ /**
301
+ * @param {*} targetData
302
+ * @param {*} aboveThan
303
+ * @param {*} [customErrorMsg]
304
+ */
305
+ expectAbove(targetData, aboveThan, customErrorMsg = '') {
306
+ // @ts-ignore
307
+ output.step(`I expect "${JSON.stringify(targetData)}" to be above ${JSON.stringify(aboveThan)}`);
308
+ return expect(targetData, customErrorMsg).to.be.above(aboveThan);
309
+ }
310
+
311
+ /**
312
+ * @param {*} targetData
313
+ * @param {*} belowThan
314
+ * @param {*} [customErrorMsg]
315
+ */
316
+ expectBelow(targetData, belowThan, customErrorMsg = '') {
317
+ // @ts-ignore
318
+ output.step(`I expect "${JSON.stringify(targetData)}" to be below ${JSON.stringify(belowThan)}`);
319
+ return expect(targetData, customErrorMsg).to.be.below(belowThan);
320
+ }
321
+
322
+ /**
323
+ * @param {*} targetData
324
+ * @param {*} lengthAboveThan
325
+ * @param {*} [customErrorMsg]
326
+ */
327
+ expectLengthAboveThan(targetData, lengthAboveThan, customErrorMsg = '') {
328
+ // @ts-ignore
329
+ output.step(`I expect "${JSON.stringify(targetData)}" to have length of above ${JSON.stringify(lengthAboveThan)}`);
330
+ return expect(targetData, customErrorMsg).to.have.lengthOf.above(
331
+ lengthAboveThan,
332
+ );
333
+ }
334
+
335
+ /**
336
+ * @param {*} targetData
337
+ * @param {*} lengthBelowThan
338
+ * @param {*} [customErrorMsg]
339
+ */
340
+ expectLengthBelowThan(targetData, lengthBelowThan, customErrorMsg = '') {
341
+ // @ts-ignore
342
+ output.step(`I expect "${JSON.stringify(targetData)}" to have length of below ${JSON.stringify(lengthBelowThan)}`);
343
+ return expect(targetData, customErrorMsg).to.have.lengthOf.below(
344
+ lengthBelowThan,
345
+ );
346
+ }
347
+
348
+ /**
349
+ * @param {*} actualValue
350
+ * @param {*} expectedValue
351
+ * @param {*} [customErrorMsg]
352
+ */
353
+ expectEqualIgnoreCase(actualValue, expectedValue, customErrorMsg = '') {
354
+ // @ts-ignore
355
+ output.step(`I expect and ingore case "${JSON.stringify(actualValue)}" to equal "${JSON.stringify(expectedValue)}"`);
356
+ return expect(actualValue, customErrorMsg).to.equalIgnoreCase(
357
+ expectedValue,
358
+ );
359
+ }
360
+
361
+ /**
362
+ * expects members of two arrays are deeply equal
363
+ * @param {*} actualValue
364
+ * @param {*} expectedValue
365
+ * @param {*} [customErrorMsg]
366
+ */
367
+ expectDeepMembers(actualValue, expectedValue, customErrorMsg = '') {
368
+ // @ts-ignore
369
+ output.step(`I expect members of "${JSON.stringify(actualValue)}" and "${JSON.stringify(expectedValue)}" arrays are deeply equal`);
370
+ return expect(actualValue, customErrorMsg).to.have.deep.members(
371
+ expectedValue,
372
+ );
373
+ }
374
+
375
+ /**
376
+ * expects an array to be a superset of another array
377
+ * @param {*} superset
378
+ * @param {*} set
379
+ * @param {*} [customErrorMsg]
380
+ */
381
+ expectDeepIncludeMembers(superset, set, customErrorMsg = '') {
382
+ // @ts-ignore
383
+ output.step(`I expect "${JSON.stringify(superset)}" array to be a superset of "${JSON.stringify(set)}" array`);
384
+ return expect(superset, customErrorMsg).to.deep.include.members(
385
+ set,
386
+ );
387
+ }
388
+
389
+ /**
390
+ * expects members of two JSON objects are deeply equal excluding some properties
391
+ * @param {*} actualValue
392
+ * @param {*} expectedValue
393
+ * @param {*} fieldsToExclude
394
+ * @param {*} [customErrorMsg]
395
+ */
396
+ expectDeepEqualExcluding(
397
+ actualValue,
398
+ expectedValue,
399
+ fieldsToExclude,
400
+ customErrorMsg = '',
401
+ ) {
402
+ // @ts-ignore
403
+ output.step(`I expect members of "${JSON.stringify(actualValue)}" and "${JSON.stringify(expectedValue)}" JSON objects are deeply equal excluding properties: ${JSON.stringify(fieldsToExclude)}`);
404
+ return expect(actualValue, customErrorMsg)
405
+ .excludingEvery(fieldsToExclude)
406
+ .to.deep.equal(expectedValue);
407
+ }
408
+
409
+ /**
410
+ * expects a JSON object matches a provided pattern
411
+ * @param {*} actualValue
412
+ * @param {*} expectedPattern
413
+ * @param {*} [customErrorMsg]
414
+ */
415
+ expectMatchesPattern(actualValue, expectedPattern, customErrorMsg = '') {
416
+ // @ts-ignore
417
+ output.step(`I expect "${JSON.stringify(actualValue)}" to match the ${JSON.stringify(expectedPattern)} pattern`);
418
+ return expect(actualValue, customErrorMsg).to.matchPattern(expectedPattern);
419
+ }
420
+ }
421
+
422
+ module.exports = ExpectHelper;
@@ -0,0 +1,228 @@
1
+ const assert = require('assert');
2
+ const path = require('path');
3
+ const fs = require('fs');
4
+
5
+ const Helper = require('@codeceptjs/helper');
6
+ const { fileExists } = require('../utils');
7
+ const { fileIncludes } = require('../assert/include');
8
+ const { fileEquals } = require('../assert/equal');
9
+
10
+ /**
11
+ * Helper for testing filesystem.
12
+ * Can be easily used to check file structures:
13
+ *
14
+ * ```js
15
+ * I.amInPath('test');
16
+ * I.seeFile('codecept.js');
17
+ * I.seeInThisFile('FileSystem');
18
+ * I.dontSeeInThisFile("WebDriver");
19
+ * ```
20
+ *
21
+ * ## Configuration
22
+ *
23
+ * Enable helper in config file:
24
+ *
25
+ * ```js
26
+ * helpers: {
27
+ * FileSystem: {},
28
+ * }
29
+ * ```
30
+ *
31
+ * ## Methods
32
+ */
33
+ class FileSystem extends Helper {
34
+ constructor() {
35
+ super();
36
+ this.dir = global.codecept_dir;
37
+ this.file = '';
38
+ }
39
+
40
+ _before() {
41
+ this.debugSection('Dir', this.dir);
42
+ }
43
+
44
+ /**
45
+ * Enters a directory In local filesystem.
46
+ * Starts from a current directory
47
+ * @param {string} openPath
48
+ */
49
+ amInPath(openPath) {
50
+ this.dir = path.join(global.codecept_dir, openPath);
51
+ this.debugSection('Dir', this.dir);
52
+ }
53
+
54
+ /**
55
+ * Writes text to file
56
+ * @param {string} name
57
+ * @param {string} text
58
+ */
59
+ writeToFile(name, text) {
60
+ fs.writeFileSync(path.join(this.dir, name), text);
61
+ }
62
+
63
+ /**
64
+ * Checks that file exists
65
+ * @param {string} name
66
+ */
67
+ seeFile(name) {
68
+ this.file = path.join(this.dir, name);
69
+ this.debugSection('File', this.file);
70
+ assert.ok(fileExists(this.file), `File ${name} not found in ${this.dir}`);
71
+ }
72
+
73
+ /**
74
+ * Waits for the file to be present in the current directory.
75
+ *
76
+ * ```js
77
+ * I.handleDownloads('downloads/largeFilesName.txt');
78
+ * I.click('Download large File');
79
+ * I.amInPath('output/downloads');
80
+ * I.waitForFile('largeFilesName.txt', 10); // wait 10 seconds for file
81
+ * ```
82
+ * @param {string} name
83
+ * @param {number} [sec=1] seconds to wait
84
+ */
85
+ async waitForFile(name, sec = 1) {
86
+ if (sec === 0) assert.fail('Use `seeFile` instead of waiting 0 seconds!');
87
+ const waitTimeout = sec * 1000;
88
+ this.file = path.join(this.dir, name);
89
+ this.debugSection('File', this.file);
90
+ return isFileExists(this.file, waitTimeout).catch(() => {
91
+ throw new Error(`file (${name}) still not present in directory ${this.dir} after ${waitTimeout / 1000} sec`);
92
+ });
93
+ }
94
+
95
+ /**
96
+ * Checks that file with a name including given text exists in the current directory.
97
+ *
98
+ *```js
99
+ * I.handleDownloads();
100
+ * I.click('Download as PDF');
101
+ * I.amInPath('output/downloads');
102
+ * I.seeFileNameMatching('.pdf');
103
+ * ```
104
+ * @param {string} text
105
+ */
106
+ seeFileNameMatching(text) {
107
+ assert.ok(
108
+ this.grabFileNames().some(file => file.includes(text)),
109
+ `File name which contains ${text} not found in ${this.dir}`,
110
+ );
111
+ }
112
+
113
+ /**
114
+ * Checks that file found by `seeFile` includes a text.
115
+ * @param {string} text
116
+ * @param {string} [encoding='utf8']
117
+ */
118
+ seeInThisFile(text, encoding = 'utf8') {
119
+ const content = getFileContents(this.file, encoding);
120
+ fileIncludes(this.file).assert(text, content);
121
+ }
122
+
123
+ /**
124
+ * Checks that file found by `seeFile` doesn't include text.
125
+ * @param {string} text
126
+ * @param {string} [encoding='utf8']
127
+ */
128
+ dontSeeInThisFile(text, encoding = 'utf8') {
129
+ const content = getFileContents(this.file, encoding);
130
+ fileIncludes(this.file).negate(text, content);
131
+ }
132
+
133
+ /**
134
+ * Checks that contents of file found by `seeFile` equal to text.
135
+ * @param {string} text
136
+ * @param {string} [encoding='utf8']
137
+ */
138
+ seeFileContentsEqual(text, encoding = 'utf8') {
139
+ const content = getFileContents(this.file, encoding);
140
+ fileEquals(this.file).assert(text, content);
141
+ }
142
+
143
+ /**
144
+ * Checks that contents of the file found by `seeFile` equal to contents of the file at `pathToReferenceFile`.
145
+ * @param {string} pathToReferenceFile
146
+ * @param {string} [encoding='utf8']
147
+ * @param {string} [encodingReference='utf8']
148
+ */
149
+ seeFileContentsEqualReferenceFile(pathToReferenceFile, encoding = 'utf8', encodingReference = '') {
150
+ const content = getFileContents(this.file, encoding);
151
+ assert.ok(fileExists(pathToReferenceFile), `Reference file ${pathToReferenceFile} not found.`);
152
+ encodingReference = encodingReference || encoding;
153
+ const expectedContent = getFileContents(pathToReferenceFile, encodingReference);
154
+ fileEquals(this.file).assert(expectedContent, content);
155
+ }
156
+
157
+ /**
158
+ * Checks that contents of file found by `seeFile` doesn't equal to text.
159
+ * @param {string} text
160
+ * @param {string} [encoding='utf8']
161
+ */
162
+ dontSeeFileContentsEqual(text, encoding = 'utf8') {
163
+ const content = getFileContents(this.file, encoding);
164
+ fileEquals(this.file).negate(text, content);
165
+ }
166
+
167
+ /**
168
+ * Returns file names in current directory.
169
+ *
170
+ * ```js
171
+ * I.handleDownloads();
172
+ * I.click('Download Files');
173
+ * I.amInPath('output/downloads');
174
+ * const downloadedFileNames = I.grabFileNames();
175
+ * ```
176
+ */
177
+ grabFileNames() {
178
+ return fs.readdirSync(this.dir)
179
+ .filter(item => !fs.lstatSync(path.join(this.dir, item)).isDirectory());
180
+ }
181
+ }
182
+
183
+ module.exports = FileSystem;
184
+
185
+ /**
186
+ * @param {string} file
187
+ * @param {string} [encoding='utf8']
188
+ * @private
189
+ * @returns {string}
190
+ */
191
+ function getFileContents(file, encoding = 'utf8') {
192
+ if (!file) assert.fail('No files were opened, please use seeFile action');
193
+ if (encoding === '') assert.fail('Encoding is an empty string, please set a valid encoding');
194
+ return fs.readFileSync(file, encoding);
195
+ }
196
+
197
+ /**
198
+ * @param {string} file
199
+ * @param {number} timeout
200
+ * @private
201
+ * @returns {Promise<any>}
202
+ */
203
+ function isFileExists(file, timeout) {
204
+ return new Promise(((resolve, reject) => {
205
+ const timer = setTimeout(() => {
206
+ watcher.close();
207
+ reject(new Error('File did not exists and was not created during the timeout.'));
208
+ }, timeout);
209
+
210
+ const dir = path.dirname(file);
211
+ const basename = path.basename(file);
212
+ const watcher = fs.watch(dir, (eventType, filename) => {
213
+ if (eventType === 'rename' && filename === basename) {
214
+ clearTimeout(timer);
215
+ watcher.close();
216
+ resolve();
217
+ }
218
+ });
219
+
220
+ fs.access(file, fs.constants.R_OK, (err) => {
221
+ if (!err) {
222
+ clearTimeout(timer);
223
+ watcher.close();
224
+ resolve();
225
+ }
226
+ });
227
+ }));
228
+ }