From 5b8bb344d45f4d25b677b59c0c7d67e62b9ff5e7 Mon Sep 17 00:00:00 2001 From: monsieurtanuki Date: Sat, 7 Sep 2024 19:42:52 +0200 Subject: [PATCH] fix: 970 - refactored around new Prices API Impacted files: * `api_prices_test.dart`: refactored around new Prices API * `open_prices_api_client.dart`: different status code for "delete user session"; simplified `updatePrice` and `updateProof` * `price_user.dart`: field `isModerator` is now optional * `price_user.g.dart`: generated * `session.dart`: additional status codes and error messages --- lib/src/open_prices_api_client.dart | 24 +-- lib/src/prices/price_user.dart | 2 +- lib/src/prices/price_user.g.dart | 2 +- lib/src/prices/session.dart | 10 ++ test/api_prices_test.dart | 269 +++++++++++++--------------- 5 files changed, 148 insertions(+), 159 deletions(-) diff --git a/lib/src/open_prices_api_client.dart b/lib/src/open_prices_api_client.dart index f6f87ca87b..f5f6693910 100644 --- a/lib/src/open_prices_api_client.dart +++ b/lib/src/open_prices_api_client.dart @@ -315,7 +315,7 @@ class OpenPricesAPIClient { uriHelper: uriHelper, bearerToken: bearerToken, ); - if (response.statusCode == 200) { + if (response.statusCode == 204) { return MaybeError.value(true); } return MaybeError.responseError(response); @@ -369,7 +369,7 @@ class OpenPricesAPIClient { /// /// This endpoint requires authentication. /// A user can update only owned prices. - static Future> updatePrice( + static Future> updatePrice( final int priceId, { required final UpdatePriceParameters parameters, final UriProductHelper uriHelper = uriHelperFoodProd, @@ -388,14 +388,9 @@ class OpenPricesAPIClient { addUserAgentParameters: false, ); if (response.statusCode == 200) { - try { - final dynamic decodedResponse = HttpHelper().jsonDecodeUtf8(response); - return MaybeError.value(Price.fromJson(decodedResponse)); - } catch (e) { - // - } + return MaybeError.value(true); } - return MaybeError.responseError(response); + return MaybeError.responseError(response); } /// Deletes a price. @@ -540,7 +535,7 @@ class OpenPricesAPIClient { /// /// This endpoint requires authentication. /// A user can update only owned proofs. - static Future> updateProof( + static Future> updateProof( final int proofId, { required final UpdateProofParameters parameters, final UriProductHelper uriHelper = uriHelperFoodProd, @@ -559,14 +554,9 @@ class OpenPricesAPIClient { addUserAgentParameters: false, ); if (response.statusCode == 200) { - try { - final dynamic decodedResponse = HttpHelper().jsonDecodeUtf8(response); - return MaybeError.value(Proof.fromJson(decodedResponse)); - } catch (e) { - // - } + return MaybeError.value(true); } - return MaybeError.responseError(response); + return MaybeError.responseError(response); } /// Deletes a proof. diff --git a/lib/src/prices/price_user.dart b/lib/src/prices/price_user.dart index 9f2a6147eb..d96cde5f93 100644 --- a/lib/src/prices/price_user.dart +++ b/lib/src/prices/price_user.dart @@ -18,7 +18,7 @@ class PriceUser extends JsonObject { /// Number of prices for this user. @JsonKey(name: 'is_moderator') - late bool isModerator; + bool? isModerator; PriceUser(); diff --git a/lib/src/prices/price_user.g.dart b/lib/src/prices/price_user.g.dart index f72ba6687a..555b32fb4d 100644 --- a/lib/src/prices/price_user.g.dart +++ b/lib/src/prices/price_user.g.dart @@ -9,7 +9,7 @@ part of 'price_user.dart'; PriceUser _$PriceUserFromJson(Map json) => PriceUser() ..userId = json['user_id'] as String ..priceCount = (json['price_count'] as num).toInt() - ..isModerator = json['is_moderator'] as bool; + ..isModerator = json['is_moderator'] as bool?; Map _$PriceUserToJson(PriceUser instance) => { 'user_id': instance.userId, diff --git a/lib/src/prices/session.dart b/lib/src/prices/session.dart index 5d3266563f..68dc2e5b6e 100644 --- a/lib/src/prices/session.dart +++ b/lib/src/prices/session.dart @@ -30,6 +30,16 @@ class Session extends JsonObject { @override Map toJson() => _$SessionToJson(this); + /// Status Code when the authentication fails. static const int invalidAuthStatusCode = 401; + + /// Error message when the authentication fails. static const String invalidAuthMessage = 'Invalid authentication credentials'; + + /// Status Code when we try an edit operation with a wrong authentication. + static const int invalidActionWithAuthStatusCode = 403; + + /// Error message when we try an edit operation with a wrong authentication. + static const String invalidActionWithAuthMessage = + 'Authentication credentials were not provided.'; } diff --git a/test/api_prices_test.dart b/test/api_prices_test.dart index 4f58b06c43..58f88c31be 100644 --- a/test/api_prices_test.dart +++ b/test/api_prices_test.dart @@ -68,16 +68,18 @@ void main() { bearerToken: bearerToken, ); expect(deleting.isError, isTrue); - expect(deleting.statusCode, Session.invalidAuthStatusCode); - expect(deleting.detailError, Session.invalidAuthMessage); + expect(deleting.statusCode, Session.invalidActionWithAuthStatusCode); + expect( + deleting.detailError, contains(Session.invalidActionWithAuthMessage)); session = await OpenPricesAPIClient.getUserSession( uriHelper: uriHelper, bearerToken: bearerToken, ); expect(session.isError, isTrue); - expect(session.statusCode, Session.invalidAuthStatusCode); - expect(session.detailError, Session.invalidAuthMessage); + expect(session.statusCode, Session.invalidActionWithAuthStatusCode); + expect( + session.detailError, contains(Session.invalidActionWithAuthMessage)); }); }); @@ -94,12 +96,11 @@ void main() { ..locationOSMType = LocationOSMType.node ..date = DateTime(2024, 1, 18); - final UpdatePriceParameters parameters = UpdatePriceParameters() - ..currency = Currency.USD - ..date = DateTime(2024, 1, 19) - ..price = 12 - ..priceWithoutDiscount = 13 - ..priceIsDiscounted = true; + final UpdatePriceParameters updatePriceParameters = + UpdatePriceParameters() + ..price = 12 + ..priceWithoutDiscount = 13 + ..priceIsDiscounted = true; String bearerToken = invalidBearerToken; @@ -110,8 +111,38 @@ void main() { uriHelper: uriHelper, ); expect(addedPrice.isError, isTrue); - expect(addedPrice.statusCode, Session.invalidAuthStatusCode); - expect(addedPrice.detailError, Session.invalidAuthMessage); + expect(addedPrice.statusCode, Session.invalidActionWithAuthStatusCode); + expect(addedPrice.detailError, + contains(Session.invalidActionWithAuthMessage)); + + final ProofType uploadProofType = ProofType.receipt; + const Currency uploadCurrency = Currency.EUR; + final DateTime uploadDate = DateTime(2024, 1, 1); + + final UpdateProofParameters updateProofParameters = + UpdateProofParameters() + ..type = ProofType.priceTag + ..currency = Currency.USD + ..date = DateTime(2024, 1, 2); + + // TODO(monsieurtanuki): more relevant image if possible + final Uri initialImageUri = + Uri.file('test/test_assets/ingredients_en.jpg'); + final MediaType initialMediaType = + HttpHelper().imagineMediaType(initialImageUri.path)!; + + // failing proof upload with invalid token + MaybeError uploadProof = await OpenPricesAPIClient.uploadProof( + proofType: uploadProofType, + imageUri: initialImageUri, + mediaType: initialMediaType, + bearerToken: bearerToken, + uriHelper: uriHelper, + ); + expect(uploadProof.isError, isTrue); + expect(uploadProof.statusCode, Session.invalidActionWithAuthStatusCode); + expect(uploadProof.detailError, + contains(Session.invalidActionWithAuthMessage)); // authentication final MaybeError token = @@ -124,7 +155,49 @@ void main() { expect(token.value, isNotEmpty); bearerToken = token.value; - // successful price creation with valid token + // successful proof upload with valid token + uploadProof = await OpenPricesAPIClient.uploadProof( + proofType: uploadProofType, + imageUri: initialImageUri, + mediaType: initialMediaType, + currency: uploadCurrency, + date: uploadDate, + bearerToken: bearerToken, + uriHelper: uriHelper, + ); + expect(uploadProof.isError, isFalse); + expect(uploadProof.value.type, uploadProofType); + expect(uploadProof.value.owner, user.userId); + expect(uploadProof.value.id, isNotNull); + expect(uploadProof.value.priceCount, 0); + expect(uploadProof.value.mimetype, initialMediaType.toString()); + expect(uploadProof.value.currency, uploadCurrency); + expect(uploadProof.value.date, uploadDate); + + final int proofId = uploadProof.value.id; + + // successful proof update + MaybeError updateProof = await OpenPricesAPIClient.updateProof( + proofId, + parameters: updateProofParameters, + bearerToken: bearerToken, + uriHelper: uriHelper, + ); + expect(updateProof.isError, isFalse); + + initialPrice.proofId = proofId; + + // failing price creation with valid token but invalid dates + addedPrice = await OpenPricesAPIClient.createPrice( + price: initialPrice, + bearerToken: bearerToken, + uriHelper: uriHelper, + ); + expect(addedPrice.isError, isTrue); + + // successful price creation + initialPrice.date = updateProofParameters.date!; + initialPrice.currency = updateProofParameters.currency!; addedPrice = await OpenPricesAPIClient.createPrice( price: initialPrice, bearerToken: bearerToken, @@ -146,38 +219,57 @@ void main() { final int priceId = addedPrice.value.id; // successful price update - addedPrice = await OpenPricesAPIClient.updatePrice( + MaybeError updatedPrice = await OpenPricesAPIClient.updatePrice( priceId, - parameters: parameters, + parameters: updatePriceParameters, bearerToken: bearerToken, uriHelper: uriHelper, ); - expect(addedPrice.isError, isFalse); - expect(addedPrice.value.price, parameters.price); - expect(addedPrice.value.priceWithoutDiscount, - parameters.priceWithoutDiscount); - expect(addedPrice.value.priceIsDiscounted, parameters.priceIsDiscounted); - expect(addedPrice.value.currency, parameters.currency); - expect(addedPrice.value.date, parameters.date); + expect(updatedPrice.isError, isFalse); // delete price first time: success - MaybeError deleted = await OpenPricesAPIClient.deletePrice( + MaybeError deletedPrice = await OpenPricesAPIClient.deletePrice( priceId: priceId, bearerToken: bearerToken, uriHelper: uriHelper, ); - expect(deleted.isError, isFalse); - expect(deleted.value, isTrue); + expect(deletedPrice.isError, isFalse); + expect(deletedPrice.value, isTrue); // delete price second time: failure - deleted = await OpenPricesAPIClient.deletePrice( + deletedPrice = await OpenPricesAPIClient.deletePrice( priceId: priceId, bearerToken: bearerToken, uriHelper: uriHelper, ); - expect(deleted.isError, isTrue); - expect(deleted.statusCode, 404); - expect(deleted.detailError, 'Price with code $priceId not found'); + expect(deletedPrice.isError, isTrue); + expect(deletedPrice.statusCode, 404); + expect( + deletedPrice.detailError, + 'No Price matches the given query.', + ); + + // delete proof first time: success + MaybeError deletedProof = await OpenPricesAPIClient.deleteProof( + proofId: proofId, + bearerToken: bearerToken, + uriHelper: uriHelper, + ); + expect(deletedProof.isError, isFalse); + expect(deletedProof.value, isTrue); + + // delete proof second time: failure + deletedProof = await OpenPricesAPIClient.deleteProof( + proofId: proofId, + bearerToken: bearerToken, + uriHelper: uriHelper, + ); + expect(deletedProof.isError, isTrue); + expect(deletedProof.statusCode, 404); + expect( + deletedProof.detailError, + 'No Proof matches the given query.', + ); // close session final MaybeError closedSession = @@ -187,7 +279,7 @@ void main() { ); expect(closedSession.isError, isFalse); expect(closedSession.value, isTrue); - }); + }, timeout: Timeout(Duration(seconds: 60))); test('get prices', () async { const UriProductHelper uriHelper = uriHelperFoodProd; @@ -354,23 +446,23 @@ void main() { expect(location.isError, isTrue); expect( location.detailError, - 'Location with id $locationId not found', + 'No Location matches the given query.', ); }); test('get non-existing OSM location', () async { - const int locationOSMId = -1; + const int locationOSMId = 123123123123123123; const LocationOSMType locationOSMType = LocationOSMType.way; final MaybeError location = await OpenPricesAPIClient.getOSMLocation( locationOSMId: locationOSMId, - locationOSMType: LocationOSMType.way, + locationOSMType: locationOSMType, uriHelper: uriHelper, ); expect(location.isError, isTrue); expect( location.detailError, - 'Location with type ${locationOSMType.offTag} & id $locationOSMId not found', + 'No Location matches the given query.', ); }); @@ -496,7 +588,7 @@ void main() { expect(maybePriceProduct.isError, isTrue); expect( maybePriceProduct.detailError, - 'Product with id $productId not found', + 'No Product matches the given query.', ); }); @@ -514,7 +606,7 @@ void main() { }); test('get non-existing product by CODE', () async { - const String productCode = 'not a code'; + const String productCode = '123123123123123123123123'; final MaybeError maybePriceProduct = await OpenPricesAPIClient.getPriceProductByCode( productCode, @@ -523,7 +615,7 @@ void main() { expect(maybePriceProduct.isError, isTrue); expect( maybePriceProduct.detailError, - 'Product with code $productCode not found', + 'No Product matches the given query.', ); }); }); @@ -658,109 +750,6 @@ void main() { expect(response.statusCode, HTTP_OK); } }); - - test('upload', () async { - final ProofType uploadProofType = ProofType.receipt; - const Currency uploadCurrency = Currency.EUR; - final DateTime uploadDate = DateTime(2024, 1, 1); - - final UpdateProofParameters parameters = UpdateProofParameters() - ..type = ProofType.priceTag - ..currency = Currency.USD - ..date = DateTime(2024, 1, 2); - - // TODO(monsieurtanuki): more relevant image if possible - final Uri initialImageUri = - Uri.file('test/test_assets/ingredients_en.jpg'); - final MediaType initialMediaType = - HttpHelper().imagineMediaType(initialImageUri.path)!; - - String bearerToken = invalidBearerToken; - - // failing proof upload with invalid token - MaybeError uploadProof = await OpenPricesAPIClient.uploadProof( - proofType: uploadProofType, - imageUri: initialImageUri, - mediaType: initialMediaType, - bearerToken: bearerToken, - uriHelper: uriHelper, - ); - expect(uploadProof.isError, isTrue); - expect(uploadProof.statusCode, Session.invalidAuthStatusCode); - expect(uploadProof.detailError, Session.invalidAuthMessage); - - // authentication - final MaybeError token = - await OpenPricesAPIClient.getAuthenticationToken( - username: user.userId, - password: user.password, - uriHelper: uriHelper, - ); - expect(token.isError, isFalse); - expect(token.value, isNotEmpty); - bearerToken = token.value; - - // successful proof upload with valid token - uploadProof = await OpenPricesAPIClient.uploadProof( - proofType: uploadProofType, - imageUri: initialImageUri, - mediaType: initialMediaType, - currency: uploadCurrency, - date: uploadDate, - bearerToken: bearerToken, - uriHelper: uriHelper, - ); - expect(uploadProof.isError, isFalse); - expect(uploadProof.value.type, uploadProofType); - expect(uploadProof.value.owner, user.userId); - expect(uploadProof.value.id, isNotNull); - expect(uploadProof.value.priceCount, 0); - expect(uploadProof.value.mimetype, initialMediaType.toString()); - expect(uploadProof.value.currency, uploadCurrency); - expect(uploadProof.value.date, uploadDate); - - final int proofId = uploadProof.value.id; - - // successful proof update - uploadProof = await OpenPricesAPIClient.updateProof( - proofId, - parameters: parameters, - bearerToken: bearerToken, - uriHelper: uriHelper, - ); - expect(uploadProof.isError, isFalse); - expect(uploadProof.value.type, parameters.type); - expect(uploadProof.value.currency, parameters.currency); - expect(uploadProof.value.date, parameters.date); - - // delete proof first time: success - MaybeError deleted = await OpenPricesAPIClient.deleteProof( - proofId: proofId, - bearerToken: bearerToken, - uriHelper: uriHelper, - ); - expect(deleted.isError, isFalse); - expect(deleted.value, isTrue); - - // delete proof second time: failure - deleted = await OpenPricesAPIClient.deleteProof( - proofId: proofId, - bearerToken: bearerToken, - uriHelper: uriHelper, - ); - expect(deleted.isError, isTrue); - expect(deleted.statusCode, 404); - expect(deleted.detailError, 'Proof with code $proofId not found'); - - // close session - final MaybeError closedSession = - await OpenPricesAPIClient.deleteUserSession( - uriHelper: uriHelper, - bearerToken: bearerToken, - ); - expect(closedSession.isError, isFalse); - expect(closedSession.value, isTrue); - }); }); group('$OpenPricesAPIClient Users', () {