2021-05-05 10:13:49 -07:00
|
|
|
|
---
|
|
|
|
|
id: 5e601bf95ac9d0ecd8b94afd
|
|
|
|
|
title: 數獨求解器
|
|
|
|
|
challengeType: 4
|
2021-06-05 00:07:40 +09:00
|
|
|
|
forumTopicId: 462357
|
2021-05-05 10:13:49 -07:00
|
|
|
|
dashedName: sudoku-solver
|
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
# --description--
|
|
|
|
|
|
|
|
|
|
構建一個 JavaScript 的全棧應用,在功能上與這個應用相似:<https://sudoku-solver.freecodecamp.rocks/>。 可以採用下面的任意一種方式完成這個挑戰:
|
|
|
|
|
|
|
|
|
|
- 克隆 [GitHub 倉庫](https://github.com/freecodecamp/boilerplate-project-sudoku-solver) 並在本地完成你的項目。
|
2021-05-12 21:25:58 +05:30
|
|
|
|
- 使用[我們的 Replit 初始化項目](https://replit.com/github/freeCodeCamp/boilerplate-project-sudoku-solver)來完成你的項目。
|
2021-05-05 10:13:49 -07:00
|
|
|
|
- 使用一個你喜歡的站點生成器來完成項目。 需要確定包含了我們 GitHub 倉庫的所有文件。
|
|
|
|
|
|
|
|
|
|
完成本項目後,請將一個正常運行的 demo(項目演示)託管在可以公開訪問的平臺。 然後在 `Solution Link` 框中提交你的項目 URL。 此外,還可以將項目的源碼提交到 `GitHub Link` 中。
|
|
|
|
|
|
|
|
|
|
# --instructions--
|
|
|
|
|
|
|
|
|
|
- 所有解謎邏輯都可以進入 `/controllers/sudoku-solver.js`
|
|
|
|
|
- `validate` 函數應該使用給定的解謎字符串,然後檢查它是否是 81 個有效的輸入字符。
|
|
|
|
|
- `check` 函數應對棋盤的 *current* 進行驗證。
|
|
|
|
|
- `solve` 函數應該處理任何給定的解謎字符串,而不僅僅是測試輸入和解決方法。 你需要寫出解決這個問題的邏輯。
|
|
|
|
|
- 所有路由邏輯都可以進入 `/routes/api.js`
|
|
|
|
|
- 閱讀 `/controllers` 中的 `puzzle-strings.js` 文件來了解一些應用程序應該解決的示例謎題
|
|
|
|
|
- 在 `.env` 文件中將 `NODE_ENV` 設置爲 `test` (沒有引號),運行這個頁面的挑戰測試。
|
2021-05-12 21:25:58 +05:30
|
|
|
|
- 使用 `npm run test` 命令在 console 中運行測試。 按 Ctrl+Shift+P(在 Mac 上是 Cmd+Shift+P),並輸入“open shell”,打開 Replit 控制檯。
|
2021-05-05 10:13:49 -07:00
|
|
|
|
|
|
|
|
|
在 `tests/1_unit-tests.js` 中寫下以下測試:
|
|
|
|
|
|
|
|
|
|
- 邏輯處理 81 個字符的解謎字符串
|
|
|
|
|
- 邏輯處理無效的解謎字符串 (不是 1-9 或 `.`)
|
|
|
|
|
- 邏輯處理一個長度不是 81 個字符的解謎字符串
|
|
|
|
|
- 邏輯處理有效行的位置
|
|
|
|
|
- 邏輯處理無效行的位置
|
|
|
|
|
- 邏輯處理一個有效的列位置
|
|
|
|
|
- 邏輯處理無效列位置
|
|
|
|
|
- 邏輯處理一個有效的區域 (3x3 網格)
|
|
|
|
|
- 邏輯處理一個無效的區域 (3x3 網格)
|
|
|
|
|
- 有效解謎字符串通過 solver
|
|
|
|
|
- 無效解謎字符串無法通過 solver
|
2021-06-28 20:01:36 +05:30
|
|
|
|
- Solver 返回一個不完整謎題的的預期解決方案
|
2021-05-05 10:13:49 -07:00
|
|
|
|
|
|
|
|
|
在 `tests/2_functional-tests.js` 中編寫下以下測試:
|
|
|
|
|
|
|
|
|
|
- 用有效的解謎字符串解決一個謎題:POST 請求到 `/api/solve`
|
|
|
|
|
- 用缺失的解謎字符串解決一個謎題:POST 請求到 `/api/solve`
|
|
|
|
|
- 用無效字符解決一個謎題:POST 請求到 `/api/solve`
|
|
|
|
|
- 用不正確的長度解決一個謎題:POST 請求到 `/api/solve`
|
|
|
|
|
- 解決一個無法解決的謎題:POST 請求到 `/api/solve`
|
|
|
|
|
- 檢查所有字段的解謎位置:POST 請求到 `/api/check`
|
|
|
|
|
- 用單個位置衝突檢查解謎位置:POST 請求到 `/api/check`
|
|
|
|
|
- 檢查一個有多個位置衝突的解謎位置: POST 請求到 `/api/check`
|
|
|
|
|
- 檢查與所有位置衝突的解謎位置: POST 請求到 `/api/check`
|
|
|
|
|
- 檢查缺失所需字段的解謎位置:POST 請求到 `/api/check`
|
|
|
|
|
- 檢查一個有無效字符的解謎位置: POST 請求到 `/api/check`
|
|
|
|
|
- 檢查不正確長度的解謎位置:POST 請求到 `/api/check`
|
|
|
|
|
- 檢查一個無效的放置座標的解謎位置:POST 請求到 `/api/check`
|
|
|
|
|
- 檢查具有無效的放置值的解謎位置:POST 請求到 `/api/check`
|
|
|
|
|
|
|
|
|
|
# --hints--
|
|
|
|
|
|
|
|
|
|
提交自己的項目,而不是示例的 URL。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
(getUserInput) => {
|
|
|
|
|
const url = getUserInput('url');
|
|
|
|
|
assert(!/.*\/sudoku-solver\.freecodecamp\.rocks/.test(getUserInput('url')));
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
可以發送 `POST` 請求到 `/api/solve`,使用包含 `puzzle` 的表單數據這將是一個包含數字 (1-9) 和點號的字符串組合,`.` 表示空格。 返回的對象將包含一個 `solution` 屬性與解決的謎題。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const output =
|
|
|
|
|
'769235418851496372432178956174569283395842761628713549283657194516924837947381625';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/solve', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'solution');
|
|
|
|
|
assert.equal(parsed.solution, output);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/solve` 的對象缺失 `puzzle`,返回的值將是 `{ error: 'Required field missing' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const output = 'Required field missing';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/solve', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ notpuzzle: input })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/solve` 謎題包含非數字或點號的值。 返回的值將是 `{ error: 'Invalid characters in puzzle' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'AA9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const output = 'Invalid characters in puzzle';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/solve', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/solve` 的謎題大於或小於 81 個字符, 返回的值將是 `{ error: 'Expected puzzle to be 81 characters long' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
2022-02-25 00:11:18 +05:30
|
|
|
|
const inputs = [
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6.',
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6...'
|
|
|
|
|
];
|
2021-05-05 10:13:49 -07:00
|
|
|
|
const output = 'Expected puzzle to be 81 characters long';
|
2022-02-25 00:11:18 +05:30
|
|
|
|
for (const input of inputs) {
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/solve', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
}
|
2021-05-05 10:13:49 -07:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/solve` 的謎題無效或無法解決, 返回的值將是 `{ error: 'Puzzle cannot be solved' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'9.9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const output = 'Puzzle cannot be solved';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/solve', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
可以發送 `POST` 請求到 `/api/check`,包含 `puzzle`、`coordinate` 和 `value` 的對象,其中 `coordinate` 是表示行的字母 A-I,後跟表示列的數字 1-9,而 `value` 是 1-9 的數字。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const coordinate = 'A1';
|
|
|
|
|
const value = '7';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'valid');
|
|
|
|
|
assert.isTrue(parsed.valid);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
發送 `POST` 請求到 `/api/check`,返回值是一個包含 `valid` 屬性的對象,如果數字可能放置在提供的座標中則是 `true`,否則是`false`。 如果錯誤,返回的對象還將包含一個 `conflict` 屬性,它是一個字符串 `"row"`、`"column"`, 和/或 取決於哪個區域使位置無效的`"region"` 。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const coordinate = 'A1';
|
|
|
|
|
const value = '1';
|
|
|
|
|
const conflict = ['row', 'column'];
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'valid');
|
|
|
|
|
assert.isFalse(parsed.valid);
|
|
|
|
|
assert.property(parsed, 'conflict');
|
|
|
|
|
assert.include(parsed.conflict, 'row');
|
|
|
|
|
assert.include(parsed.conflict, 'column');
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/check` 的 `value` 已放置在該 `coordinate` 上的 `puzzle`中,如果 `value` 不衝突,則返回的是 `valid` 屬性爲 `true` 的對象。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const coordinate = 'C3';
|
|
|
|
|
const value = '2';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'valid');
|
|
|
|
|
assert.isTrue(parsed.valid);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/check` 的謎題包含非數字或點號的值。 返回的值將是 `{ error: 'Invalid characters in puzzle' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'AA9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const coordinate = 'A1';
|
|
|
|
|
const value = '1';
|
|
|
|
|
const output = 'Invalid characters in puzzle';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/check` 的謎題大於或小於 81 個字符, 返回的值將是 `{ error: 'Expected puzzle to be 81 characters long' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
2022-02-25 00:11:18 +05:30
|
|
|
|
const inputs = [
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6.',
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6...'
|
|
|
|
|
];
|
2021-05-05 10:13:49 -07:00
|
|
|
|
const coordinate = 'A1';
|
|
|
|
|
const value = '1';
|
|
|
|
|
const output = 'Expected puzzle to be 81 characters long';
|
2022-02-25 00:11:18 +05:30
|
|
|
|
for (const input of inputs) {
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
}
|
2021-05-05 10:13:49 -07:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/check` 的對象缺失 `puzzle`、`coordinate` 或 `value`,返回的值將是 `{ error: Required field(s) missing }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
2022-02-25 00:11:18 +05:30
|
|
|
|
const inputs = [
|
|
|
|
|
{
|
|
|
|
|
puzzle: '..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..',
|
|
|
|
|
value: '1',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
puzzle: '..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..',
|
|
|
|
|
coordinate: 'A1',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
coordinate: 'A1',
|
|
|
|
|
value: '1'
|
|
|
|
|
}
|
|
|
|
|
];
|
|
|
|
|
for (const input of inputs) {
|
|
|
|
|
const output = 'Required field(s) missing';
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify(input)
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
}
|
2021-05-05 10:13:49 -07:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `api/check` 的座標不指向現有的網格單元格, 返回的值將是 `{ error: 'Invalid coordinate'}`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const output = 'Invalid coordinate';
|
2022-02-25 00:11:18 +05:30
|
|
|
|
const coordinates = ['A0', 'A10', 'J1', 'A', '1', 'XZ18'];
|
2021-05-05 10:13:49 -07:00
|
|
|
|
const value = '7';
|
2022-02-25 00:11:18 +05:30
|
|
|
|
for (const coordinate of coordinates) {
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
}
|
2021-05-05 10:13:49 -07:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
如果提交給 `/api/check` 的 `value` 不是一個介於 1 到 9 之間的數字,則返回的值將是 `{ error: 'Invalid value' }`
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
const input =
|
|
|
|
|
'..9..5.1.85.4....2432......1...69.83.9.....6.62.71...9......1945....4.37.4.3..6..';
|
|
|
|
|
const output = 'Invalid value';
|
|
|
|
|
const coordinate = 'A1';
|
2022-02-25 00:11:18 +05:30
|
|
|
|
const values = ['0', '10', 'A'];
|
|
|
|
|
for (const value of values) {
|
|
|
|
|
const data = await fetch(getUserInput('url') + '/api/check', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
|
body: JSON.stringify({ puzzle: input, coordinate, value })
|
|
|
|
|
});
|
|
|
|
|
const parsed = await data.json();
|
|
|
|
|
assert.property(parsed, 'error');
|
|
|
|
|
assert.equal(parsed.error, output);
|
|
|
|
|
}
|
2021-05-05 10:13:49 -07:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
所有 12 個單元的測試都已完成並通過。 請參閱 `/tests/1_unit-tests.js` 來了解你應該寫的測試的預期行爲。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
try {
|
|
|
|
|
const getTests = await $.get(getUserInput('url') + '/_api/get-tests');
|
|
|
|
|
assert.isArray(getTests);
|
|
|
|
|
const units = getTests.filter((el) => el.context.includes('UnitTests'));
|
|
|
|
|
assert.isAtLeast(units.length, 12, 'At least 12 tests passed');
|
|
|
|
|
units.forEach((test) => {
|
|
|
|
|
assert.equal(test.state, 'passed', 'Test in Passed State');
|
|
|
|
|
assert.isAtLeast(
|
|
|
|
|
test.assertions.length,
|
|
|
|
|
1,
|
|
|
|
|
'At least one assertion per test'
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
} catch (err) {
|
|
|
|
|
throw new Error(err.responseText || err.message);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
所有 14 項功能測試都已完成並通過。 請參閱 `/tests/2_functional-tests.js` 來了解你應該編寫的測試的功能。
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
async (getUserInput) => {
|
|
|
|
|
try {
|
|
|
|
|
const getTests = await $.get(getUserInput('url') + '/_api/get-tests');
|
|
|
|
|
assert.isArray(getTests);
|
|
|
|
|
const funcs = getTests.filter((el) =>
|
|
|
|
|
el.context.includes('Functional Tests')
|
|
|
|
|
);
|
|
|
|
|
assert.isAtLeast(funcs.length, 14, 'At least 14 tests passed');
|
|
|
|
|
funcs.forEach((test) => {
|
|
|
|
|
assert.equal(test.state, 'passed', 'Test in Passed State');
|
|
|
|
|
assert.isAtLeast(
|
|
|
|
|
test.assertions.length,
|
|
|
|
|
1,
|
|
|
|
|
'At least one assertion per test'
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
} catch (err) {
|
|
|
|
|
throw new Error(err.responseText || err.message);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
# --solutions--
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
/**
|
|
|
|
|
Backend challenges don't need solutions,
|
|
|
|
|
because they would need to be tested against a full working project.
|
|
|
|
|
Please check our contributing guidelines to learn more.
|
|
|
|
|
*/
|
|
|
|
|
```
|