Files
freeCodeCamp/curriculum/challenges/chinese-traditional/06-quality-assurance/quality-assurance-projects/sudoku-solver.md
2022-02-25 03:41:18 +09:00

15 KiB
Raw Blame History

id, title, challengeType, forumTopicId, dashedName
id title challengeType forumTopicId dashedName
5e601bf95ac9d0ecd8b94afd 數獨求解器 4 462357 sudoku-solver

--description--

構建一個 JavaScript 的全棧應用,在功能上與這個應用相似:https://sudoku-solver.freecodecamp.rocks/。 可以採用下面的任意一種方式完成這個挑戰:

  • 克隆 GitHub 倉庫 並在本地完成你的項目。
  • 使用我們的 Replit 初始化項目來完成你的項目。
  • 使用一個你喜歡的站點生成器來完成項目。 需要確定包含了我們 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 (沒有引號),運行這個頁面的挑戰測試。
  • 使用 npm run test 命令在 console 中運行測試。 按 Ctrl+Shift+P在 Mac 上是 Cmd+Shift+P並輸入“open shell”打開 Replit 控制檯。

tests/1_unit-tests.js 中寫下以下測試:

  • 邏輯處理 81 個字符的解謎字符串
  • 邏輯處理無效的解謎字符串 (不是 1-9 或 .)
  • 邏輯處理一個長度不是 81 個字符的解謎字符串
  • 邏輯處理有效行的位置
  • 邏輯處理無效行的位置
  • 邏輯處理一個有效的列位置
  • 邏輯處理無效列位置
  • 邏輯處理一個有效的區域 (3x3 網格)
  • 邏輯處理一個無效的區域 (3x3 網格)
  • 有效解謎字符串通過 solver
  • 無效解謎字符串無法通過 solver
  • Solver 返回一個不完整謎題的的預期解決方案

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。

(getUserInput) => {
  const url = getUserInput('url');
  assert(!/.*\/sudoku-solver\.freecodecamp\.rocks/.test(getUserInput('url')));
};

可以發送 POST 請求到 /api/solve,使用包含 puzzle 的表單數據這將是一個包含數字 (1-9) 和點號的字符串組合,. 表示空格。 返回的對象將包含一個 solution 屬性與解決的謎題。

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' }

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' }

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' }

async (getUserInput) => {
  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...'
  ];
  const output = 'Expected puzzle to be 81 characters long';
  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);
  }
};

如果提交給 /api/solve 的謎題無效或無法解決, 返回的值將是 { error: 'Puzzle cannot be solved' }

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,包含 puzzlecoordinatevalue 的對象,其中 coordinate 是表示行的字母 A-I後跟表示列的數字 1-9value 是 1-9 的數字。

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"

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/checkvalue 已放置在該 coordinate 上的 puzzle中,如果 value 不衝突,則返回的是 valid 屬性爲 true 的對象。

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' }

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' }

async (getUserInput) => {
  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...'
  ];
  const coordinate = 'A1';
  const value = '1';
  const output = 'Expected puzzle to be 81 characters long';
  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);
  }
};

如果提交給 /api/check 的對象缺失 puzzlecoordinatevalue,返回的值將是 { error: Required field(s) missing }

async (getUserInput) => {
  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);
  }
};

如果提交給 api/check 的座標不指向現有的網格單元格, 返回的值將是 { error: 'Invalid coordinate'}

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';
  const coordinates = ['A0', 'A10', 'J1', 'A', '1', 'XZ18'];
  const value = '7';
  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);
  }
};

如果提交給 /api/checkvalue 不是一個介於 1 到 9 之間的數字,則返回的值將是 { error: 'Invalid value' }

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';
  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);
  }
};

所有 12 個單元的測試都已完成並通過。 請參閱 /tests/1_unit-tests.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 來了解你應該編寫的測試的功能。

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--

/**
  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.
*/