diff --git a/lib/openfoodfacts.dart b/lib/openfoodfacts.dart
index d79b99f602..a095a41f96 100644
--- a/lib/openfoodfacts.dart
+++ b/lib/openfoodfacts.dart
@@ -143,6 +143,8 @@ export 'src/utils/server_type.dart';
export 'src/utils/suggestion_manager.dart';
export 'src/utils/tag_type.dart';
export 'src/utils/tag_type_autocompleter.dart';
+export 'src/utils/too_many_requests_exception.dart';
+export 'src/utils/too_many_requests_manager.dart';
export 'src/utils/unit_helper.dart';
export 'src/utils/uri_helper.dart';
export 'src/utils/uri_reader.dart';
diff --git a/lib/src/open_food_api_client.dart b/lib/src/open_food_api_client.dart
index 0fd91fdf1d..8c00920b78 100644
--- a/lib/src/open_food_api_client.dart
+++ b/lib/src/open_food_api_client.dart
@@ -47,6 +47,7 @@ import 'utils/product_query_configurations.dart';
import 'utils/product_search_query_configuration.dart';
import 'utils/tag_type.dart';
import 'utils/taxonomy_query_configuration.dart';
+import 'utils/too_many_requests_exception.dart';
import 'utils/uri_helper.dart';
/// Client calls of the Open Food Facts API
@@ -356,6 +357,7 @@ class OpenFoodAPIClient {
final UriProductHelper uriHelper = uriHelperFoodProd,
}) async {
final Response response = await configuration.getResponse(user, uriHelper);
+ TooManyRequestsException.check(response);
return response.body;
}
@@ -519,6 +521,7 @@ class OpenFoodAPIClient {
final UriProductHelper uriHelper = uriHelperFoodProd,
}) async {
final Response response = await configuration.getResponse(user, uriHelper);
+ TooManyRequestsException.check(response);
final String jsonStr = _replaceQuotes(response.body);
final SearchResult result = SearchResult.fromJson(
HttpHelper().jsonDecode(jsonStr),
diff --git a/lib/src/prices/get_locations_order.dart b/lib/src/prices/get_locations_order.dart
index 69886dbb89..ec51e163d5 100644
--- a/lib/src/prices/get_locations_order.dart
+++ b/lib/src/prices/get_locations_order.dart
@@ -2,6 +2,7 @@ import 'order_by.dart';
/// Field for the "order by" clause of "get locations".
enum GetLocationsOrderField implements OrderByField {
+ priceCount(offTag: 'price_count'),
created(offTag: 'created'),
updated(offTag: 'updated');
diff --git a/lib/src/prices/get_proofs_order.dart b/lib/src/prices/get_proofs_order.dart
index 4efd306b1e..b3f581746a 100644
--- a/lib/src/prices/get_proofs_order.dart
+++ b/lib/src/prices/get_proofs_order.dart
@@ -2,6 +2,7 @@ import 'order_by.dart';
/// Field for the "order by" clause of "get proofs".
enum GetProofsOrderField implements OrderByField {
+ priceCount(offTag: 'price_count'),
created(offTag: 'created');
const GetProofsOrderField({required this.offTag});
diff --git a/lib/src/utils/too_many_requests_exception.dart b/lib/src/utils/too_many_requests_exception.dart
new file mode 100644
index 0000000000..97e37d7df0
--- /dev/null
+++ b/lib/src/utils/too_many_requests_exception.dart
@@ -0,0 +1,19 @@
+import 'package:http/http.dart';
+
+/// Exception when the server returns "Too many requests".
+class TooManyRequestsException implements Exception {
+ const TooManyRequestsException();
+
+ /// Start of the response body when the server received too many requests.
+ static const String _tooManyRequestsError =
+ '
TOO MANY REQUESTS
';
+
+ static void check(final Response response) {
+ if (response.body.startsWith(_tooManyRequestsError)) {
+ throw TooManyRequestsException();
+ }
+ }
+
+ @override
+ String toString() => 'Too many requests';
+}
diff --git a/lib/src/utils/too_many_requests_manager.dart b/lib/src/utils/too_many_requests_manager.dart
new file mode 100644
index 0000000000..f5bd250c2f
--- /dev/null
+++ b/lib/src/utils/too_many_requests_manager.dart
@@ -0,0 +1,46 @@
+/// Manager dedicated to "too many requests" server response.
+///
+/// Typically, the server may limit the number of requests to a [maxCount]
+/// during a specific [duration].
+class TooManyRequestsManager {
+ TooManyRequestsManager({
+ required this.maxCount,
+ required this.duration,
+ });
+
+ final int maxCount;
+ final Duration duration;
+
+ final List _requestTimestamps = [];
+
+ /// Waits the needed duration in order to avoid "too many requests" error.
+ Future waitIfNeeded() async {
+ while (_requestTimestamps.length >= maxCount) {
+ final int previousInMillis = _requestTimestamps.first;
+ final int nowInMillis = DateTime.now().millisecondsSinceEpoch;
+ final int waitingInMillis =
+ duration.inMilliseconds - nowInMillis + previousInMillis;
+ if (waitingInMillis > 0) {
+ await Future.delayed(Duration(milliseconds: waitingInMillis));
+ }
+ _requestTimestamps.removeAt(0);
+ }
+ final DateTime now = DateTime.now();
+ final int nowInMillis = now.millisecondsSinceEpoch;
+ _requestTimestamps.add(nowInMillis);
+ }
+}
+
+/// [TooManyRequestsManager] dedicated to "searchProducts" queries in PROD.
+final TooManyRequestsManager searchProductsTooManyRequestsManager =
+ TooManyRequestsManager(
+ maxCount: 10,
+ duration: Duration(minutes: 1),
+);
+
+/// [TooManyRequestsManager] dedicated to "getProduct" queries in PROD.
+final TooManyRequestsManager getProductTooManyRequestsManager =
+ TooManyRequestsManager(
+ maxCount: 100,
+ duration: Duration(minutes: 1),
+);
diff --git a/test/api_get_localized_product_test.dart b/test/api_get_localized_product_test.dart
index dac8bad6e6..be985d0052 100644
--- a/test/api_get_localized_product_test.dart
+++ b/test/api_get_localized_product_test.dart
@@ -7,6 +7,13 @@ void main() {
OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
+ Future getProductV3InProd(
+ ProductQueryConfiguration configuration,
+ ) async {
+ await getProductTooManyRequestsManager.waitIfNeeded();
+ return OpenFoodAPIClient.getProductV3(configuration);
+ }
+
group('$OpenFoodAPIClient get localized product fields', () {
test('get packaging text in languages (Coca-Cola)', () async {
const String barcode = '5449000000996';
@@ -22,7 +29,7 @@ void main() {
fields: [ProductField.PACKAGING_TEXT_IN_LANGUAGES],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -42,8 +49,7 @@ void main() {
OpenFoodFactsLanguage.FRENCH,
];
- final ProductResultV3 productResult =
- await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 productResult = await getProductV3InProd(
ProductQueryConfiguration(
BARCODE_DANISH_BUTTER_COOKIES,
languages: languages,
@@ -107,7 +113,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -265,7 +271,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -325,7 +331,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -396,7 +402,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -447,7 +453,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
diff --git a/test/api_get_product_image_ids_test.dart b/test/api_get_product_image_ids_test.dart
index 4ba797f842..1873a68367 100644
--- a/test/api_get_product_image_ids_test.dart
+++ b/test/api_get_product_image_ids_test.dart
@@ -8,6 +8,7 @@ void main() {
test('get product images (all, main and raw)', () async {
const String barcode = '3019081238643';
+ await getProductTooManyRequestsManager.waitIfNeeded();
final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
ProductQueryConfiguration(
barcode,
diff --git a/test/api_get_product_test.dart b/test/api_get_product_test.dart
index f0126d96c1..6ebd4ea274 100644
--- a/test/api_get_product_test.dart
+++ b/test/api_get_product_test.dart
@@ -13,6 +13,13 @@ void main() {
OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
+ Future getProductV3InProd(
+ ProductQueryConfiguration configuration,
+ ) async {
+ await getProductTooManyRequestsManager.waitIfNeeded();
+ return OpenFoodAPIClient.getProductV3(configuration);
+ }
+
void findExpectedIngredients(
final List ingredients,
final List labels,
@@ -56,7 +63,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -85,7 +92,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -119,7 +126,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -186,7 +193,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -210,7 +217,7 @@ void main() {
ProductResultV3 result;
late Nutriments nutriments;
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
'5060517883638',
language: language,
@@ -229,7 +236,7 @@ void main() {
isNull,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
'7612100018477',
language: language,
@@ -245,7 +252,7 @@ void main() {
);
expect(nutriments.getValue(Nutrient.biotin, PerSize.serving), isNull);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
'3057640257773',
language: language,
@@ -264,7 +271,7 @@ void main() {
.015,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
'4260556630007',
language: language,
@@ -307,7 +314,7 @@ void main() {
.00002,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
'3155251205319',
language: language,
@@ -344,7 +351,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -448,7 +455,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ ProductResultV3 result = await getProductV3InProd(
configurations,
);
expect(result.product, isNull);
@@ -463,7 +470,7 @@ void main() {
fields: [ProductField.ALL],
version: ProductQueryVersion.v3,
);
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -483,7 +490,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -509,7 +516,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -539,7 +546,7 @@ void main() {
fields: [ProductField.NAME, ProductField.LANGUAGE],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configurations,
);
@@ -559,7 +566,7 @@ void main() {
fields: [ProductField.NAME, ProductField.COUNTRIES],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configurations,
);
@@ -581,7 +588,7 @@ void main() {
fields: [ProductField.NAME, ProductField.COUNTRIES_TAGS],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configurations,
);
@@ -606,7 +613,7 @@ void main() {
fields: [ProductField.NAME, ProductField.ATTRIBUTE_GROUPS],
version: ProductQueryVersion.v3,
);
- ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ ProductResultV3 result = await getProductV3InProd(
configurations,
);
@@ -656,7 +663,7 @@ void main() {
const int numberOfImages = 53; // was 53 in 20231125
//Get product without setting OpenFoodFactsLanguage or ProductField
- ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ ProductResultV3 result = await getProductV3InProd(
ProductQueryConfiguration(
barcode,
version: ProductQueryVersion.v3,
@@ -700,7 +707,7 @@ void main() {
'https://images.openfoodfacts.org/images/products/500/011/254/8167/ingredients_de.7.400.jpg');
//Get product without setting ProductField
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
barcode,
language: OpenFoodFactsLanguage.GERMAN,
@@ -753,7 +760,7 @@ void main() {
'https://images.openfoodfacts.org/images/products/500/011/254/8167/ingredients_de.7.400.jpg');
//Get product without setting OpenFoodFactsLanguage
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
ProductQueryConfiguration(
barcode,
fields: [ProductField.ALL],
@@ -834,7 +841,7 @@ void main() {
test(
'vegan, vegetarian and palm oil ingredients of Danish Butter Cookies & Chocolate Chip Cookies',
() async {
- final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 result = await getProductV3InProd(
ProductQueryConfiguration(
'3017620429484',
language: OpenFoodFactsLanguage.FRENCH,
@@ -860,8 +867,7 @@ void main() {
'nutriscore_2023',
'root',
};
- final ProductResultV3 productResult =
- await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 productResult = await getProductV3InProd(
ProductQueryConfiguration(
BARCODE_DANISH_BUTTER_COOKIES,
language: OpenFoodFactsLanguage.FRENCH,
@@ -1017,7 +1023,7 @@ void main() {
fields: [ProductField.COMPARED_TO_CATEGORY],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configuration,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -1032,7 +1038,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configuration,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -1047,7 +1053,7 @@ void main() {
fields: [ProductField.OBSOLETE],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configuration,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -1060,7 +1066,7 @@ void main() {
fields: [ProductField.OBSOLETE],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configuration,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -1087,7 +1093,7 @@ void main() {
],
version: ProductQueryVersion.v3,
);
- result = await OpenFoodAPIClient.getProductV3(
+ result = await getProductV3InProd(
configuration,
);
expect(result.status, ProductResultV3.statusSuccess);
@@ -1136,7 +1142,6 @@ void main() {
group('$OpenFoodAPIClient get new packagings field', () {
const String barcode = '3661344723290';
- const String searchTerms = 'skyr les 2 vaches';
const OpenFoodFactsLanguage language = OpenFoodFactsLanguage.FRENCH;
const OpenFoodFactsCountry country = OpenFoodFactsCountry.FRANCE;
const ProductQueryVersion version = ProductQueryVersion.v3;
@@ -1159,8 +1164,7 @@ void main() {
}
test('as a single field on a barcode search', () async {
- final ProductResultV3 productResult =
- await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 productResult = await getProductV3InProd(
ProductQueryConfiguration(
barcode,
fields: [ProductField.PACKAGINGS],
@@ -1175,8 +1179,7 @@ void main() {
});
test('as a part of ALL fields on a barcode search', () async {
- final ProductResultV3 productResult =
- await OpenFoodAPIClient.getProductV3(
+ final ProductResultV3 productResult = await getProductV3InProd(
ProductQueryConfiguration(
barcode,
fields: [ProductField.ALL],
@@ -1189,79 +1192,5 @@ void main() {
expect(productResult.product, isNotNull);
checkProduct(productResult.product!);
});
-
- test('as a single field on a search query', () async {
- final SearchResult searchResult = await OpenFoodAPIClient.searchProducts(
- null,
- ProductSearchQueryConfiguration(
- parametersList: [
- SearchTerms(terms: [searchTerms])
- ],
- fields: [ProductField.PACKAGINGS, ProductField.BARCODE],
- language: language,
- country: country,
- version: version,
- ),
- );
- expect(searchResult.products, isNotNull);
- expect(searchResult.products, isNotEmpty);
- bool found = false;
- for (final Product product in searchResult.products!) {
- if (product.barcode != barcode) {
- continue;
- }
- found = true;
- checkProduct(product);
- }
- expect(found, isTrue);
- });
-
- test('as a part of ALL fields on a search query', () async {
- final SearchResult searchResult = await OpenFoodAPIClient.searchProducts(
- null,
- ProductSearchQueryConfiguration(
- parametersList: [
- SearchTerms(terms: [searchTerms])
- ],
- fields: [ProductField.ALL],
- language: language,
- country: country,
- version: version,
- ),
- );
- expect(searchResult.products, isNotNull);
- expect(searchResult.products, isNotEmpty);
- bool found = false;
- for (final Product product in searchResult.products!) {
- if (product.barcode != barcode) {
- continue;
- }
- found = true;
- checkProduct(product);
- }
- expect(found, isTrue);
- });
-
- test('as a part of RAW fields on a search query', () async {
- try {
- await OpenFoodAPIClient.searchProducts(
- null,
- ProductSearchQueryConfiguration(
- parametersList: [
- SearchTerms(terms: [searchTerms])
- ],
- fields: [ProductField.RAW],
- language: language,
- country: country,
- version: version,
- ),
- );
- } catch (e) {
- // In RAW mode the packagings are mere String's instead of LocalizedTag's.
- // Therefore we expect an Exception.
- return;
- }
- fail('On Raw');
- });
});
}
diff --git a/test/api_get_to_be_completed_products_test.dart b/test/api_get_to_be_completed_products_test.dart
deleted file mode 100644
index 2e46cf95b1..0000000000
--- a/test/api_get_to_be_completed_products_test.dart
+++ /dev/null
@@ -1,114 +0,0 @@
-import 'package:openfoodfacts/openfoodfacts.dart';
-import 'package:test/test.dart';
-
-import 'test_constants.dart';
-
-/// Integration tests related to the "to-be-completed" products
-void main() {
- OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
- OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
-
- group('$OpenFoodAPIClient get all to-be-completed products', () {
- Future getCount(
- final OpenFoodFactsCountry country,
- final OpenFoodFactsLanguage language,
- final String store,
- ) async {
- final String reason = '($country, $language)';
- final ProductSearchQueryConfiguration configuration =
- ProductSearchQueryConfiguration(
- country: country,
- language: language,
- fields: [
- ProductField.BARCODE,
- ProductField.STATES_TAGS,
- ],
- parametersList: [
- StatesTagsParameter(map: {ProductState.COMPLETED: false}),
- TagFilter.fromType(
- tagFilterType: TagFilterType.STORES,
- tagName: store,
- ),
- ],
- version: ProductQueryVersion.v3,
- );
-
- final SearchResult result;
- try {
- result = await OpenFoodAPIClient.searchProducts(
- OpenFoodAPIConfiguration.globalUser,
- configuration,
- );
- } catch (e) {
- fail('Could not retrieve data for $reason: $e');
- }
- expect(result.page, 1, reason: reason); // default
- expect(result.products, isNotNull, reason: reason);
- for (final Product product in result.products!) {
- expect(product.statesTags, isNotNull);
- expect(product.statesTags!, contains('en:to-be-completed'));
- }
- return result.count;
- }
-
- Future getCountForAllLanguages(
- final OpenFoodFactsCountry country,
- final String store,
- ) async {
- final List languages = [
- OpenFoodFactsLanguage.ENGLISH,
- OpenFoodFactsLanguage.FRENCH,
- OpenFoodFactsLanguage.ITALIAN,
- ];
- int? result;
- for (final OpenFoodFactsLanguage language in languages) {
- final int? count = await getCount(country, language, store);
- if (result != null) {
- expect(count, result, reason: language.toString());
- }
- result = count;
- }
- return result!;
- }
-
- Future checkTypeCount(
- final OpenFoodFactsCountry country,
- final String store,
- final int minimalExpectedCount,
- ) async {
- final int count = await getCountForAllLanguages(country, store);
- expect(count, greaterThanOrEqualTo(minimalExpectedCount));
- }
-
- test(
- 'in France',
- () async => checkTypeCount(
- OpenFoodFactsCountry.FRANCE,
- 'Carrefour',
- // 2023-08-12: was 14778
- 10000,
- ));
-
- test(
- 'in Italy',
- () async => checkTypeCount(
- OpenFoodFactsCountry.ITALY,
- 'Carrefour',
- // 2023-07-09: was 2394
- 1500,
- ));
-
- test(
- 'in Spain',
- () async => checkTypeCount(
- OpenFoodFactsCountry.SPAIN,
- 'El Corte Inglès',
- // 2023-07-09: was 608
- 500,
- ));
- },
- timeout: Timeout(
- // some tests can be slow here
- Duration(seconds: 180),
- ));
-}
diff --git a/test/api_get_user_products_test.dart b/test/api_get_user_products_test.dart
deleted file mode 100644
index a1803d0403..0000000000
--- a/test/api_get_user_products_test.dart
+++ /dev/null
@@ -1,135 +0,0 @@
-import 'package:openfoodfacts/openfoodfacts.dart';
-import 'package:test/test.dart';
-
-import 'test_constants.dart';
-
-void main() {
- OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
- OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
-
- group('$OpenFoodAPIClient get user products', () {
- const String userId = 'monsieurtanuki';
- // should be big enough to get everything on page1
- const int pageSize = 100;
- final String toBeCompletedTag = ProductState.COMPLETED.toBeCompletedTag;
-
- Future getCount(
- final TagFilterType type,
- final OpenFoodFactsLanguage language,
- final bool toBeCompleted, {
- final void Function(Product)? additionalCheck,
- }) async {
- final String reason = '($language, $type)';
- final ProductSearchQueryConfiguration configuration =
- ProductSearchQueryConfiguration(
- parametersList: [
- TagFilter.fromType(tagFilterType: type, tagName: userId),
- PageSize(size: pageSize),
- if (toBeCompleted)
- TagFilter.fromType(
- tagFilterType: TagFilterType.STATES,
- tagName: toBeCompletedTag,
- ),
- ],
- language: language,
- fields: [
- ProductField.BARCODE,
- ProductField.STATES_TAGS,
- ],
- version: ProductQueryVersion.v3,
- );
-
- final SearchResult result;
- try {
- result = await OpenFoodAPIClient.searchProducts(
- OpenFoodAPIConfiguration.globalUser,
- configuration,
- );
- } catch (e) {
- fail('Could not retrieve data for $reason: $e');
- }
- expect(result.page, 1, reason: reason); // default
- expect(result.pageSize, pageSize, reason: reason);
- expect(result.products, isNotNull, reason: reason);
- expect(result.products!.length, result.pageCount, reason: reason);
- if (additionalCheck != null) {
- for (final Product product in result.products!) {
- additionalCheck(product);
- }
- }
- return result.pageCount!;
- }
-
- Future getCountForAllLanguages(
- final TagFilterType type,
- final bool toBeCompleted, {
- final void Function(Product)? additionalCheck,
- }) async {
- final List languages = [
- OpenFoodFactsLanguage.ENGLISH,
- OpenFoodFactsLanguage.FRENCH,
- OpenFoodFactsLanguage.ITALIAN,
- ];
- int? result;
- for (final OpenFoodFactsLanguage language in languages) {
- final int count = await getCount(
- type,
- language,
- toBeCompleted,
- additionalCheck: additionalCheck,
- );
- if (result != null) {
- expect(count, result, reason: language.toString());
- }
- result = count;
- }
- return result!;
- }
-
- Future checkTypeCount(
- final TagFilterType type,
- final int minimalExpectedCount, {
- final void Function(Product)? additionalCheck,
- final bool toBeCompleted = false,
- }) async {
- final int count = await getCountForAllLanguages(
- type,
- toBeCompleted,
- additionalCheck: additionalCheck,
- );
- expect(count, greaterThanOrEqualTo(minimalExpectedCount));
- }
-
- test(
- 'contributor',
- () async => checkTypeCount(TagFilterType.CREATOR, 2) // as of 20221229
- ,
- );
-
- test(
- 'informer',
- () async =>
- await checkTypeCount(TagFilterType.INFORMERS, 73) // as of 20221229
- ,
- );
-
- test(
- 'photographer',
- () async =>
- checkTypeCount(TagFilterType.PHOTOGRAPHERS, 48) // as of 20221229
- ,
- );
-
- test(
- 'to be completed',
- () async => checkTypeCount(
- TagFilterType.INFORMERS, 0, // you never know...
- toBeCompleted: true,
- additionalCheck: (final Product product) {
- expect(product.statesTags, isNotNull);
- expect(product.statesTags, contains(toBeCompletedTag));
- },
- ),
- );
- });
-}
diff --git a/test/api_json_to_from_test.dart b/test/api_json_to_from_test.dart
index 3c18bb56b2..5d1841d73c 100644
--- a/test/api_json_to_from_test.dart
+++ b/test/api_json_to_from_test.dart
@@ -10,6 +10,7 @@ void main() {
group('$OpenFoodAPIClient json to/from conversions', () {
test('images', () async {
+ await getProductTooManyRequestsManager.waitIfNeeded();
final ProductResultV3 productResult =
await OpenFoodAPIClient.getProductV3(
ProductQueryConfiguration(
diff --git a/test/api_matched_product_v1_test.dart b/test/api_matched_product_v1_test.dart
index 8b17245cbf..bd37a28f1a 100644
--- a/test/api_matched_product_v1_test.dart
+++ b/test/api_matched_product_v1_test.dart
@@ -53,6 +53,7 @@ void main() {
fields: [ProductField.NAME, ProductField.ATTRIBUTE_GROUPS],
version: ProductQueryVersion.v3,
);
+ await getProductTooManyRequestsManager.waitIfNeeded();
final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
configurations,
user: TestConstants.PROD_USER,
diff --git a/test/api_matched_product_v2_test.dart b/test/api_matched_product_v2_test.dart
deleted file mode 100644
index ecdd0d33a9..0000000000
--- a/test/api_matched_product_v2_test.dart
+++ /dev/null
@@ -1,185 +0,0 @@
-import 'package:http/http.dart' as http;
-import 'package:openfoodfacts/openfoodfacts.dart';
-import 'package:test/test.dart';
-
-import 'test_constants.dart';
-
-class _Score {
- _Score(this.score, this.status);
-
- final double score;
- final MatchedProductStatusV2 status;
-}
-
-void main() {
- const int HTTP_OK = 200;
-
- const OpenFoodFactsLanguage language = OpenFoodFactsLanguage.FRENCH;
- OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
- OpenFoodAPIConfiguration.globalCountry = OpenFoodFactsCountry.FRANCE;
- OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
- OpenFoodAPIConfiguration.globalLanguages = [language];
-
- const String BARCODE_KNACKI = '7613035937420';
- const String BARCODE_CORDONBLEU = '4000405005026';
- const String BARCODE_ORIENTALES = '4032277007211';
- const String BARCODE_HACK = '7613037672756';
- const String BARCODE_SCHNITZEL = '4061458069878';
- const String BARCODE_CHIPOLATA = '3770016162098';
- const String BARCODE_FLEISCHWURST = '4003171036379'; // now veggie!
- const String BARCODE_POULET = '40897837';
- const String BARCODE_SAUCISSON = '20045456';
- const String BARCODE_PIZZA = '4260414150470';
- const String BARCODE_ARDECHE = '20712570';
- const String BARCODE_CHORIZO = '8480000591074';
-
- final List inputBarcodes = [
- BARCODE_CHIPOLATA,
- BARCODE_FLEISCHWURST,
- BARCODE_KNACKI,
- BARCODE_CORDONBLEU,
- BARCODE_SAUCISSON,
- BARCODE_PIZZA,
- BARCODE_ORIENTALES,
- BARCODE_ARDECHE,
- BARCODE_HACK,
- BARCODE_CHORIZO,
- BARCODE_SCHNITZEL,
- BARCODE_POULET,
- ];
- final Map expectedScores = {
- BARCODE_KNACKI: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_CORDONBLEU: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_ORIENTALES: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_HACK: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_SCHNITZEL: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_CHIPOLATA: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_FLEISCHWURST: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
- BARCODE_POULET: _Score(0, MatchedProductStatusV2.UNKNOWN_MATCH),
- BARCODE_SAUCISSON: _Score(0, MatchedProductStatusV2.UNKNOWN_MATCH),
- BARCODE_PIZZA: _Score(0, MatchedProductStatusV2.DOES_NOT_MATCH),
- BARCODE_ARDECHE: _Score(0, MatchedProductStatusV2.DOES_NOT_MATCH),
- BARCODE_CHORIZO: _Score(0, MatchedProductStatusV2.DOES_NOT_MATCH),
- };
- final List expectedBarcodeOrder = [
- BARCODE_CHIPOLATA,
- BARCODE_FLEISCHWURST,
- BARCODE_KNACKI,
- BARCODE_CORDONBLEU,
- BARCODE_ORIENTALES,
- BARCODE_HACK,
- BARCODE_SCHNITZEL,
- BARCODE_SAUCISSON,
- BARCODE_POULET,
- BARCODE_PIZZA,
- BARCODE_ARDECHE,
- BARCODE_CHORIZO,
- ];
-
- Future> downloadProducts() async {
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- OpenFoodAPIConfiguration.globalUser,
- ProductSearchQueryConfiguration(
- parametersList: [BarcodeParameter.list(inputBarcodes)],
- language: language,
- fields: [ProductField.BARCODE, ProductField.ATTRIBUTE_GROUPS],
- version: ProductQueryVersion.v3,
- ),
- );
- expect(result.count, expectedScores.keys.length);
- expect(result.page, 1);
- expect(result.products, isNotNull);
- final List products = result.products!;
- // sorting them again by the input order
- products.sort(
- (final Product a, final Product b) => inputBarcodes
- .indexOf(a.barcode!)
- .compareTo(inputBarcodes.indexOf(b.barcode!)),
- );
- expect(products.length, inputBarcodes.length);
- return products;
- }
-
- Future getManager() async {
- final Map attributeImportances = {};
- final ProductPreferencesManager manager = ProductPreferencesManager(
- ProductPreferencesSelection(
- setImportance: (String attributeId, String importanceIndex) async {
- attributeImportances[attributeId] = importanceIndex;
- },
- getImportance: (String attributeId) =>
- attributeImportances[attributeId] ??
- PreferenceImportance.ID_NOT_IMPORTANT,
- ),
- );
- final String languageCode = language.code;
- final String importanceUrl =
- AvailablePreferenceImportances.getUrl(languageCode);
- final String attributeGroupUrl =
- AvailableAttributeGroups.getUrl(languageCode);
- http.Response response;
- response = await http.get(Uri.parse(importanceUrl));
- expect(response.statusCode, HTTP_OK);
- final String preferenceImportancesString = response.body;
- response = await http.get(Uri.parse(attributeGroupUrl));
- expect(response.statusCode, HTTP_OK);
- final String attributeGroupsString = response.body;
- manager.availableProductPreferences =
- AvailableProductPreferences.loadFromJSONStrings(
- preferenceImportancesString: preferenceImportancesString,
- attributeGroupsString: attributeGroupsString,
- );
- await manager.setImportance(
- Attribute.ATTRIBUTE_VEGETARIAN,
- PreferenceImportance.ID_MANDATORY,
- );
- return manager;
- }
-
- /// Tests around Matched Product v2.
- group('$OpenFoodAPIClient matched product v2', () {
- test('matched product', () async {
- final ProductPreferencesManager manager = await getManager();
-
- final List products = await downloadProducts();
-
- final List actuals =
- MatchedProductV2.sort(products, manager);
-
- expect(actuals.length, expectedBarcodeOrder.length);
- for (int i = 0; i < actuals.length; i++) {
- final MatchedProductV2 matched = actuals[i];
- final String barcode = expectedBarcodeOrder[i];
- expect(matched.product.barcode, barcode);
- expect(matched.barcode, barcode);
- expect(expectedScores[barcode], isNotNull);
- final _Score score = expectedScores[barcode]!;
- expect(matched.status, score.status);
- expect(matched.score, score.score);
- }
- });
-
- test('matched score', () async {
- final ProductPreferencesManager manager = await getManager();
-
- final List products = await downloadProducts();
-
- final List actuals = [];
- for (final Product product in products) {
- actuals.add(MatchedScoreV2(product, manager));
- }
- MatchedScoreV2.sort(actuals);
-
- expect(actuals.length, expectedBarcodeOrder.length);
- for (int i = 0; i < actuals.length; i++) {
- final MatchedScoreV2 matched = actuals[i];
- final String barcode = expectedBarcodeOrder[i];
- expect(matched.barcode, barcode);
- expect(expectedScores[barcode], isNotNull);
- final _Score score = expectedScores[barcode]!;
- expect(matched.status, score.status);
- expect(matched.score, score.score);
- }
- });
- });
-}
diff --git a/test/api_not_food_get_product_test.dart b/test/api_not_food_get_product_test.dart
index 4be4e4d682..41a44db4e5 100644
--- a/test/api_not_food_get_product_test.dart
+++ b/test/api_not_food_get_product_test.dart
@@ -34,6 +34,7 @@ void main() {
fields: [ProductField.BARCODE],
version: ProductQueryVersion(2),
);
+ await getProductTooManyRequestsManager.waitIfNeeded();
final OldProductResult result = await OpenFoodAPIClient.getOldProduct(
configurations,
uriHelper: uriHelper,
diff --git a/test/api_ocr_ingredients_test.dart b/test/api_ocr_ingredients_test.dart
index 19a4111d24..1251e8464f 100644
--- a/test/api_ocr_ingredients_test.dart
+++ b/test/api_ocr_ingredients_test.dart
@@ -125,6 +125,7 @@ void main() {
fields: [ProductField.INGREDIENTS_TEXT],
version: ProductQueryVersion.v3,
);
+ await getProductTooManyRequestsManager.waitIfNeeded();
final ProductResultV3 result = await OpenFoodAPIClient.getProductV3(
configurations,
user: TestConstants.PROD_USER,
diff --git a/test/api_search_products_test.dart b/test/api_search_products_test.dart
index e2edd196ec..21e0620248 100644
--- a/test/api_search_products_test.dart
+++ b/test/api_search_products_test.dart
@@ -1,16 +1,34 @@
import 'dart:math';
+import 'package:http/http.dart';
import 'package:openfoodfacts/openfoodfacts.dart';
import 'package:test/test.dart';
import 'test_constants.dart';
+class _Score {
+ _Score(this.score, this.status);
+
+ final double score;
+ final MatchedProductStatusV2 status;
+}
+
void main() {
OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
const ProductQueryVersion version = ProductQueryVersion.v3;
const int defaultPageSize = 50;
+ Future searchProductsInProd(
+ final AbstractQueryConfiguration configuration,
+ ) async {
+ await searchProductsTooManyRequestsManager.waitIfNeeded();
+ return OpenFoodAPIClient.searchProducts(
+ TestConstants.PROD_USER,
+ configuration,
+ );
+ }
+
// additional parameter for faster response time
const Parameter optimParameter = SearchTerms(terms: ['pizza']);
@@ -56,8 +74,7 @@ void main() {
if (currentOption != null) SortBy(option: currentOption)
];
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
ProductSearchQueryConfiguration(
parametersList: parameters,
fields: [ProductField.BARCODE],
@@ -137,8 +154,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -165,8 +181,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -213,8 +228,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -341,8 +355,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -369,8 +382,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -400,8 +412,7 @@ void main() {
version: version,
);
- SearchResult result = await OpenFoodAPIClient.searchProducts(
- null,
+ SearchResult result = await searchProductsInProd(
configuration,
);
@@ -430,8 +441,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -466,8 +476,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -551,8 +560,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -600,8 +608,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -629,8 +636,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -694,8 +700,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -711,6 +716,7 @@ void main() {
});
test('product freshness', () async {
+ await searchProductsTooManyRequestsManager.waitIfNeeded();
final Map result =
await OpenFoodAPIClient.getProductFreshness(
barcodes: BARCODES,
@@ -746,8 +752,9 @@ void main() {
version: version,
);
- final result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER, configuration);
+ final result = await searchProductsInProd(
+ configuration,
+ );
if (result.products == null || result.products!.isEmpty) {
break;
}
@@ -775,8 +782,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -804,8 +810,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -829,8 +834,7 @@ void main() {
// single filters
for (final int novaGroup in novaMinCounts.keys) {
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
ProductSearchQueryConfiguration(
parametersList: [
TagFilter.fromType(
@@ -867,7 +871,7 @@ void main() {
},
timeout: Timeout(
// some tests can be slow here
- Duration(seconds: 90),
+ Duration(seconds: 300),
));
/// Returns random and different int's.
@@ -911,8 +915,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -1021,8 +1024,7 @@ void main() {
version: version,
);
- final SearchResult result = await OpenFoodAPIClient.searchProducts(
- TestConstants.PROD_USER,
+ final SearchResult result = await searchProductsInProd(
configuration,
);
@@ -1111,4 +1113,499 @@ void main() {
// some tests can be slow here
Duration(seconds: 300),
));
+
+ group('$OpenFoodAPIClient get all to-be-completed products', () {
+ Future getCount(
+ final OpenFoodFactsCountry country,
+ final OpenFoodFactsLanguage language,
+ final String store,
+ ) async {
+ final String reason = '($country, $language)';
+ final ProductSearchQueryConfiguration configuration =
+ ProductSearchQueryConfiguration(
+ country: country,
+ language: language,
+ fields: [
+ ProductField.BARCODE,
+ ProductField.STATES_TAGS,
+ ],
+ parametersList: [
+ StatesTagsParameter(map: {ProductState.COMPLETED: false}),
+ TagFilter.fromType(
+ tagFilterType: TagFilterType.STORES,
+ tagName: store,
+ ),
+ ],
+ version: ProductQueryVersion.v3,
+ );
+
+ final SearchResult result;
+ try {
+ result = await searchProductsInProd(
+ configuration,
+ );
+ } catch (e) {
+ fail('Could not retrieve data for $reason: $e');
+ }
+ expect(result.page, 1, reason: reason); // default
+ expect(result.products, isNotNull, reason: reason);
+ for (final Product product in result.products!) {
+ expect(product.statesTags, isNotNull);
+ expect(product.statesTags!, contains('en:to-be-completed'));
+ }
+ return result.count;
+ }
+
+ Future getCountForAllLanguages(
+ final OpenFoodFactsCountry country,
+ final String store,
+ ) async {
+ final List languages = [
+ OpenFoodFactsLanguage.ENGLISH,
+ OpenFoodFactsLanguage.FRENCH,
+ OpenFoodFactsLanguage.ITALIAN,
+ ];
+ int? result;
+ for (final OpenFoodFactsLanguage language in languages) {
+ final int? count = await getCount(country, language, store);
+ if (result != null) {
+ expect(count, result, reason: language.toString());
+ }
+ result = count;
+ }
+ return result!;
+ }
+
+ Future checkTypeCount(
+ final OpenFoodFactsCountry country,
+ final String store,
+ final int minimalExpectedCount,
+ ) async {
+ final int count = await getCountForAllLanguages(country, store);
+ expect(count, greaterThanOrEqualTo(minimalExpectedCount));
+ }
+
+ test(
+ 'in France',
+ () async => checkTypeCount(
+ OpenFoodFactsCountry.FRANCE,
+ 'Carrefour',
+ // 2023-08-12: was 14778
+ 10000,
+ ));
+
+ test(
+ 'in Italy',
+ () async => checkTypeCount(
+ OpenFoodFactsCountry.ITALY,
+ 'Carrefour',
+ // 2023-07-09: was 2394
+ 1500,
+ ));
+
+ test(
+ 'in Spain',
+ () async => checkTypeCount(
+ OpenFoodFactsCountry.SPAIN,
+ 'El Corte Inglès',
+ // 2023-07-09: was 608
+ 500,
+ ));
+ },
+ timeout: Timeout(
+ // some tests can be slow here
+ Duration(seconds: 180),
+ ));
+
+ group('$OpenFoodAPIClient get user products', () {
+ const String userId = 'monsieurtanuki';
+ // should be big enough to get everything on page1
+ const int pageSize = 100;
+ final String toBeCompletedTag = ProductState.COMPLETED.toBeCompletedTag;
+
+ Future getCount(
+ final TagFilterType type,
+ final OpenFoodFactsLanguage language,
+ final bool toBeCompleted, {
+ final void Function(Product)? additionalCheck,
+ }) async {
+ final String reason = '($language, $type)';
+ final ProductSearchQueryConfiguration configuration =
+ ProductSearchQueryConfiguration(
+ parametersList: [
+ TagFilter.fromType(tagFilterType: type, tagName: userId),
+ PageSize(size: pageSize),
+ if (toBeCompleted)
+ TagFilter.fromType(
+ tagFilterType: TagFilterType.STATES,
+ tagName: toBeCompletedTag,
+ ),
+ ],
+ language: language,
+ fields: [
+ ProductField.BARCODE,
+ ProductField.STATES_TAGS,
+ ],
+ version: ProductQueryVersion.v3,
+ );
+
+ final SearchResult result;
+ try {
+ result = await searchProductsInProd(
+ configuration,
+ );
+ } catch (e) {
+ fail('Could not retrieve data for $reason: $e');
+ }
+ expect(result.page, 1, reason: reason); // default
+ expect(result.pageSize, pageSize, reason: reason);
+ expect(result.products, isNotNull, reason: reason);
+ expect(result.products!.length, result.pageCount, reason: reason);
+ if (additionalCheck != null) {
+ for (final Product product in result.products!) {
+ additionalCheck(product);
+ }
+ }
+ return result.pageCount!;
+ }
+
+ Future getCountForAllLanguages(
+ final TagFilterType type,
+ final bool toBeCompleted, {
+ final void Function(Product)? additionalCheck,
+ }) async {
+ final List languages = [
+ OpenFoodFactsLanguage.ENGLISH,
+ OpenFoodFactsLanguage.FRENCH,
+ OpenFoodFactsLanguage.ITALIAN,
+ ];
+ int? result;
+ for (final OpenFoodFactsLanguage language in languages) {
+ final int count = await getCount(
+ type,
+ language,
+ toBeCompleted,
+ additionalCheck: additionalCheck,
+ );
+ if (result != null) {
+ expect(count, result, reason: language.toString());
+ }
+ result = count;
+ }
+ return result!;
+ }
+
+ Future checkTypeCount(
+ final TagFilterType type,
+ final int minimalExpectedCount, {
+ final void Function(Product)? additionalCheck,
+ final bool toBeCompleted = false,
+ }) async {
+ final int count = await getCountForAllLanguages(
+ type,
+ toBeCompleted,
+ additionalCheck: additionalCheck,
+ );
+ expect(count, greaterThanOrEqualTo(minimalExpectedCount));
+ }
+
+ test(
+ 'contributor',
+ () async => checkTypeCount(TagFilterType.CREATOR, 2) // as of 20221229
+ ,
+ );
+
+ test(
+ 'informer',
+ () async =>
+ await checkTypeCount(TagFilterType.INFORMERS, 73) // as of 20221229
+ ,
+ );
+
+ test(
+ 'photographer',
+ () async =>
+ checkTypeCount(TagFilterType.PHOTOGRAPHERS, 48) // as of 20221229
+ ,
+ );
+
+ test(
+ 'to be completed',
+ () async => checkTypeCount(
+ TagFilterType.INFORMERS, 0, // you never know...
+ toBeCompleted: true,
+ additionalCheck: (final Product product) {
+ expect(product.statesTags, isNotNull);
+ expect(product.statesTags, contains(toBeCompletedTag));
+ },
+ ),
+ );
+ }, timeout: Timeout(Duration(seconds: 300)));
+
+ group('$OpenFoodAPIClient get new packagings field', () {
+ const String barcode = '3661344723290';
+ const String searchTerms = 'skyr les 2 vaches';
+ const OpenFoodFactsLanguage language = OpenFoodFactsLanguage.FRENCH;
+ const OpenFoodFactsCountry country = OpenFoodFactsCountry.FRANCE;
+ const ProductQueryVersion version = ProductQueryVersion.v3;
+
+ void checkProduct(final Product product) {
+ void checkLocalizedTag(final LocalizedTag? tag) {
+ expect(tag, isNotNull);
+ expect(tag!.id, isNotNull);
+ expect(tag.lcName, isNotNull);
+ }
+
+ expect(product.packagings, isNotNull);
+ expect(product.packagings!.length, greaterThanOrEqualTo(3));
+ for (final ProductPackaging packaging in product.packagings!) {
+ checkLocalizedTag(packaging.shape);
+ checkLocalizedTag(packaging.material);
+ checkLocalizedTag(packaging.recycling);
+ expect(packaging.recycling!.id, 'en:recycle');
+ }
+ }
+
+ test('as a single field on a search query', () async {
+ final SearchResult searchResult = await searchProductsInProd(
+ ProductSearchQueryConfiguration(
+ parametersList: [
+ SearchTerms(terms: [searchTerms])
+ ],
+ fields: [ProductField.PACKAGINGS, ProductField.BARCODE],
+ language: language,
+ country: country,
+ version: version,
+ ),
+ );
+ expect(searchResult.products, isNotNull);
+ expect(searchResult.products, isNotEmpty);
+ bool found = false;
+ for (final Product product in searchResult.products!) {
+ if (product.barcode != barcode) {
+ continue;
+ }
+ found = true;
+ checkProduct(product);
+ }
+ expect(found, isTrue);
+ });
+
+ test('as a part of ALL fields on a search query', () async {
+ final SearchResult searchResult = await searchProductsInProd(
+ ProductSearchQueryConfiguration(
+ parametersList: [
+ SearchTerms(terms: [searchTerms])
+ ],
+ fields: [ProductField.ALL],
+ language: language,
+ country: country,
+ version: version,
+ ),
+ );
+ expect(searchResult.products, isNotNull);
+ expect(searchResult.products, isNotEmpty);
+ bool found = false;
+ for (final Product product in searchResult.products!) {
+ if (product.barcode != barcode) {
+ continue;
+ }
+ found = true;
+ checkProduct(product);
+ }
+ expect(found, isTrue);
+ });
+
+ test('as a part of RAW fields on a search query', () async {
+ try {
+ await searchProductsInProd(
+ ProductSearchQueryConfiguration(
+ parametersList: [
+ SearchTerms(terms: [searchTerms])
+ ],
+ fields: [ProductField.RAW],
+ language: language,
+ country: country,
+ version: version,
+ ),
+ );
+ } catch (e) {
+ // In RAW mode the packagings are mere String's instead of LocalizedTag's.
+ // Therefore we expect an Exception.
+ return;
+ }
+ fail('On Raw');
+ });
+ }, timeout: Timeout(Duration(minutes: 2)));
+
+ group('$OpenFoodAPIClient matched product v2', () {
+ const int HTTP_OK = 200;
+
+ const OpenFoodFactsLanguage language = OpenFoodFactsLanguage.FRENCH;
+ OpenFoodAPIConfiguration.userAgent = TestConstants.TEST_USER_AGENT;
+ OpenFoodAPIConfiguration.globalCountry = OpenFoodFactsCountry.FRANCE;
+ OpenFoodAPIConfiguration.globalUser = TestConstants.PROD_USER;
+ OpenFoodAPIConfiguration.globalLanguages = [
+ language
+ ];
+
+ const String BARCODE_KNACKI = '7613035937420';
+ const String BARCODE_CORDONBLEU = '4000405005026';
+ const String BARCODE_ORIENTALES = '4032277007211';
+ const String BARCODE_HACK = '7613037672756';
+ const String BARCODE_SCHNITZEL = '4061458069878';
+ const String BARCODE_CHIPOLATA = '3770016162098';
+ const String BARCODE_FLEISCHWURST = '4003171036379'; // now veggie!
+ const String BARCODE_POULET = '40897837';
+ const String BARCODE_SAUCISSON = '20045456';
+ const String BARCODE_PIZZA = '4260414150470';
+ const String BARCODE_ARDECHE = '20712570';
+ const String BARCODE_CHORIZO = '8480000591074';
+
+ final List inputBarcodes = [
+ BARCODE_CHIPOLATA,
+ BARCODE_FLEISCHWURST,
+ BARCODE_KNACKI,
+ BARCODE_CORDONBLEU,
+ BARCODE_SAUCISSON,
+ BARCODE_PIZZA,
+ BARCODE_ORIENTALES,
+ BARCODE_ARDECHE,
+ BARCODE_HACK,
+ BARCODE_CHORIZO,
+ BARCODE_SCHNITZEL,
+ BARCODE_POULET,
+ ];
+ final Map expectedScores = {
+ BARCODE_KNACKI: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_CORDONBLEU: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_ORIENTALES: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_HACK: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_SCHNITZEL: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_CHIPOLATA: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_FLEISCHWURST: _Score(100, MatchedProductStatusV2.VERY_GOOD_MATCH),
+ BARCODE_POULET: _Score(0, MatchedProductStatusV2.UNKNOWN_MATCH),
+ BARCODE_SAUCISSON: _Score(0, MatchedProductStatusV2.UNKNOWN_MATCH),
+ BARCODE_PIZZA: _Score(0, MatchedProductStatusV2.DOES_NOT_MATCH),
+ BARCODE_ARDECHE: _Score(0, MatchedProductStatusV2.DOES_NOT_MATCH),
+ BARCODE_CHORIZO: _Score(0, MatchedProductStatusV2.DOES_NOT_MATCH),
+ };
+ final List expectedBarcodeOrder = [
+ BARCODE_CHIPOLATA,
+ BARCODE_FLEISCHWURST,
+ BARCODE_KNACKI,
+ BARCODE_CORDONBLEU,
+ BARCODE_ORIENTALES,
+ BARCODE_HACK,
+ BARCODE_SCHNITZEL,
+ BARCODE_SAUCISSON,
+ BARCODE_POULET,
+ BARCODE_PIZZA,
+ BARCODE_ARDECHE,
+ BARCODE_CHORIZO,
+ ];
+
+ Future> downloadProducts() async {
+ final SearchResult result = await searchProductsInProd(
+ ProductSearchQueryConfiguration(
+ parametersList: [BarcodeParameter.list(inputBarcodes)],
+ language: language,
+ fields: [ProductField.BARCODE, ProductField.ATTRIBUTE_GROUPS],
+ version: ProductQueryVersion.v3,
+ ),
+ );
+ expect(result.count, expectedScores.keys.length);
+ expect(result.page, 1);
+ expect(result.products, isNotNull);
+ final List products = result.products!;
+ // sorting them again by the input order
+ products.sort(
+ (final Product a, final Product b) => inputBarcodes
+ .indexOf(a.barcode!)
+ .compareTo(inputBarcodes.indexOf(b.barcode!)),
+ );
+ expect(products.length, inputBarcodes.length);
+ return products;
+ }
+
+ Future getManager() async {
+ final Map attributeImportances = {};
+ final ProductPreferencesManager manager = ProductPreferencesManager(
+ ProductPreferencesSelection(
+ setImportance: (String attributeId, String importanceIndex) async {
+ attributeImportances[attributeId] = importanceIndex;
+ },
+ getImportance: (String attributeId) =>
+ attributeImportances[attributeId] ??
+ PreferenceImportance.ID_NOT_IMPORTANT,
+ ),
+ );
+ final String languageCode = language.code;
+ final String importanceUrl =
+ AvailablePreferenceImportances.getUrl(languageCode);
+ final String attributeGroupUrl =
+ AvailableAttributeGroups.getUrl(languageCode);
+ Response response;
+ response = await get(Uri.parse(importanceUrl));
+ expect(response.statusCode, HTTP_OK);
+ final String preferenceImportancesString = response.body;
+ response = await get(Uri.parse(attributeGroupUrl));
+ expect(response.statusCode, HTTP_OK);
+ final String attributeGroupsString = response.body;
+ manager.availableProductPreferences =
+ AvailableProductPreferences.loadFromJSONStrings(
+ preferenceImportancesString: preferenceImportancesString,
+ attributeGroupsString: attributeGroupsString,
+ );
+ await manager.setImportance(
+ Attribute.ATTRIBUTE_VEGETARIAN,
+ PreferenceImportance.ID_MANDATORY,
+ );
+ return manager;
+ }
+
+ test('matched product', () async {
+ final ProductPreferencesManager manager = await getManager();
+
+ final List products = await downloadProducts();
+
+ final List actuals =
+ MatchedProductV2.sort(products, manager);
+
+ expect(actuals.length, expectedBarcodeOrder.length);
+ for (int i = 0; i < actuals.length; i++) {
+ final MatchedProductV2 matched = actuals[i];
+ final String barcode = expectedBarcodeOrder[i];
+ expect(matched.product.barcode, barcode);
+ expect(matched.barcode, barcode);
+ expect(expectedScores[barcode], isNotNull);
+ final _Score score = expectedScores[barcode]!;
+ expect(matched.status, score.status);
+ expect(matched.score, score.score);
+ }
+ });
+
+ test('matched score', () async {
+ final ProductPreferencesManager manager = await getManager();
+
+ final List products = await downloadProducts();
+
+ final List actuals = [];
+ for (final Product product in products) {
+ actuals.add(MatchedScoreV2(product, manager));
+ }
+ MatchedScoreV2.sort(actuals);
+
+ expect(actuals.length, expectedBarcodeOrder.length);
+ for (int i = 0; i < actuals.length; i++) {
+ final MatchedScoreV2 matched = actuals[i];
+ final String barcode = expectedBarcodeOrder[i];
+ expect(matched.barcode, barcode);
+ expect(expectedScores[barcode], isNotNull);
+ final _Score score = expectedScores[barcode]!;
+ expect(matched.status, score.status);
+ expect(matched.score, score.score);
+ }
+ });
+ });
}