1
0
mirror of https://github.com/twirl/The-API-Book.git synced 2025-01-05 10:20:22 +02:00

Fresh builds with Annotation and Annex to Section I added

This commit is contained in:
Sergey Konstantinov 2021-01-05 17:03:42 +03:00
parent 0377e344ee
commit f3b18c6ae7
6 changed files with 337 additions and 30 deletions

Binary file not shown.

View File

@ -240,6 +240,16 @@ a.anchor {
 ·
<a href="https://www.linkedin.com/in/twirl/">https://www.linkedin.com/in/twirl/</a>
</p>
<p>
The API-first development is one of the hottest technical topics in
2020, since many companies started to realize that API serves as a
multiplicator to their opportunities—but it also amplifies the design
mistakes as well.
</p>
<p>
The book is dedicated to designing APIs: how to build the architecture
properly, from a high-level planning down to final interfaces.
</p>
<p>Illustrations by Maria Konstantinova</p>
<img class="cc-by-nc-img" src="">
@ -265,7 +275,7 @@ a.anchor {
<path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path>
<path d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2" fill="currentColor" style="transform-origin: 130px 106px" class="octo-arm"></path>
<path d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z" fill="currentColor" class="octo-body"></path></svg></a>
<div class="page-break"></div><nav><h2 class="toc">Table of Contents</h2><ul class="table-of-contents"><li><a href="#section-1">Introduction</a><ul><li><a href="#chapter-1">Chapter 1. On the Structure of This Book</a></li><li><a href="#chapter-2">Chapter 2. The API Definition</a></li><li><a href="#chapter-3">Chapter 3. API Quality Criteria</a></li><li><a href="#chapter-4">Chapter 4. Backwards Compatibility</a></li><li><a href="#chapter-5">Chapter 5. On versioning</a></li><li><a href="#chapter-6">Chapter 6. Terms and Notation Keys</a></li></ul></li><li><a href="#section-2">Section I. The API Design</a><ul><li><a href="#chapter-7">Chapter 7. The API Contexts Pyramid</a></li><li><a href="#chapter-8">Chapter 8. Defining an Application Field</a></li><li><a href="#chapter-9">Chapter 9. Separating Abstraction Levels</a></li><li><a href="#chapter-10">Chapter 10. Isolating Responsibility Areas</a></li><li><a href="#chapter-11">Chapter 11. Describing Final Interfaces</a></li></ul></li></ul></nav><div class="page-break"></div><h2><a href="#section-1" class="anchor" name="section-1">Introduction</a></h2><h3><a href="#chapter-1" class="anchor" name="chapter-1">Chapter 1. On the Structure of This Book</a></h3>
<div class="page-break"></div><nav><h2 class="toc">Table of Contents</h2><ul class="table-of-contents"><li><a href="#section-1">Introduction</a><ul><li><a href="#chapter-1">Chapter 1. On the Structure of This Book</a></li><li><a href="#chapter-2">Chapter 2. The API Definition</a></li><li><a href="#chapter-3">Chapter 3. API Quality Criteria</a></li><li><a href="#chapter-4">Chapter 4. Backwards Compatibility</a></li><li><a href="#chapter-5">Chapter 5. On Versioning</a></li><li><a href="#chapter-6">Chapter 6. Terms and Notation Keys</a></li></ul></li><li><a href="#section-2">Section I. The API Design</a><ul><li><a href="#chapter-7">Chapter 7. The API Contexts Pyramid</a></li><li><a href="#chapter-8">Chapter 8. Defining an Application Field</a></li><li><a href="#chapter-9">Chapter 9. Separating Abstraction Levels</a></li><li><a href="#chapter-10">Chapter 10. Isolating Responsibility Areas</a></li><li><a href="#chapter-11">Chapter 11. Describing Final Interfaces</a></li><li><a href="#chapter-12">Chapter 12. Annex to Section I. Generic API Example</a></li></ul></li></ul></nav><div class="page-break"></div><h2><a href="#section-1" class="anchor" name="section-1">Introduction</a></h2><h3><a href="#chapter-1" class="anchor" name="chapter-1">Chapter 1. On the Structure of This Book</a></h3>
<p>The book you're holding in your hands comprises this Introduction and three large sections.</p>
<p>In Section I we'll discuss designing APIs as a concept: how to build the architecture properly, from a high-level planning down to final interfaces.</p>
<p>Section II is dedicated to an API's lifecycle: how interfaces evolve over time, and how to elaborate the product to match users' needs.</p>
@ -304,7 +314,7 @@ a.anchor {
<p>Large companies, which occupy firm market positions, could afford implying such a taxation. Furthermore, they may introduce penalties for those who refuse to adapt their code to new API versions, up to disabling their applications.</p>
<p>From our point of view such practice cannot be justified. Don't imply hidden taxes on your customers. If you're able to avoid breaking backwards compatibility — never break it.</p>
<p>Of course, maintaining old API versions is a sort of a tax either. Technology changes, and you cannot foresee everything, regardless of how nice your API is initially designed. At some point keeping old API versions results in an inability to provide new functionality and support new platforms, and you will be forced to release new version. But at least you will be able to explain to your customers why they need to make an effort.</p>
<p>We will discuss API lifecycle and version policies in Section II.</p><div class="page-break"></div><h3><a href="#chapter-5" class="anchor" name="chapter-5">Chapter 5. On versioning</a></h3>
<p>We will discuss API lifecycle and version policies in Section II.</p><div class="page-break"></div><h3><a href="#chapter-5" class="anchor" name="chapter-5">Chapter 5. On Versioning</a></h3>
<p>Here and throughout we firmly stick to <a href="https://semver.org/">semver</a> principles of versioning:</p>
<ol>
<li>API versions are denoted with three numbers, i.e. <code>1.2.3</code>.</li>
@ -647,7 +657,7 @@ GET /sensors
{ "order_id" }
</code></pre>
<p>The <code>POST /orders</code> handler checks all order parameters, puts a hold of corresponding sum on user's credit card, forms a request to run, and calls the execution level. First, correct execution program needs to be fetched:</p>
<pre><code>POST /v1/programs/match
<pre><code>POST /v1/program-matcher
{ "recipe", "coffee-machine" }
{ "program_id" }
@ -669,7 +679,7 @@ GET /sensors
</code></pre>
<p>Please note that knowing the coffee machine API kind isn't required at all; that's why we're making abstractions! We could possibly make interfaces more specific, implementing different <code>run</code> and <code>match</code> endpoints for different coffee machines:</p>
<ul>
<li><code>POST /v1/programs/{api_type}/match</code></li>
<li><code>POST /v1/program-matcher/{api_type}</code></li>
<li><code>POST /v1/programs/{api_type}/{program_id}/run</code></li>
</ul>
<p>This approach has some benefits, like a possibility to provide different sets of parameters, specific to the API kind. But we see no need in such fragmentation. <code>run</code> method handler is capable of extracting all the program metadata and perform one of two actions:</p>
@ -885,7 +895,7 @@ let recipes = api.getRecipes();
// Retrieve a list of all available coffee machines
let coffeeMachines = api.getCoffeeMachines();
// Build a spatial index
let coffeeMachineRecipesIndex = buildGeoIndex(recipes, coffee-machines);
let coffeeMachineRecipesIndex = buildGeoIndex(recipes, coffeeMachines);
// Select coffee machines matching user's needs
let matchingCoffeeMachines = coffeeMachineRecipesIndex.query(
parameters,
@ -901,7 +911,7 @@ app.display(coffeeMachines);
<li>display nearby cafes where a user could order a particular type of coffee — for users seeking a certain beverage type.</li>
</ul>
<p>Then our new interface would look like:</p>
<pre><code>POST /v1/coffee-machines/search
<pre><code>POST /v1/offers/search
{
// optional
"recipes": ["lungo", "americano"],
@ -924,16 +934,16 @@ app.display(coffeeMachines);
<li>an <code>offer</code> — is a marketing bid: on what conditions a user could have the requested coffee beverage (if specified in request), or a some kind of marketing offering — prices for the most popular or interesting products (if no specific preference was set);</li>
<li>a <code>place</code> — is a spot (café, restaurant, street vending machine) where the coffee machine is located; we never introduced this entity before, but it's quite obvious that users need more convenient guidance to find a proper coffee machine than just geographical coordinates.</li>
</ul>
<p><strong>NB</strong>. We could have been enriched the existing <code>/coffee-machines</code> endpoint instead of adding a new one. This decision, however, looks less semantically viable: coupling in one interface different modes of listing entities, by relevance and by order, is usually a bad idea, because these two types of rankings implies different usage features and scenarios.</p>
<p><strong>NB</strong>. We could have been enriched the existing <code>/coffee-machines</code> endpoint instead of adding a new one. This decision, however, looks less semantically viable: coupling in one interface different modes of listing entities, by relevance and by order, is usually a bad idea, because these two types of rankings implies different usage features and scenarios. Furthermore, enriching the search with ‘offers’ pulls this functionality out of <code>coffee-machines</code> namespace: the fact of getting offers to prepare specific beverage in specific conditions is a key feature to users, with specifying the coffee-machine being just a part of an offer.</p>
<p>Coming back to the code developers are writing, it would now look like that:</p>
<pre><code>// Searching for coffee machines
<pre><code>// Searching for offers
// matching a user's intent
let coffeeMachines = api.search(parameters);
let offers = api.search(parameters);
// Display them to a user
app.display(coffeeMachines);
app.display(offers);
</code></pre>
<h4>Helpers</h4>
<p>Methods similar to newly invented <code>coffee-machines/search</code> are called <em>helpers</em>. The purpose they exist is to generalize known API usage scenarios and facilitate implementing them. By ‘facilitating’ we mean not only reducing wordiness (getting rid of ‘boilerplates’), but also helping developers to avoid common problems and mistakes.</p>
<p>Methods similar to newly invented <code>offers/search</code> are called <em>helpers</em>. The purpose they exist is to generalize known API usage scenarios and facilitate implementing them. By ‘facilitating’ we mean not only reducing wordiness (getting rid of ‘boilerplates’), but also helping developers to avoid common problems and mistakes.</p>
<p>For instance, let's consider the order price question. Our search function returns some ‘offers’ with prices. But ‘price’ is volatile; coffee could cost less during ‘happy hours’, for example. Developers could make a mistake thrice while implementing this functionality:</p>
<ul>
<li>cache search results on a client device for too long (as a result, the price will always be nonactual);</li>
@ -1055,7 +1065,7 @@ The invalid price error is resolvable: client could obtain a new price offer and
</ul>
<p>Let's try to group it together:</p>
<pre><code>{
"results": {
"results": [{
// Place data
"place": { "name", "location" },
// Coffee machine properties
@ -1073,7 +1083,7 @@ The invalid price error is resolvable: client could obtain a new price offer and
"pricing": { "currency_code", "price", "localized_price" },
"estimated_waiting_time"
}
}
}, …]
}
</code></pre>
<p>Such decomposed API is much easier to read than a long sheet of different attributes. Furthermore, it's probably better to group even more entities in advance. For example, <code>place</code> and <code>route</code> could be joined in a single <code>location</code> structure, or <code>offer</code> and <code>pricing</code> might be combined into a some generalized object.</p>
@ -1895,6 +1905,147 @@ POST /v1/orders
<p>Sometimes explicit location passing is not enough since there are lots of territorial conflicts in a world. How the API should behave when user coordinates lie within disputed regions is a legal matter, regretfully. Author of this books once had to implement a ‘state A territory according to state B official position’ concept.</p>
<p><strong>Important</strong>: mark a difference between localization for end users and localization for developers. Take a look at the example in #12 rule: <code>localized_message</code> is meant for the user; the app should show it if there is no specific handler for this error exists in code. This message must be written in user's language and formatted according to user's location. But <code>details.checks_failed[].message</code> is meant to be read by developers examining the problem. So it must be written and formatted in a manner which suites developers best. In a software development world it usually means ‘in English’.</p>
<p>Worth mentioning is that <code>localized_</code> prefix in the example is used to differentiate messages to users from messages to developers. A concept like that must be, of course, explicitly stated in your API docs.</p>
<p>And one more thing: all strings must be UTF-8, no exclusions.</p><div class="page-break"></div>
<p>And one more thing: all strings must be UTF-8, no exclusions.</p><div class="page-break"></div><h3><a href="#chapter-12" class="anchor" name="chapter-12">Chapter 12. Annex to Section I. Generic API Example</a></h3>
<p>Let's summarize the current state of our API study.</p>
<h5><a href="#chapter-12-paragraph-1" name="chapter-12-paragraph-1" class="anchor">1. Offer search</a></h5>
<pre><code>POST /v1/offers/search
{
// optional
"recipes": ["lungo", "americano"],
"position": &#x3C;geographical coordinates>,
"sort_by": [
{ "field": "distance" }
],
"limit": 10
}
{
"results": [{
// Place data
"place": { "name", "location" },
// Coffee machine properties
"coffee-machine": { "brand", "type" },
// Route data
"route": { "distance", "duration", "location_tip" },
"offers": {
// Recipe data
"recipe": { "id", "name", "description" },
// Recipe specific options
"options": { "volume" },
// Offer metadata
"offer": { "id", "valid_until" },
// Pricing
"pricing": { "currency_code", "price", "localized_price" },
"estimated_waiting_time"
}
}, …],
"cursor"
}
</code></pre>
<h5><a href="#chapter-12-paragraph-2" name="chapter-12-paragraph-2" class="anchor">2. Working with recipes</a></h5>
<pre><code>// Returns a list of recipes
// Cursor parameter is optional
GET /v1/recipes?cursor=&#x3C;cursor>
{ "recipes", "cursor" }
</code></pre>
<pre><code>// Returns the recipe by its id
GET /v1/recipes/{id}
{ "recipe_id", "name", "description" }
</code></pre>
<h5><a href="#chapter-12-paragraph-3" name="chapter-12-paragraph-3" class="anchor">3. Working with orders</a></h5>
<pre><code>// Creates an order
POST /v1/orders
{
"coffee_machine_id",
"currency_code",
"price",
"recipe": "lungo",
// Optional
"offer_id",
// Optional
"volume": "800ml"
}
{ "order_id" }
</code></pre>
<pre><code>// Returns the order by its id
GET /v1/orders/{id}
{ "order_id", "status" }
</code></pre>
<pre><code>// Cancels the order
POST /v1/orders/{id}/cancel
</code></pre>
<h5><a href="#chapter-12-paragraph-4" name="chapter-12-paragraph-4" class="anchor">4. Working with programs</a></h5>
<pre><code>// Returns an identifier of the program
// corresponding to specific recipe
// on specific coffee-machine
POST /v1/program-matcher
{ "recipe", "coffee-machine" }
{ "program_id" }
</code></pre>
<pre><code>// Return program description
// by its id
GET /v1/programs/{id}
{
"program_id",
"api_type",
"commands": [
{
"sequence_id",
"type": "set_cup",
"parameters"
},
]
}
</code></pre>
<h5><a href="#chapter-12-paragraph-5" name="chapter-12-paragraph-5" class="anchor">5. Running programs</a></h5>
<pre><code>// Runs the specified program
// on the specefied coffee-machine
// with specific parameters
POST /v1/programs/{id}/run
{
"order_id",
"coffee_machine_id",
"parameters": [
{
"name": "volume",
"value": "800ml"
}
]
}
{ "program_run_id" }
</code></pre>
<pre><code>// Stops program running
POST /v1/runs/{id}/cancel
</code></pre>
<h5><a href="#chapter-12-paragraph-6" name="chapter-12-paragraph-6" class="anchor">6. Managing runtimes</a></h5>
<pre><code>// Creates a new runtime
POST /v1/runtimes
{ "coffee_machine_id", "program_id", "parameters" }
{ "runtime_id", "state" }
</code></pre>
<pre><code>// Returns the state
// of the specified runtime
GET /v1/runtimes/{runtime_id}/state
{
"status": "ready_waiting",
// Command being currently executed
// (optional)
"command_sequence_id",
"resolution": "success",
"variables"
}
</code></pre>
<pre><code>// Terminates the runtime
POST /v1/runtimes/{id}/terminate
</code></pre><div class="page-break"></div>
</div></article>
</body></html>

Binary file not shown.

Binary file not shown.

View File

@ -240,6 +240,17 @@ a.anchor {
 ·
<a href="https://www.linkedin.com/in/twirl/">https://www.linkedin.com/in/twirl/</a>
</p>
<p>
«API-first» подход — одна из самых горячих горячих тем в разработке
программного обеспечения в 2020. Многие компании начали понимать, что
API выступает мультипликатором их возможностей — но также умножает и
допущенные ошибки.
</p>
<p>
Эта книга посвящена проектированию API: как правильно выстроить
архитектуру, начиная с высокоуровневого планирования из заканчивая
деталями реализации конкретных интерфейсов.
</p>
<p>Иллюстрации: Мария Константинова</p>
<img class="cc-by-nc-img" src="">
@ -264,7 +275,7 @@ a.anchor {
<path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path>
<path d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2" fill="currentColor" style="transform-origin: 130px 106px" class="octo-arm"></path>
<path d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z" fill="currentColor" class="octo-body"></path></svg></a>
<div class="page-break"></div><nav><h2 class="toc">Содержание</h2><ul class="table-of-contents"><li><a href="#section-1">Введение</a><ul><li><a href="#chapter-1">Глава 1. О структуре этой книги</a></li><li><a href="#chapter-2">Глава 2. Определение API</a></li><li><a href="#chapter-3">Глава 3. Критерии качества API</a></li><li><a href="#chapter-4">Глава 4. Обратная совместимость</a></li><li><a href="#chapter-5">Глава 5. О версионировании</a></li><li><a href="#chapter-6">Глава 6. Условные обозначения и терминология</a></li></ul></li><li><a href="#section-2">Раздел I. Проектирование API</a><ul><li><a href="#chapter-7">Глава 7. Пирамида контекстов API</a></li><li><a href="#chapter-8">Глава 8. Определение области применения</a></li><li><a href="#chapter-9">Глава 9. Разделение уровней абстракции</a></li><li><a href="#chapter-10">Глава 10. Разграничение областей ответственности</a></li><li><a href="#chapter-11">Глава 11. Описание конечных интерфейсов</a></li></ul></li></ul></nav><div class="page-break"></div><h2><a href="#section-1" class="anchor" name="section-1">Введение</a></h2><h3><a href="#chapter-1" class="anchor" name="chapter-1">Глава 1. О структуре этой книги</a></h3>
<div class="page-break"></div><nav><h2 class="toc">Содержание</h2><ul class="table-of-contents"><li><a href="#section-1">Введение</a><ul><li><a href="#chapter-1">Глава 1. О структуре этой книги</a></li><li><a href="#chapter-2">Глава 2. Определение API</a></li><li><a href="#chapter-3">Глава 3. Критерии качества API</a></li><li><a href="#chapter-4">Глава 4. Обратная совместимость</a></li><li><a href="#chapter-5">Глава 5. О версионировании</a></li><li><a href="#chapter-6">Глава 6. Условные обозначения и терминология</a></li></ul></li><li><a href="#section-2">Раздел I. Проектирование API</a><ul><li><a href="#chapter-7">Глава 7. Пирамида контекстов API</a></li><li><a href="#chapter-8">Глава 8. Определение области применения</a></li><li><a href="#chapter-9">Глава 9. Разделение уровней абстракции</a></li><li><a href="#chapter-10">Глава 10. Разграничение областей ответственности</a></li><li><a href="#chapter-11">Глава 11. Описание конечных интерфейсов</a></li><li><a href="#chapter-12">Глава 12. Приложение к разделу I. Модельное API</a></li></ul></li></ul></nav><div class="page-break"></div><h2><a href="#section-1" class="anchor" name="section-1">Введение</a></h2><h3><a href="#chapter-1" class="anchor" name="chapter-1">Глава 1. О структуре этой книги</a></h3>
<p>Книга, которую вы держите в руках, состоит из введения и трех больших разделов.</p>
<p>В первом разделе мы поговорим о проектировании API на стадии разработки концепции — как грамотно выстроить архитектуру, от крупноблочного планирования до конечных интерфейсов.</p>
<p>Второй раздел будет посвящён жизненному циклу API — как интерфейсы эволюционируют со временем и как развивать продукт так, чтобы отвечать потребностям пользователей.</p>
@ -639,7 +650,7 @@ GET /sensors
{ "order_id" }
</code></pre>
<p>Имплементация функции <code>POST /orders</code> проверит все параметры заказа, заблокирует его стоимость на карте пользователя, сформирует полный запрос на исполнение и обратится к уровню исполнения. Сначала необходимо подобрать правильную программу исполнения:</p>
<pre><code>POST /v1/programs/match
<pre><code>POST /v1/program-matcher
{ "recipe", "coffee-machine" }
{ "program_id" }
@ -661,7 +672,7 @@ GET /sensors
</code></pre>
<p>Обратите внимание, что во всей этой цепочке вообще никак не участвует тип API кофе-машины — собственно, ровно для этого мы и абстрагировали. Мы могли бы сделать интерфейсы более конкретными, разделив функциональность <code>run</code> и <code>match</code> для разных API, т.е. ввести раздельные endpoint-ы:</p>
<ul>
<li><code>POST /v1/programs/{api_type}/match</code></li>
<li><code>POST /v1/program-matcher/{api_type}</code></li>
<li><code>POST /v1/programs/{api_type}/{program_id}/run</code></li>
</ul>
<p>Достоинством такого подхода была бы возможность передавать в match и run не унифицированные наборы параметров, а только те, которые имеют значение в контексте указанного типа API. Однако в нашем дизайне API такой необходимости не прослеживается. Обработчик <code>run</code> сам может извлечь нужные параметры из мета-информации о программе и выполнить одно из двух действий:</p>
@ -873,7 +884,7 @@ GET /sensors
<p>Очевидно, первый шаг — нужно предоставить пользователю возможность выбора, чего он, собственно хочет. И первый же шаг обнажает неудобство использования нашего API: никаких методов, позволяющих пользователю что-то выбрать в нашем API нет. Разработчику придётся сделать что-то типа такого:</p>
<ul>
<li>получить все доступные рецепты из <code>GET /v1/recipes</code>;</li>
<li>получить список всех кофе-машины из <code>GET /v1/coffee-machines</code>;</li>
<li>получить список всех кофе-машин из <code>GET /v1/coffee-machines</code>;</li>
<li>самостоятельно выбрать нужные данные.</li>
</ul>
<p>В псевдокоде это будет выглядеть примерно вот так:</p>
@ -882,7 +893,7 @@ let recipes = api.getRecipes();
// Получить все доступные кофе-машины
let coffeeMachines = api.getCoffeeMachines();
// Построить пространственный индекс
let coffeeMachineRecipesIndex = buildGeoIndex(recipes, coffee-machines);
let coffeeMachineRecipesIndex = buildGeoIndex(recipes, coffeeMachines);
// Выбрать кофе-машины, соответствующие запросу пользователя
let matchingCoffeeMachines = coffeeMachineRecipesIndex.query(
parameters,
@ -898,7 +909,7 @@ app.display(coffeeMachines);
<li>показать ближайшие кофейни, где можно заказать конкретный вид кофе — для пользователей, которым нужен конкретный напиток.</li>
</ul>
<p>Тогда наш новый интерфейс будет выглядеть примерно вот так:</p>
<pre><code>POST /v1/coffee-machines/search
<pre><code>POST /v1/offers/search
{
// опционально
"recipes": ["lungo", "americano"],
@ -921,15 +932,16 @@ app.display(coffeeMachines);
<li><code>offer</code> — некоторое «предложение»: на каких условиях можно заказать запрошенные виды кофе, если они были указаны, либо какое-то маркетинговое предложение — цены на самые популярные / интересные напитки, если пользователь не указал конкретные рецепты для поиска;</li>
<li><code>place</code> — место (кафе, автомат, ресторан), где находится машина; мы не вводили эту сущность ранее, но, очевидно, пользователю потребуются какие-то более понятные ориентиры, нежели географические координаты, чтобы найти нужную кофе-машину.</li>
</ul>
<p><strong>NB</strong>. Мы могли бы не добавлять новый эндпойнт, а обогатить существующий <code>/coffee-machines</code>. Однако такое решение выглядит менее семантично: не стоит в рамках одного интерфейса смешивать способ перечисления объектов по порядку и по релевантности запросу, поскольку эти два вида ранжирования обладают существенно разными свойствами и сценариями использования.</p>
<p><strong>NB</strong>. Мы могли бы не добавлять новый эндпойнт, а обогатить существующий <code>/coffee-machines</code>. Однако такое решение выглядит менее семантично: не стоит в рамках одного интерфейса смешивать способ перечисления объектов по порядку и по релевантности запросу, поскольку эти два вида ранжирования обладают существенно разными свойствами и сценариями использования. К тому же, обогащение поиска «предложениями» скорее выводит эту функциональность из неймспейса «кофе-машины»: для пользователя всё-таки первичен факт получения предложения приготовить напиток на конкретных условиях, и кофе-машина — лишь одно из них. <code>/v1/offers/search</code> — более логичное имя для такого эндпойнта.</p>
<p>Вернёмся к коду, который напишет разработчик. Теперь он будет выглядеть примерно так:</p>
<pre><code>// Ищем кофе-машины, соответствующие запросу пользователя
let coffeeMachines = api.search(parameters);
<pre><code>// Ищем предложения,
// соответствующие запросу пользователя
let offers = api.offerSearch(parameters);
// Показываем пользователю
app.display(coffeeMachines);
app.display(offers);
</code></pre>
<h4>Хэлперы</h4>
<p>Методы, подобные только что изобретённому нами <code>coffee-machines/search</code>, принято называть <em>хэлперами</em>. Цель их существования — обобщить понятные сценарии использования API и облегчить их. Под «облегчить» мы имеем в виду не только сократить многословность («бойлерплейт»), но и помочь разработчику избежать частых проблем и ошибок.</p>
<p>Методы, подобные только что изобретённому нами <code>offers/search</code>, принято называть <em>хэлперами</em>. Цель их существования — обобщить понятные сценарии использования API и облегчить их. Под «облегчить» мы имеем в виду не только сократить многословность («бойлерплейт»), но и помочь разработчику избежать частых проблем и ошибок.</p>
<p>Рассмотрим, например, вопрос стоимости заказа. Наша функция поиска возвращает какие-то «предложения» с ценой. Но ведь цена может меняться: в «счастливый час» кофе может стоить меньше. Разработчик может ошибиться в имплементации этой функциональности трижды:</p>
<ul>
<li>кэшировать на клиентском устройстве результаты поиска слишком долго (в результате цена всегда будет неактуальна),</li>
@ -956,7 +968,7 @@ app.display(coffeeMachines);
}
</code></pre>
<p>Поступая так, мы не только помогаем разработчику понять, когда ему надо обновить цены, но и решаем UX-задачу: как показать пользователю, что «счастливый час» скоро закончится. Идентификатор предложения может при этом быть stateful (фактически, аналогом сессии пользователя) или stateless (если мы точно знаем, до какого времени действительна цены, мы может просто закодировать это время в идентификаторе).</p>
<p>Альтернативно, кстати, можно было бы разделить функциональность поиска по заданным параметрам и получения офферов, т.е. добавить эндпойнт, только актуализирующий цены в конкретных кофейнях.</p>
<p>Альтернативно, кстати, можно было бы разделить функциональность поиска по заданным параметрам и получения предложений, т.е. добавить эндпойнт, только актуализирующий цены в конкретных кофейнях.</p>
<h4>Обработка ошибок</h4>
<p>Сделаем ещё один небольшой шаг в сторону улучшения жизни разработчика. А каким образом будет выглядеть ошибка «неверная цена»?</p>
<pre><code>POST /v1/orders
@ -1058,7 +1070,7 @@ app.display(coffeeMachines);
</ul>
<p>Попробуем сгруппировать:</p>
<pre><code>{
"results": {
"results": [{
// Данные о заведении
"place": { "name", "location" },
// Данные о кофе-машине
@ -1078,7 +1090,7 @@ app.display(coffeeMachines);
"pricing": { "currency_code", "price", "localized_price" },
"estimated_waiting_time"
}
}
}, …]
}
</code></pre>
<p>Такое API читать и воспринимать гораздо удобнее, нежели сплошную простыню различных атрибутов. Более того, возможно, стоит на будущее сразу дополнительно сгруппировать, например, <code>place</code> и <code>route</code> в одну структуру <code>location</code>, или <code>offer</code> и <code>pricing</code> в одну более общую структуру.</p>
@ -1592,7 +1604,7 @@ GET /v1/recipes
}]
}
</code></pre>
<p>По сути, для клиента всё произошло ожидаемым образом: изменения были внесены, и последний полученный ответ всегда корректен. Однако по сути состояние ресурса после первого запросе отличалось от состояния ресурса после второго запроса, что противоречит самому определению идемпотентности.</p>
<p>По сути, для клиента всё произошло ожидаемым образом: изменения были внесены, и последний полученный ответ всегда корректен. Однако по сути состояние ресурса после первого запроса отличалось от состояния ресурса после второго запроса, что противоречит самому определению идемпотентности.</p>
<p>Более корректно было бы при получении повторного запроса с тем же токеном ничего не делать и возвращать ту же разбивку ошибок, что была дана на первый запрос — но для этого придётся её каким-то образом хранить в истории изменений.</p>
<p>На всякий случай уточним, что вложенные операции должны быть сами по себе идемпотентны. Если же это не так, то следует сгенерировать внутренние ключи идемпотентности на каждую вложенную операцию в отдельности.</p>
<h5><a href="#chapter-11-paragraph-15" name="chapter-11-paragraph-15" class="anchor">15. Указывайте политики кэширования</a></h5>
@ -1900,6 +1912,150 @@ POST /v1/orders
<p>Следует иметь в виду, что явной передачи локации может оказаться недостаточно, поскольку в мире существуют территориальные конфликты и спорные территории. Каким образом API должно себя вести при попадании координат пользователя на такие территории — вопрос, к сожалению, в первую очередь юридический. Автору этой книги приходилось как-то разрабатывать API, в котором пришлось вводить концепцию «территория государства A по мнению официальных органов государства Б».</p>
<p><strong>Важно</strong>: различайте локализацию для конечного пользователя и локализацию для разработчика. В примере из п. 12 сообщение <code>localized_message</code> адресовано пользователю — его должно показать приложение, если в коде обработка такой ошибки не предусмотрена. Это сообщение должно быть написано на указанном в запросе языке и отформатировано согласно правилам локации пользователя. А вот сообщение <code>details.checks_failed[].message</code> написано не для пользователя, а для разработчика, который будет разбираться с проблемой. Соответственно, написано и отформатировано оно должно быть понятным для разработчика образом — что, скорее всего, означает «на английском языке», т.к. английский де факто является стандартом в мире разработки программного обеспечения.</p>
<p>Следует отметить, что индикация, какие сообщения следует показать пользователю, а какие написаны для разработчика, должна, разумеется, быть явной конвенцией вашего API. В примере для этого используется префикс <code>localized_</code>.</p>
<p>И ещё одна вещь: все строки должны быть в кодировке UTF-8 и никакой другой.</p><div class="page-break"></div>
<p>И ещё одна вещь: все строки должны быть в кодировке UTF-8 и никакой другой.</p><div class="page-break"></div><h3><a href="#chapter-12" class="anchor" name="chapter-12">Глава 12. Приложение к разделу I. Модельное API</a></h3>
<p>Суммируем текущее состояние нашего учебного API.</p>
<h5><a href="#chapter-12-paragraph-1" name="chapter-12-paragraph-1" class="anchor">1. Поиск предложений</a></h5>
<pre><code>POST /v1/offers/search
{
// опционально
"recipes": ["lungo", "americano"],
"position": &#x3C;географические координаты>,
"sort_by": [
{ "field": "distance" }
],
"limit": 10
}
{
"results": [{
// Данные о заведении
"place": { "name", "location" },
// Данные о кофе-машине
"coffee-machine": { "brand", "type" },
// Как добраться
"route": { "distance", "duration", "location_tip" },
// Предложения напитков
"offers": {
// Рецепт
"recipe": { "id", "name", "description" },
// Данные относительно того,
// как рецепт готовят на конкретной кофе-машине
"options": { "volume" },
// Метаданные предложения
"offer": { "id", "valid_until" },
// Цена
"pricing": { "currency_code", "price", "localized_price" },
"estimated_waiting_time"
}
}, …]
"cursor"
}
</code></pre>
<h5><a href="#chapter-12-paragraph-2" name="chapter-12-paragraph-2" class="anchor">2. Работа с рецептами</a></h5>
<pre><code>// Возвращает список рецептов
// Параметр cursor необязателен
GET /v1/recipes?cursor=&#x3C;курсор>
{ "recipes", "cursor" }
</code></pre>
<pre><code>// Возвращает конкретный рецепт
// по его идентификатору
GET /v1/recipes/{id}
{ "recipe_id", "name", "description" }
</code></pre>
<h5><a href="#chapter-12-paragraph-3" name="chapter-12-paragraph-3" class="anchor">3. Работа с заказами</a></h5>
<pre><code>// Размещает заказ
POST /v1/orders
{
"coffee_machine_id",
"currency_code",
"price",
"recipe": "lungo",
// Опционально
"offer_id",
// Опционально
"volume": "800ml"
}
{ "order_id" }
</code></pre>
<pre><code>// Возвращает состояние заказа
GET /v1/orders/{id}
{ "order_id", "status" }
</code></pre>
<pre><code>// Отменяет заказ
POST /v1/orders/{id}/cancel
</code></pre>
<h5><a href="#chapter-12-paragraph-4" name="chapter-12-paragraph-4" class="anchor">4. Работа с программами</a></h5>
<pre><code>// Возвращает идентификатор программы,
// соответствующей указанному рецепту
// на указанной кофе-машине
POST /v1/program-matcher
{ "recipe", "coffee-machine" }
{ "program_id" }
</code></pre>
<pre><code>// Возвращает описание
// программы по её идентификатору
GET /v1/programs/{id}
{
"program_id",
"api_type",
"commands": [
{
"sequence_id",
"type": "set_cup",
"parameters"
},
]
}
</code></pre>
<h5><a href="#chapter-12-paragraph-5" name="chapter-12-paragraph-5" class="anchor">5. Исполнение программ</a></h5>
<pre><code>// Запускает исполнение программы
// с указанным идентификатором
// на указанной машине
// с указанными параметрами
POST /v1/programs/{id}/run
{
"order_id",
"coffee_machine_id",
"parameters": [
{
"name": "volume",
"value": "800ml"
}
]
}
{ "program_run_id" }
</code></pre>
<pre><code>// Останавливает исполнение программы
POST /v1/runs/{id}/cancel
</code></pre>
<h5><a href="#chapter-12-paragraph-6" name="chapter-12-paragraph-6" class="anchor">6. Управление рантаймами</a></h5>
<pre><code>// Создаёт новый рантайм
POST /v1/runtimes
{ "coffee_machine_id", "program_id", "parameters" }
{ "runtime_id", "state" }
</code></pre>
<pre><code>// Возвращает текущее состояние рантайма
// по его id
GET /v1/runtimes/{runtime_id}/state
{
"status": "ready_waiting",
// Текущая исполняемая команда (необязательное)
"command_sequence_id",
"resolution": "success",
"variables"
}
</code></pre>
<pre><code>// Прекращает исполнение рантайма
POST /v1/runtimes/{id}/terminate
</code></pre><div class="page-break"></div>
</div></article>
</body></html>

Binary file not shown.