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

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

Коли ви завершили, переконайтеся, що ця демоверсія вашого проєкту розміщена у відкритому місці. Потім введіть URL-адресу у поле Solution Link. При необхідності, також введіть посилання на джерело коду вашого проєкту у полі GitHub Link.

--instructions--

  • Уся логіка головоломки може ввійти в /controllers/sudoku-solver.js
    • Функція validate повинна взяти вказаний рядок головоломки й перевірити його, чи є в ньому 81 допустимий символ для входу.
    • Функції check повинні бути перевіреними відповідно до статусу current на сторінці.
    • Функція solve повинна опрацьовувати вирішення будь-якого заданого рядка допустимої головоломки, а не тільки входу та вирішенню тесту. Щоб вирішити це, ви маєте розшифрувати логіку.
  • Вся логіка маршрутизації може переміститися в /routes/api.js
  • Подивіться на файл puzzle-strings.js в /controllers для деяких зразків головоломок, які має розв'язати ваша програма
  • Щоб запустити завдання на цій сторінці, налаштуйте NODE_ENV на test без лапок у файлі .env
  • Щоб запустити завдання, використовуйте команду npm run test. Щоб відкрити консоль Replit, натисніть Ctrl+Shift+P (Cmd, якщо з Mac) та наберіть "open shell"

Напишіть наступні тести в tests/1_unit-tests.js:

  • Логіка обробляє допустимий рядок з 81 символом
  • Логіка опрацьовує рядок головоломки з доступними символами (не 1-9 чи .)
  • Логіка опрацьовує рядок головоломки, який не містить 81 символ
  • Логіка опрацьовує допустиме розміщення рядка
  • Логіка опрацьовує недопустиме розміщення рядка
  • Логіка опрацьовує допустиме розміщення стовпця
  • Логіка опрацьовує недопустиме розміщення стовпця
  • Логіка опрацьовує допустиме розміщення області (сітка 3x3)
  • Логіка опрацьовує недопустиме розміщення області (сітка 3x3)
  • Допустимі рядки головоломки передають вирішення
  • Недопустимі рядки головоломки не передають вирішення
  • Розв'язувач повертає очікуване рішення для незавершеного завдання

Напишіть наступні тести в 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, який містить puzzle, coordinate, і value, де coordinate - букви A-I, які позначають рядок, потім числа 1-9, які позначають стовпець і value - числа 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", і /or "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');
};

Якщо value представлене в /api/check уже поміщене в puzzle на ту coordinate, то зворотне значення буде об'єкт, в якому є властивість valid з true, якщо 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 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 немає puzzle, coordinate чи value, то зворотне значення буде { 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);
  }
};

Якщо value представлене в /api/check не є числом між 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.
*/