From 30818226b6ae3b7ba6647fa871a715ee8c47d00a Mon Sep 17 00:00:00 2001 From: yaroslav-epam <138673581+yaroslav-epam@users.noreply.github.com> Date: Tue, 7 Nov 2023 14:39:22 +0200 Subject: [PATCH] MODSOURMAN-1021: add endpoint to get IncomingRecord by id with DAO and service layers functionality (#823) --- NEWS.md | 1 + descriptors/ModuleDescriptor-template.json | 15 ++++++ .../java/org/folio/dao/IncomingRecordDao.java | 9 ++++ .../org/folio/dao/IncomingRecordDaoImpl.java | 29 +++++++++++- .../folio/rest/impl/MetadataProviderImpl.java | 23 +++++++++ .../folio/services/IncomingRecordService.java | 10 ++++ .../services/IncomingRecordServiceImpl.java | 7 +++ .../folio/dao/IncomingRecordDaoImplTest.java | 47 ++++++++++++++----- .../org/folio/rest/impl/AbstractRestTest.java | 16 ++++--- .../MetadataProviderJobExecutionAPITest.java | 41 ++++++++++++++++ .../IncomingRecordServiceImplUnitTest.java | 8 +++- ramls/metadata-provider.raml | 14 ++++++ 12 files changed, 200 insertions(+), 20 deletions(-) diff --git a/NEWS.md b/NEWS.md index 5adc98b5c..0fd6ae2bb 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,6 +1,7 @@ ## 2023-xx-xx v3.8.0-SNAPSHOT * [MODSOURMAN-1085](https://issues.folio.org/browse/MODSOURMAN-1085) MARC record with a 100 tag without a $a is being discarded on import. * [MODSOURMAN-1020](https://issues.folio.org/browse/MODSOURMAN-1020) Add table to save incoming records for DI logs +* [MODSOURMAN-1021](https://issues.folio.org/browse/MODSOURMAN-1021) Provide endpoint for getting parsed content for DI log * [MODSOURMAN-1030](https://issues.folio.org/browse/MODSOURMAN-1030) The number of updated records is not correct displayed in the 'SRS Marc' column in the 'Log summary' table * [MODSOURMAN-976](https://issues.folio.org/browse/MODSOURMAN-976) Incorrect error counts * [MODSOURMAN-1093](https://issues.folio.org/browse/MODSOURMAN-1093) EventHandlingUtil hangs forever on error diff --git a/descriptors/ModuleDescriptor-template.json b/descriptors/ModuleDescriptor-template.json index e9d14a4ca..d011f5534 100644 --- a/descriptors/ModuleDescriptor-template.json +++ b/descriptors/ModuleDescriptor-template.json @@ -501,6 +501,15 @@ "permissionsRequired": [ "metadata-provider.jobexecutions.get" ] + }, + { + "methods": [ + "GET" + ], + "pathPattern": "/metadata-provider/incomingRecords/{recordId}", + "permissionsRequired": [ + "metadata-provider.incomingrecords.get" + ] } ] }, @@ -651,6 +660,11 @@ "displayName": "Metadata Provider - get jobExecution logs", "description": "Get JobExecutionLogDto" }, + { + "permissionName": "metadata-provider.incomingrecords.get", + "displayName": "Metadata Provider - get incoming record", + "description": "Get IncomingRecord" + }, { "permissionName": "change-manager.jobexecutions.post", "displayName": "Change Manager - create jobExecutions", @@ -718,6 +732,7 @@ "subPermissions": [ "metadata-provider.jobexecutions.get", "metadata-provider.logs.get", + "metadata-provider.incomingrecords.get", "change-manager.jobexecutions.post", "change-manager.jobexecutions.put", "change-manager.jobexecutions.get", diff --git a/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDao.java b/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDao.java index 062f98584..abc33e5be 100644 --- a/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDao.java +++ b/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDao.java @@ -6,12 +6,21 @@ import org.folio.rest.jaxrs.model.IncomingRecord; import java.util.List; +import java.util.Optional; /** * DAO interface for the {@link IncomingRecord} entity */ public interface IncomingRecordDao { + /** + * Searches for {@link IncomingRecord} by id + * + * @param id incomingRecord id + * @return optional of incomingRecord + */ + Future> getById(String id, String tenantId); + /** * Saves {@link IncomingRecord} entities into DB * diff --git a/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDaoImpl.java b/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDaoImpl.java index 6b69139a4..966c3be40 100644 --- a/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDaoImpl.java +++ b/mod-source-record-manager-server/src/main/java/org/folio/dao/IncomingRecordDaoImpl.java @@ -14,6 +14,7 @@ import org.springframework.stereotype.Repository; import java.util.List; +import java.util.Optional; import java.util.UUID; import static java.lang.String.format; @@ -24,14 +25,30 @@ public class IncomingRecordDaoImpl implements IncomingRecordDao { private static final Logger LOGGER = LogManager.getLogger(); public static final String INCOMING_RECORDS_TABLE = "incoming_records"; + private static final String GET_BY_ID_SQL = "SELECT * FROM %s.%s WHERE id = $1"; private static final String INSERT_SQL = "INSERT INTO %s.%s (id, job_execution_id, incoming_record) VALUES ($1, $2, $3)"; @Autowired private PostgresClientFactory pgClientFactory; + @Override + public Future> getById(String id, String tenantId) { + LOGGER.debug("getById:: Get IncomingRecord by id = {} from the {} table", id, INCOMING_RECORDS_TABLE); + Promise> promise = Promise.promise(); + try { + String query = format(GET_BY_ID_SQL, convertToPsqlStandard(tenantId), INCOMING_RECORDS_TABLE); + pgClientFactory.createInstance(tenantId).selectRead(query, Tuple.of(UUID.fromString(id)), promise); + } catch (Exception e) { + LOGGER.warn("getById:: Error getting IncomingRecord by id", e); + promise.fail(e); + } + return promise.future().map(rowSet -> rowSet.rowCount() == 0 ? Optional.empty() + : Optional.of(mapRowToIncomingRecord(rowSet.iterator().next()))); + } + @Override public Future>> saveBatch(List incomingRecords, String tenantId) { - LOGGER.info("saveBatch:: Save IncomingRecord entity to the {} table", INCOMING_RECORDS_TABLE); + LOGGER.debug("saveBatch:: Save IncomingRecord entity to the {} table", INCOMING_RECORDS_TABLE); Promise>> promise = Promise.promise(); try { String query = format(INSERT_SQL, convertToPsqlStandard(tenantId), INCOMING_RECORDS_TABLE); @@ -45,6 +62,16 @@ public Future>> saveBatch(List incomingRecords, return promise.future().onFailure(e -> LOGGER.warn("saveBatch:: Error saving IncomingRecord entity", e)); } + private IncomingRecord mapRowToIncomingRecord(Row row) { + JsonObject jsonObject = row.getJsonObject("incoming_record"); + return new IncomingRecord().withId(String.valueOf(row.getUUID("id"))) + .withJobExecutionId(String.valueOf(row.getUUID("job_execution_id"))) + .withRecordType(IncomingRecord.RecordType.fromValue(jsonObject.getString("recordType"))) + .withOrder(jsonObject.getInteger("order")) + .withRawRecordContent(jsonObject.getString("rawRecordContent")) + .withParsedRecordContent(jsonObject.getString("parsedRecordContent")); + } + private Tuple prepareInsertQueryParameters(IncomingRecord incomingRecord) { return Tuple.of(UUID.fromString(incomingRecord.getId()), UUID.fromString(incomingRecord.getJobExecutionId()), JsonObject.mapFrom(incomingRecord)); diff --git a/mod-source-record-manager-server/src/main/java/org/folio/rest/impl/MetadataProviderImpl.java b/mod-source-record-manager-server/src/main/java/org/folio/rest/impl/MetadataProviderImpl.java index bd3473605..8a91fd01e 100644 --- a/mod-source-record-manager-server/src/main/java/org/folio/rest/impl/MetadataProviderImpl.java +++ b/mod-source-record-manager-server/src/main/java/org/folio/rest/impl/MetadataProviderImpl.java @@ -19,6 +19,7 @@ import org.folio.rest.jaxrs.model.MetadataProviderJobLogEntriesJobExecutionIdGetEntityType; import org.folio.rest.jaxrs.resource.MetadataProvider; import org.folio.rest.tools.utils.TenantTool; +import org.folio.services.IncomingRecordService; import org.folio.services.JobExecutionService; import org.folio.services.JournalRecordService; import org.folio.spring.SpringContextUtil; @@ -51,6 +52,8 @@ public class MetadataProviderImpl implements MetadataProvider { private JobExecutionService jobExecutionService; @Autowired private JournalRecordService journalRecordService; + @Autowired + private IncomingRecordService incomingRecordService; private String tenantId; public MetadataProviderImpl(Vertx vertx, String tenantId) { //NOSONAR @@ -197,6 +200,26 @@ public void getMetadataProviderJobExecutionsUsers(String totalRecords, int offse }); } + @Override + public void getMetadataProviderIncomingRecordsByRecordId(String recordId, Map okapiHeaders, Handler> asyncResultHandler, Context vertxContext) { + vertxContext.runOnContext(v -> { + try { + LOGGER.debug("getMetadataProviderIncomingRecordsByRecordId:: tenantId {}", tenantId); + incomingRecordService.getById(recordId, tenantId) + .map(incomingRecordOptional -> incomingRecordOptional + .map(GetMetadataProviderIncomingRecordsByRecordIdResponse::respond200WithApplicationJson) + .orElseGet(() -> GetMetadataProviderIncomingRecordsByRecordIdResponse + .respond404WithTextPlain(format("IncomingRecord by id: '%s' was not found", recordId)))) + .map(Response.class::cast) + .otherwise(ExceptionHelper::mapExceptionToResponse) + .onComplete(asyncResultHandler); + } catch (Exception e) { + LOGGER.warn("getMetadataProviderIncomingRecordsByRecordId:: Failed to retrieve IncomingRecord by id", e); + asyncResultHandler.handle(Future.succeededFuture(ExceptionHelper.mapExceptionToResponse(e))); + } + }); + } + private JobExecutionFilter buildJobExecutionFilter(List statusAny, List profileIdNotAny, String statusNot, List uiStatusAny, String hrIdPattern, String fileNamePattern, List fileNameNotAny, List profileIdAny, List subordinationTypeNotAny, diff --git a/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordService.java b/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordService.java index b2f8623db..e4900251c 100644 --- a/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordService.java +++ b/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordService.java @@ -1,14 +1,24 @@ package org.folio.services; +import io.vertx.core.Future; import org.folio.rest.jaxrs.model.IncomingRecord; import java.util.List; +import java.util.Optional; /** * {@link IncomingRecord} Service interface */ public interface IncomingRecordService { + /** + * Searches for {@link IncomingRecord} by id + * + * @param id incomingRecord id + * @return future with optional incomingRecord + */ + Future> getById(String id, String tenantId); + /** * Saves {@link IncomingRecord}s into DB * diff --git a/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordServiceImpl.java b/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordServiceImpl.java index 0358f8372..ebf60452f 100644 --- a/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordServiceImpl.java +++ b/mod-source-record-manager-server/src/main/java/org/folio/services/IncomingRecordServiceImpl.java @@ -1,11 +1,13 @@ package org.folio.services; +import io.vertx.core.Future; import org.folio.dao.IncomingRecordDao; import org.folio.rest.jaxrs.model.IncomingRecord; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import java.util.List; +import java.util.Optional; @Service public class IncomingRecordServiceImpl implements IncomingRecordService { @@ -13,6 +15,11 @@ public class IncomingRecordServiceImpl implements IncomingRecordService { @Autowired private IncomingRecordDao incomingRecordDao; + @Override + public Future> getById(String id, String tenantId) { + return incomingRecordDao.getById(id, tenantId); + } + @Override public void saveBatch(List incomingRecords, String tenantId) { incomingRecordDao.saveBatch(incomingRecords, tenantId); diff --git a/mod-source-record-manager-server/src/test/java/org/folio/dao/IncomingRecordDaoImplTest.java b/mod-source-record-manager-server/src/test/java/org/folio/dao/IncomingRecordDaoImplTest.java index 3a9e83ca3..132c18da3 100644 --- a/mod-source-record-manager-server/src/test/java/org/folio/dao/IncomingRecordDaoImplTest.java +++ b/mod-source-record-manager-server/src/test/java/org/folio/dao/IncomingRecordDaoImplTest.java @@ -7,7 +7,6 @@ import org.folio.dao.util.PostgresClientFactory; import org.folio.rest.impl.AbstractRestTest; import org.folio.rest.jaxrs.model.IncomingRecord; -import org.folio.rest.jaxrs.model.InitJobExecutionsRsDto; import org.folio.rest.jaxrs.model.JobExecution; import org.junit.Before; import org.junit.Test; @@ -37,21 +36,41 @@ public void setUp(TestContext context) throws IOException { } @Test - public void saveBatch(TestContext context) { + public void shouldGetById(TestContext context) { Async async = context.async(); - InitJobExecutionsRsDto response = constructAndPostInitJobExecutionRqDto(1); - List createdJobExecutions = response.getJobExecutions(); - final String jobExecutionId = createdJobExecutions.get(0).getId(); + List createdJobExecutions = constructAndPostInitJobExecutionRqDto(1).getJobExecutions(); + String jobExecutionId = createdJobExecutions.get(0).getId(); + + String id = UUID.randomUUID().toString(); + IncomingRecord incomingRecord = buildIncomingRecord(id, jobExecutionId); + + incomingRecordDao.saveBatch(List.of(incomingRecord), TENANT_ID) + .compose(r -> + incomingRecordDao.getById(id, TENANT_ID) + .onComplete(ar -> { + context.assertTrue(ar.succeeded()); + context.assertTrue(ar.result().isPresent()); + IncomingRecord result = ar.result().get(); + context.assertEquals(id, result.getId()); + context.assertEquals(jobExecutionId, result.getJobExecutionId()); + context.assertEquals("rawRecord", result.getRawRecordContent()); + context.assertEquals("parsedRecord", result.getParsedRecordContent()); + async.complete(); + })); + } + + @Test + public void shouldSaveBatch(TestContext context) { + Async async = context.async(); + + List createdJobExecutions = constructAndPostInitJobExecutionRqDto(1).getJobExecutions(); + String jobExecutionId = createdJobExecutions.get(0).getId(); String id1 = UUID.randomUUID().toString(); String id2 = UUID.randomUUID().toString(); - IncomingRecord incomingRecord1 = new IncomingRecord() - .withId(id1).withJobExecutionId(jobExecutionId).withRecordType(IncomingRecord.RecordType.MARC_BIB).withOrder(0) - .withRawRecordContent("rawRecord").withParsedRecordContent("parsedRecord"); - IncomingRecord incomingRecord2 = new IncomingRecord() - .withId(id2).withJobExecutionId(jobExecutionId).withRecordType(IncomingRecord.RecordType.MARC_BIB).withOrder(0) - .withRawRecordContent("rawRecord").withParsedRecordContent("parsedRecord"); + IncomingRecord incomingRecord1 = buildIncomingRecord(id1, jobExecutionId); + IncomingRecord incomingRecord2 = buildIncomingRecord(id2, jobExecutionId); incomingRecordDao.saveBatch(List.of(incomingRecord1, incomingRecord2), TENANT_ID) .onComplete(ar -> { @@ -60,4 +79,10 @@ public void saveBatch(TestContext context) { async.complete(); }); } + + private static IncomingRecord buildIncomingRecord(String id, String jobExecutionId) { + return new IncomingRecord() + .withId(id).withJobExecutionId(jobExecutionId).withRecordType(IncomingRecord.RecordType.MARC_BIB).withOrder(0) + .withRawRecordContent("rawRecord").withParsedRecordContent("parsedRecord"); + } } diff --git a/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/AbstractRestTest.java b/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/AbstractRestTest.java index a8bb46d95..82a18488d 100644 --- a/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/AbstractRestTest.java +++ b/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/AbstractRestTest.java @@ -73,6 +73,7 @@ import static com.github.tomakehurst.wiremock.client.WireMock.post; import static net.mguenther.kafka.junit.EmbeddedKafkaCluster.provisionWith; import static net.mguenther.kafka.junit.EmbeddedKafkaClusterConfig.defaultClusterConfig; +import static org.folio.dao.IncomingRecordDaoImpl.INCOMING_RECORDS_TABLE; import static org.folio.dataimport.util.RestUtil.OKAPI_TENANT_HEADER; import static org.folio.dataimport.util.RestUtil.OKAPI_URL_HEADER; import static org.folio.kafka.KafkaTopicNameHelper.getDefaultNameSpace; @@ -504,13 +505,14 @@ private void clearTable(TestContext context) { PostgresClient pgClient = PostgresClient.getInstance(vertx, TENANT_ID); pgClient.delete(CHUNKS_TABLE_NAME, new Criterion(), event1 -> pgClient.delete(JOURNAL_RECORDS_TABLE, new Criterion(), event2 -> - pgClient.delete(JOB_EXECUTION_PROGRESS_TABLE, new Criterion(), event3 -> - pgClient.delete(JOB_EXECUTIONS_TABLE_NAME, new Criterion(), event4 -> { - if (event3.failed()) { - context.fail(event3.cause()); - } - async.complete(); - })))); + pgClient.delete(INCOMING_RECORDS_TABLE, new Criterion(), event3 -> + pgClient.delete(JOB_EXECUTION_PROGRESS_TABLE, new Criterion(), event4 -> + pgClient.delete(JOB_EXECUTIONS_TABLE_NAME, new Criterion(), event5 -> { + if (event4.failed()) { + context.fail(event4.cause()); + } + async.complete(); + }))))); } protected InitJobExecutionsRsDto constructAndPostInitJobExecutionRqDto(int filesNumber) { diff --git a/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/metadataProvider/MetadataProviderJobExecutionAPITest.java b/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/metadataProvider/MetadataProviderJobExecutionAPITest.java index 18942bac7..431c13d40 100644 --- a/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/metadataProvider/MetadataProviderJobExecutionAPITest.java +++ b/mod-source-record-manager-server/src/test/java/org/folio/rest/impl/metadataProvider/MetadataProviderJobExecutionAPITest.java @@ -9,6 +9,7 @@ import io.vertx.ext.unit.TestContext; import io.vertx.ext.unit.junit.VertxUnitRunner; import org.apache.http.HttpStatus; +import org.folio.dao.IncomingRecordDaoImpl; import org.folio.dao.JournalRecordDaoImpl; import org.folio.dao.util.PostgresClientFactory; import org.folio.rest.impl.AbstractRestTest; @@ -16,6 +17,7 @@ import org.folio.rest.jaxrs.model.DeleteJobExecutionsReq; import org.folio.rest.jaxrs.model.DeleteJobExecutionsResp; import org.folio.rest.jaxrs.model.EntityType; +import org.folio.rest.jaxrs.model.IncomingRecord; import org.folio.rest.jaxrs.model.InitJobExecutionsRsDto; import org.folio.rest.jaxrs.model.JobExecution; import org.folio.rest.jaxrs.model.JobExecutionDto; @@ -95,6 +97,7 @@ public class MetadataProviderJobExecutionAPITest extends AbstractRestTest { private static final String GET_JOB_EXECUTION_SUMMARY_PATH = "/metadata-provider/jobSummary"; private static final String GET_JOB_EXECUTION_JOB_PROFILES_PATH = "/metadata-provider/jobExecutions/jobProfiles"; private static final String GET_UNIQUE_USERS_INFO = "/metadata-provider/jobExecutions/users"; + private static final String GET_INCOMING_RECORDS_BY_ID = "/metadata-provider/incomingRecords/"; private final JsonObject userResponse = new JsonObject() .put("users", @@ -108,6 +111,9 @@ public class MetadataProviderJobExecutionAPITest extends AbstractRestTest { @Spy @InjectMocks private JournalRecordDaoImpl journalRecordDao; + @Spy + @InjectMocks + private IncomingRecordDaoImpl incomingRecordDao; private AutoCloseable mocks; @Before @@ -1743,4 +1749,39 @@ public void shouldNotReturnUsersForParentJobExecutions() { .body("totalRecords", is(0)); } + @Test + public void shouldReturnNotFoundIncomingRecordById() { + RestAssured.given() + .spec(spec) + .when() + .get(GET_INCOMING_RECORDS_BY_ID + UUID.randomUUID()) + .then() + .statusCode(HttpStatus.SC_NOT_FOUND); + } + + @Test + public void shouldReturnIncomingRecordById(TestContext context) { + Async async = context.async(); + List createdJobExecutions = constructAndPostInitJobExecutionRqDto(1).getJobExecutions(); + JobExecution jobExecution = createdJobExecutions.get(0); + String jobExecutionId = jobExecution.getId(); + String id = UUID.randomUUID().toString(); + + IncomingRecord incomingRecord = new IncomingRecord() + .withId(id).withJobExecutionId(jobExecutionId).withRecordType(IncomingRecord.RecordType.MARC_BIB).withOrder(0) + .withRawRecordContent("rawRecord").withParsedRecordContent("parsedRecord"); + + incomingRecordDao.saveBatch(List.of(incomingRecord), TENANT_ID) + .onComplete(v -> { + RestAssured.given() + .spec(spec) + .when() + .get(GET_INCOMING_RECORDS_BY_ID + id) + .then() + .statusCode(HttpStatus.SC_OK) + .body("id", is(id)) + .body("jobExecutionId", is(jobExecutionId)); + async.complete(); + }); + } } diff --git a/mod-source-record-manager-server/src/test/java/org/folio/services/IncomingRecordServiceImplUnitTest.java b/mod-source-record-manager-server/src/test/java/org/folio/services/IncomingRecordServiceImplUnitTest.java index 505d636b0..9118b25f8 100644 --- a/mod-source-record-manager-server/src/test/java/org/folio/services/IncomingRecordServiceImplUnitTest.java +++ b/mod-source-record-manager-server/src/test/java/org/folio/services/IncomingRecordServiceImplUnitTest.java @@ -27,7 +27,13 @@ public void setUp() { } @Test - public void saveBatch() { + public void shouldGetById() { + incomingRecordService.getById(any(), any()); + verify(incomingRecordDao).getById(any(), any()); + } + + @Test + public void shouldSaveBatch() { incomingRecordService.saveBatch(any(), any()); verify(incomingRecordDao).saveBatch(any(), any()); } diff --git a/ramls/metadata-provider.raml b/ramls/metadata-provider.raml index 370ab6500..dcb51ade4 100644 --- a/ramls/metadata-provider.raml +++ b/ramls/metadata-provider.raml @@ -23,6 +23,7 @@ types: jobExecutionSummaryDto: !include raml-storage/schemas/dto/jobExecutionSummaryDto.json jobProfileInfoCollection: !include raml-storage/schemas/common/profileInfoCollection.json jobExecutionUserInfoCollection: !include raml-storage/schemas/dto/jobExecutionUserInfoCollection.json + incomingRecord: !include raml-storage/schemas/mod-source-record-manager/incomingRecord.json traits: validate: !include raml-storage/raml-util/traits/validation.raml @@ -242,4 +243,17 @@ resourceTypes: body: text/plain: example: "Internal server error" + /incomingRecords/{recordId}: + get: + description: get incoming record by id + responses: + 200: + body: + application/json: + type: incomingRecord + 404: + description: "Not found" + body: + text/plain: + example: "Not found"