freeCodeCamp/curriculum/challenges/chinese/06-quality-assurance/quality-assurance-projects/american-british-translator.md

12 KiB
Raw Blame History

id title challengeType dashedName
5e601c0d5ac9d0ecd8b94afe 美式英语英式英语转换器 4 american-british-translator

--description--

构建一个 JavaScript 的全栈应用,在功能上与这个应用相似:https://american-british-translator.freecodecamp.rocks/。 可以采用下面的任意一种方式完成这个挑战:

当完成本项目,请确认有一个可以公开访问的正常运行 demo 。 然后将 URL 提交到 Solution Link 中。 此外,还可以将项目的源码提交到 GitHub Link 中。

--instructions--

  • 所有逻辑都可以进入 /components/translator.js
  • /routes/api.js 中完成 /api/translate 路由
  • tests/1_unit-tests.jstests/2_functional-tests.js 中创建所有 unit/functional 测试
  • 查看 /components 中的 JavaScript 文件以获取应用程序应该翻译的条款以及不同的拼写
  • .env 文件中将 NODE_ENV 设置为 test(没有引号),在 Replit 上运行测试。
  • 使用 npm run test 命令,在 console 运行测试。 按 Ctrl+Shift+P在 Mac 上是 Cmd+Shift+P并输入“open shell”打开 Replit 控制台。

tests/1_unit-tests.js 中写下以下测试:

  • Mangoes are my favorite fruit. 转换成英式英语。
  • I ate yogurt for breakfast. 转换成英式英语。
  • We had a party at my friend's condo. 转换成英式英语。
  • Can you toss this in the trashcan for me? 转换成英式英语。
  • The parking lot was full. 转换成英式英语。
  • Like a high tech Rube Goldberg machine. 转换成英式英语。
  • To play hooky means to skip class or work. 转换成英式英语。
  • No Mr. Bond, I expect you to die. 转换成英式英语。
  • Dr. Grosh will see you now. 转换成英式英语。
  • Lunch is at 12:15 today. 转换成英式英语。
  • We watched the footie match for a while. 转换成美式英语。
  • Paracetamol takes up to an hour to work. 转换成美式英语。
  • First, caramelise the onions. 转换成美式英语。
  • I spent the bank holiday at the funfair. 转换成美式英语。
  • I had a bicky then went to the chippy. 转换成美式英语。
  • I've just got bits and bobs in my bum bag. 转换成美式英语。
  • The car boot sale at Boxted Airfield was called off. 转换成美式英语。
  • Have you met Mrs Kalyani? 转换成美式英语。
  • Prof Joyner of King's College, London. 转换成美式英语。
  • Tea time is usually around 4 or 4.30. 转换成美式英语。
  • Mangoes are my favorite fruit. 里的转换高亮。
  • 高亮 I ate yogurt for breakfast. 里的转换。
  • 高亮 We watched the footie match for a while. 里的转换。
  • 高亮 Paracetamol takes up to an hour to work. 里的转换。

tests/2_functional-tests.js 中写下以下测试:

  • 翻译文本字段和本地化字段: POST 请求到 /api/translate
  • 翻译文本字段和无效的本地化字段: POST 请求到 /api/translate
  • 翻译缺失的文本字段: POST 请求到 /api/translate
  • 翻译缺失的本地化字段: POST 请求到 /api/translate
  • 翻译空的文本: POST 请求到 /api/translate
  • 翻译无需翻译的文本: POST 请求到 /api/translate

--hints--

我可以提供我自己的项目,而不是示例 URL。

(getUserInput) => {
  assert(
    !/.*\/american-british-translator\.freecodecamp\.rocks/.test(
      getUserInput('url')
    )
  );
};

可以向 /api/translate 发送 POST 请求,对请求体内的 text 文本进行翻译, locale 字段可以是 american-to-britishbritish-to-american。 返回的对象应该包含提交的 text 以及翻译的文本 translation

async (getUserInput) => {
  try {
    const text = 'Mangoes are my favorite fruit.';
    const locale = 'american-to-british';
    const output = {
      text: 'Mangoes are my favorite fruit.',
      translation:
        'Mangoes are my <span class="highlight">favourite</span> fruit.'
    };
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text, locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'text');
    assert.property(parsed, 'translation');
    assert.deepEqual(parsed, output);
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

/api/translate 路由应该可以处理用英美方式英语写的时间。 例如十点半英式英语写为“10.30”而美式英语写为“10:30”。 span 元素应该包裹整个时间字符串,即 <span class="highlight">10:30</span>

async (getUserInput) => {
  try {
    const text = 'Lunch is at 12:15 today.';
    const locale = 'american-to-british';
    const output = {
      text: text,
      translation: 'Lunch is at <span class="highlight">12.15</span> today.'
    };
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text, locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'text');
    assert.property(parsed, 'translation');
    assert.deepEqual(parsed, output);
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

/api/translate 路由也应该处理美式英语和英式英语中头衔/尊称的缩写方式。 例如Doctor Wright 在英式英语中缩写为 “Dr Wright”在美式英语中缩写为 “Dr. Wright"”。 查看 /public/american-to-british-titles.js 了应用程序应该处理的不同的头衔。

async (getUserInput) => {
  try {
    const text = 'Dr. Grosh will see you now.';
    const locale = 'american-to-british';
    const output = {
      text: text,
      translation: '<span class="highlight">Dr</span> Grosh will see you now.'
    };
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text, locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'text');
    assert.property(parsed, 'translation');
    assert.deepEqual(parsed, output);
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

将任何翻译过的拼写或条目放在 <span class="highlight">...</span> 标签内以使其显示为绿色。

async (getUserInput) => {
  try {
    const text = 'Mangoes are my favorite fruit.';
    const locale = 'american-to-british';
    const output = {
      text: 'Mangoes are my favorite fruit.',
      translation:
        'Mangoes are my <span class="highlight">favourite</span> fruit.'
    };
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text, locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'text');
    assert.property(parsed, 'translation');
    assert.deepEqual(parsed, output);
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

如果缺少一个或多个必填字段,返回 { error: 'Required field(s) missing' }

async (getUserInput) => {
  try {
    const locale = 'american-to-british';
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'error');
    assert.equal(parsed.error, 'Required field(s) missing');
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

如果 text 为空,返回 { error: 'No text to translate' }

async (getUserInput) => {
  try {
    const locale = 'american-to-british';
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text: '', locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'error');
    assert.equal(parsed.error, 'No text to translate');
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

如果 locale 与两个指定的 locales 都不匹配,返回 { error: 'Invalid value for locale field' }

async (getUserInput) => {
  try {
    const text = "Ceci n'est pas une pipe";
    const locale = 'french-to-american';
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text, locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.property(parsed, 'error');
    assert.equal(parsed.error, 'Invalid value for locale field');
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

如果 text 不需要翻译,返回的 translation 值为"Everything looks good to me!"

async (getUserInput) => {
  try {
    const locale = 'british-to-american';
    const output = {
      text: 'SaintPeter and nhcarrigan give their regards!',
      translation: 'Everything looks good to me!'
    };
    let data = await fetch(getUserInput('url') + '/api/translate', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ text: output.text, locale })
    });
    let parsed = await data.json();
    assert.isObject(parsed);
    assert.isObject(parsed);
    assert.property(parsed, 'text');
    assert.property(parsed, 'translation');
    assert.deepEqual(parsed, output);
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

所有 24 个单元的测试都已完成并通过。 请参阅 /tests/1_unit-tests.js 来了解你应该写的测试的预期行为。

async (getUserInput) => {
  try {
    const getTests = await $.get(getUserInput('url') + '/_api/get-tests');
    assert.isArray(getTests);
    const unitTests = getTests.filter((test) => {
      return !!test.context.match(/Unit Tests/gi);
    });
    assert.isAtLeast(unitTests.length, 24, 'At least 24 tests passed');
    unitTests.forEach((test) => {
      assert.equal(test.state, 'passed', 'Tests in Passed State');
      assert.isAtLeast(
        test.assertions.length,
        1,
        'At least one assertion per test'
      );
    });
  } catch (err) {
    throw new Error(err.responseText || err.message);
  }
};

所有 6 项功能测试都已完成并通过。 请参阅 /tests/2_functional-tests.js 来了解你应该写的测试的功能。

async (getUserInput) => {
  try {
    const getTests = await $.get(getUserInput('url') + '/_api/get-tests');
    assert.isArray(getTests);
    const functTests = getTests.filter((test) => {
      return !!test.context.match(/Functional Tests/gi);
    });
    assert.isAtLeast(functTests.length, 6, 'At least 6 tests passed');
    functTests.forEach((test) => {
      assert.equal(test.state, 'passed', 'Tests 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.
*/