diff --git a/.code-samples.meilisearch.yaml b/.code-samples.meilisearch.yaml index 8af0f272..db106fe8 100644 --- a/.code-samples.meilisearch.yaml +++ b/.code-samples.meilisearch.yaml @@ -2,76 +2,25 @@ # Every example written here will be automatically fetched by # the documentation and the landing page on build --- -search_parameter_guide_show_ranking_score_1: |- - await client.index('movies').search('dragon', SearchQuery(showRankingScore: true)); -facet_search_1: |- - await client.index('books').facetSearch(FacetSearchQuery( - facetQuery: 'fiction', - facetName: 'genres', - filter: 'rating > 3' - ), - ); -facet_search_2: |- - await client.index('books').updateFaceting(Faceting(sortFacetValuesBy: {'genres': 'count'})); -facet_search_3: |- - await client.index('books').facetSearch(FacetSearchQuery( - facetQuery: 'c', - facetName: 'genres', - ), - ); -search_parameter_guide_attributes_to_search_on_1: |- - await client.index('movies').search('adventure', SearchQuery(attributesToSearchOn: ['overview'])); -get_documents_post_1: |- - await client.index('movies').getDocuments( - params: DocumentsQuery( - filter: '(rating > 3 AND (genres = Adventure OR genres = Fiction)) AND language = English', - fields: ['title', 'genres', 'rating', 'language'], - limit: 3 - ) - ); -delete_documents_by_filter_1: |- - await client.index('movies') - .deleteDocuments(DeleteDocumentsQuery(filter: 'genres = action OR genres = adventure')); -get_documents_1: |- - await client.index('movies') - .getDocuments(params: DocumentsQuery(limit: 2, filter: 'genres = action')); -multi_search_1: |- - await client.multiSearch(MultiSearchQuery(queries: [ - IndexSearchQuery( - query: 'pooh', - indexUid: 'movies', - limit: 5 - ), - IndexSearchQuery( - query: 'nemo', - indexUid: 'movies', - limit: 5 - ), - IndexSearchQuery( - query: 'us', - indexUid: 'movies_ratings' - ), - ])); -faceted_search_update_settings_1: |- - await client.index('movie_ratings').updateFilterableAttributes(['genres', 'rating', 'language']); -search_parameter_guide_facet_stats_1: |- - await client.index('movie_ratings').search('Batman', SearchQuery(facets: ['genres', 'rating'])); -faceted_search_1: |- - await client.index('books').search('', SearchQuery(facets: ['genres', 'rating', 'language'])); -filtering_guide_nested_1: |- - await client.index('movie_ratings').search('thriller', SearchQuery(filter: 'rating.users >= 90')); -sorting_guide_sort_nested_1: |- - await client.index('movie_ratings').search('thriller', SearchQuery(sort: ['rating.users:asc'])); -async_guide_filter_by_date_1: |- - await client.getTasks(params: TasksQuery(afterEnqueuedAt: DateTime(2020, 10, 11, 11, 49, 53))); -async_guide_multiple_filters_1: |- - await client.getTasks(params: TasksQuery(indexUids: ['movies'], types: ['documentAdditionOrUpdate','documentDeletion'], statuses: ['processing'])); -async_guide_filter_by_ids_1: |- - await client.getTasks(params: TasksQuery(uids: [5, 10, 13])); -async_guide_filter_by_statuses_1: |- - await client.getTasks(params: TasksQuery(statuses: ['failed', 'canceled'])); -async_guide_filter_by_types_1: |- - await client.getTasks(params: TasksQuery(types: ['dumpCreation', 'indexSwap'])); +search_parameter_guide_show_ranking_score_1: "await client\n .index('movies')\n .search('dragon', SearchQuery(showRankingScore: true));" +facet_search_1: "await client.index('books').facetSearch(\n FacetSearchQuery(\n facetQuery: 'fiction',\n facetName: 'genres',\n filter: 'rating > 3',\n ),\n );" +facet_search_2: "await client.index('books').updateFaceting(\n Faceting(\n sortFacetValuesBy: {\n 'genres': FacetingSortTypes.count,\n },\n ),\n );" +facet_search_3: "await client.index('books').facetSearch(\n FacetSearchQuery(\n facetQuery: 'c',\n facetName: 'genres',\n ),\n );" +search_parameter_guide_attributes_to_search_on_1: "await client.index('books').facetSearch(\n FacetSearchQuery(\n facetQuery: 'c',\n facetName: 'genres',\n ),\n );" +get_documents_post_1: "await client.index('movies').getDocuments(\n params: DocumentsQuery(\n filterExpression: Meili.and([\n 'language'.toMeiliAttribute().eq('English'.toMeiliValue()),\n Meili.and([\n 'rating'.toMeiliAttribute().gt(3.toMeiliValue()),\n Meili.or([\n 'genres'.toMeiliAttribute().eq('Adventure'.toMeiliValue()),\n 'genres'.toMeiliAttribute().eq('Fiction'.toMeiliValue()),\n ]),\n ]),\n ]),\n fields: ['title', 'genres', 'rating', 'language'],\n limit: 3,\n ),\n );" +delete_documents_by_filter_1: "await client.index('movies').deleteDocuments(\n DeleteDocumentsQuery(\n filterExpression: Meili.or([\n Meili.attr('genres').eq(Meili.value('action')),\n Meili.attr('genres').eq(Meili.value('adventure')),\n ]),\n ),\n );" +get_documents_1: "await client.index('movies').getDocuments(\n params: DocumentsQuery(\n limit: 2,\n filter: Meili.attr('genres').eq('action'.toMeiliValue()),\n ),\n );" +multi_search_1: "await client.multiSearch(MultiSearchQuery(queries: [\n IndexSearchQuery(query: 'pooh', indexUid: 'movies', limit: 5),\n IndexSearchQuery(query: 'nemo', indexUid: 'movies', limit: 5),\n IndexSearchQuery(query: 'us', indexUid: 'movies_ratings'),\n]));" +faceted_search_update_settings_1: "await client\n .index('movie_ratings')\n .updateFilterableAttributes(['genres', 'rating', 'language']);" +search_parameter_guide_facet_stats_1: "await client\n .index('movie_ratings')\n .search('Batman', SearchQuery(facets: ['genres', 'rating']));" +faceted_search_1: "await client\n .index('books')\n .search('', SearchQuery(facets: ['genres', 'rating', 'language']));" +filtering_guide_nested_1: "await client.index('movie_ratings').search(\n 'thriller',\n SearchQuery(\n filterExpression: Meili.gte(\n \/\/or Meili.attr('rating.users')\n \/\/or 'rating.users'.toMeiliAttribute()\n Meili.attrFromParts(['rating', 'users']),\n Meili.value(90),\n ),\n ),\n );" +sorting_guide_sort_nested_1: "await client\n .index('movie_ratings')\n .search('thriller', SearchQuery(sort: ['rating.users:asc']));" +async_guide_filter_by_date_1: "await client.getTasks(\n params: TasksQuery(\n afterEnqueuedAt: DateTime(2020, 10, 11, 11, 49, 53),\n ),\n);" +async_guide_multiple_filters_1: "await client.getTasks(\n params: TasksQuery(\n indexUids: ['movies'],\n types: ['documentAdditionOrUpdate', 'documentDeletion'],\n statuses: ['processing'],\n ),\n);" +async_guide_filter_by_ids_1: "await client.getTasks(\n params: TasksQuery(\n uids: [5, 10, 13],\n ),\n);" +async_guide_filter_by_statuses_1: "await client.getTasks(\n params: TasksQuery(\n statuses: ['failed', 'canceled'],\n ),\n);" +async_guide_filter_by_types_1: "await client.getTasks(\n params: TasksQuery(\n types: ['dumpCreation', 'indexSwap'],\n ),\n);" async_guide_filter_by_index_uids_1: |- await client.getTasks(params: TasksQuery(indexUids: ['movies'])); delete_tasks_1: |- @@ -80,48 +29,27 @@ cancel_tasks_1: |- await client.cancelTasks(params: CancelTasksQuery(uids: [1, 2])); async_guide_canceled_by_1: |- await client.getTasks(params: TasksQuery(canceledBy: [9, 15])); -swap_indexes_1: |- - await client.swapIndexes([SwapIndex(['indexA', 'indexB']), SwapIndex(['indexX', 'indexY'])]); -search_parameter_guide_hitsperpage_1: |- - await client.index('movies').search('', SearchQuery(hitsPerPage: 15)) as PaginatedSearchResult; -search_parameter_guide_page_1: |- - await client.index('movies').search('', SearchQuery(page: 2)) as PaginatedSearchResult; -synonyms_guide_1: |- - await client.index('movies').updateSynonyms({ - 'great': ['fantastic'], - 'fantastic': ['great'] - }); -date_guide_index_1: |- - import 'dart:io'; - import 'dart:convert'; - - final json = await File('games.json').readAsString(); - - await client.index('games').addDocumentsJson(json); -date_guide_filterable_attributes_1: |- - await client.index('games').updateFilterableAttributes(['release_timestamp']); -date_guide_filter_1: |- - await client.index('games').search('', - SearchQuery(filter: - 'release_timestamp >= 1514761200 AND release_timestamp < 1672527600')); -date_guide_sortable_attributes_1: |- - await client.index('games').updateSortableAttributes(['release_timestamp']); -date_guide_sort_1: |- - await client.index('games').search('', SearchQuery(sort: ['release_timestamp:desc'])); +swap_indexes_1: "await client.swapIndexes([\n SwapIndex(['indexA', 'indexB']),\n SwapIndex(['indexX', 'indexY']),\n]);" +search_parameter_guide_hitsperpage_1: "await client\n .index('movies')\n .search('', SearchQuery(hitsPerPage: 15))\n .asPaginatedResult();" +search_parameter_guide_page_1: "await client\n .index('movies')\n .search('', SearchQuery(page: 2))\n .asPaginatedResult();" +synonyms_guide_1: "await client.index('movies').updateSynonyms({\n 'great': ['fantastic'],\n 'fantastic': ['great'],\n});" +date_guide_index_1: "\/\/import 'dart:io';\n\/\/import 'dart:convert';\nfinal json = await File('games.json').readAsString();\nawait client.index('games').addDocumentsJson(json);" +date_guide_filterable_attributes_1: "await client\n .index('games')\n .updateFilterableAttributes(['release_timestamp']);" +date_guide_filter_1: "await client.index('games').search(\n '',\n SearchQuery(\n filterExpression: Meili.and([\n Meili.gte(\n 'release_timestamp'.toMeiliAttribute(),\n Meili.value(DateTime(2017, 12, 31, 23, 0)),\n ),\n Meili.lt(\n 'release_timestamp'.toMeiliAttribute(),\n Meili.value(DateTime(2022, 12, 31, 23, 0)),\n ),\n ]),\n ),\n );" +date_guide_sortable_attributes_1: "await client\n .index('games')\n .updateSortableAttributes(['release_timestamp']);" +date_guide_sort_1: "await client\n .index('games')\n .search('', SearchQuery(sort: ['release_timestamp:desc']));" get_all_tasks_paginating_1: |- await client.getTasks(params: TasksQuery(limit: 2, from: 10)); get_all_tasks_paginating_2: |- await client.getTasks(params: TasksQuery(limit: 2, from: 8)); get_pagination_settings_1: |- await client.index('movies').getPagination(); -update_pagination_settings_1: |- - await client.index('books').updatePagination(Pagination(maxTotalHits: 100)); +update_pagination_settings_1: "await client\n .index('books')\n .updatePagination(Pagination(maxTotalHits: 100));" reset_pagination_settings_1: |- await client.index('movies').resetPagination(); get_faceting_settings_1: |- await client.index('movies').getFaceting(); -update_faceting_settings_1: |- - await client.index('books').updateFaceting(Faceting(maxValuesPerFacet: 2, sortFacetValuesBy: {'*': 'alpha', 'genres': 'count'})); +update_faceting_settings_1: "await client.index('books').updateFaceting(Faceting(\n maxValuesPerFacet: 2,\n sortFacetValuesBy: {\n '*': FacetingSortTypes.alpha,\n 'genres': FacetingSortTypes.count\n }));" reset_faceting_settings_1: |- await client.index('movies').resetFaceting(); get_one_index_1: |- @@ -134,8 +62,7 @@ update_an_index_1: |- await client.index('movies').update(primaryKey: 'id'); delete_an_index_1: |- await client.index('movies').delete(); -get_one_document_1: |- - await client.index('movies').getDocument(25684, fields: ['id', 'title', 'poster', 'release_date']); +get_one_document_1: "await client.index('movies').getDocument(25684,\n fields: ['id', 'title', 'poster', 'release_date']);" add_or_replace_documents_1: |- await client.index('movies').addDocuments([ { @@ -148,21 +75,12 @@ add_or_replace_documents_1: |- 'release_date': '2019-03-23' } ]); -add_or_update_documents_1: |- - await client.index('movies').updateDocuments([ - { - 'id': 287947, - 'title': 'Shazam ⚡️', - 'genres': 'comedy' - } - ]); +add_or_update_documents_1: "await client.index('movies').updateDocuments([\n {\n 'id': 287947,\n 'title': 'Shazam ⚡️',\n 'genres': 'comedy',\n }\n]);" delete_all_documents_1: |- await client.index('movies').deleteAllDocuments(); delete_one_document_1: |- await client.index('movies').deleteDocument(25684); -delete_documents_by_batch_1: |- - await client.index('movies') - .deleteDocuments(DeleteDocumentsQuery(ids: [23488, 153738, 437035, 363869])); +delete_documents_by_batch_1: "await client.index('movies').deleteDocuments(\n DeleteDocumentsQuery(\n ids: [23488, 153738, 437035, 363869],\n ),\n );" search_post_1: |- await client.index('movies').search('American ninja'); get_task_1: |- @@ -171,40 +89,12 @@ get_all_tasks_1: |- await client.getTasks(); get_settings_1: |- await client.index('movies').getSettings(); -update_settings_1: |- - await client.index('movies').updateSettings(IndexSettings( - rankingRules: [ - 'words', - 'typo', - 'proximity', - 'attribute', - 'sort', - 'exactness', - 'release_date:desc', - 'rank:desc' - ], - distinctAttribute: 'movie_id', - searchableAttributes: ['title', 'overview', 'genres'], - displayedAttributes: ['title', 'overview', 'genres', 'release_date'], - stopWords: ['the', 'a', 'an'], - sortableAttributes: ['title', 'release_date'], - synonyms: { - 'wolverine': ['xmen', 'logan'], - 'logan': ['wolverine'], - }, - )); +update_settings_1: "await client.index('movies').updateSettings(\n IndexSettings(\n rankingRules: [\n 'words',\n 'typo',\n 'proximity',\n 'attribute',\n 'sort',\n 'exactness',\n 'release_date:desc',\n 'rank:desc'\n ],\n distinctAttribute: 'movie_id',\n searchableAttributes: ['title', 'overview', 'genres'],\n displayedAttributes: [\n 'title',\n 'overview',\n 'genres',\n 'release_date'\n ],\n stopWords: ['the', 'a', 'an'],\n sortableAttributes: ['title', 'release_date'],\n synonyms: {\n 'wolverine': ['xmen', 'logan'],\n 'logan': ['wolverine'],\n },\n ),\n );" reset_settings_1: |- await client.index('movies').resetSettings(); get_synonyms_1: |- await client.index('movies').getSynonyms(); -update_synonyms_1: |- - await client.index('movies').updateSynonyms( - { - 'wolverine': ['xmen', 'logan'], - 'logan': ['wolverine', 'xmen'], - 'wow': ['world of warcraft'] - } - ); +update_synonyms_1: "await client.index('movies').updateSynonyms({\n 'wolverine': ['xmen', 'logan'],\n 'logan': ['wolverine', 'xmen'],\n 'wow': ['world of warcraft'],\n});" reset_synonyms_1: |- await client.index('movies').resetSynonyms(); get_stop_words_1: |- @@ -215,17 +105,7 @@ reset_stop_words_1: |- await client.index('movies').resetStopWords(); get_ranking_rules_1: |- await client.index('movies').getRankingRules(); -update_ranking_rules_1: |- - await client.index('movies').updateRankingRules([ - 'words', - 'typo', - 'proximity', - 'attribute', - 'sort', - 'exactness', - 'release_date:asc', - 'rank:desc' - ]); +update_ranking_rules_1: "await client.index('movies').updateRankingRules([\n 'words',\n 'typo',\n 'proximity',\n 'attribute',\n 'sort',\n 'exactness',\n 'release_date:asc',\n 'rank:desc',\n]);" reset_ranking_rules_1: |- await client.index('movies').resetRankingRules(); get_distinct_attribute_1: |- @@ -236,32 +116,17 @@ reset_distinct_attribute_1: |- await client.index('shoes').resetDistinctAttribute(); get_filterable_attributes_1: |- await client.index('movies').getFilterableAttributes(); -update_filterable_attributes_1: |- - await client.index('movies').updateFilterableAttributes([ - 'genres', - 'director' - ]); +update_filterable_attributes_1: "await client\n .index('movies')\n .updateFilterableAttributes(['genres', 'director']);" reset_filterable_attributes_1: |- await client.index('movies').resetFilterableAttributes(); get_searchable_attributes_1: |- await client.index('movies').getSearchableAttributes(); -update_searchable_attributes_1: |- - await client.index('movies').updateSearchableAttributes([ - 'title', - 'overview', - 'genres' - ]); +update_searchable_attributes_1: "await client\n .index('movies')\n .updateSearchableAttributes(['title', 'overview', 'genres']);" reset_searchable_attributes_1: |- await client.index('movies').resetSearchableAttributes(); get_displayed_attributes_1: |- await client.index('movies').getDisplayedAttributes(); -update_displayed_attributes_1: |- - await client.index('movies').updateDisplayedAttributes([ - 'title', - 'overview', - 'genres', - 'release_date' - ]); +update_displayed_attributes_1: "await client.index('movies').updateDisplayedAttributes([\n 'title',\n 'overview',\n 'genres',\n 'release_date',\n]);" reset_displayed_attributes_1: |- await client.index('movies').resetDisplayedAttributes(); get_typo_tolerance_1: |- @@ -287,54 +152,24 @@ get_version_1: |- await client.getVersion(); distinct_attribute_guide_1: |- await client.index('jackets').updateDistinctAttribute('product_id'); -field_properties_guide_searchable_1: |- - await client.index('movies').updateSearchableAttributes(['title', 'overview', 'genres']); -field_properties_guide_displayed_1: |- - await client.index('movies').updateDisplayedAttributes(['title', 'overview', 'genres', 'release_date']); -filtering_guide_1: |- - await client - .index('movie_ratings') - .search('Avengers', SearchQuery(filter: 'release_date > 795484800')); -filtering_guide_2: |- - await client.index('movie_ratings') - .search('Batman', - SearchQuery(filter: 'release_date > 795484800 AND (director = "Tim Burton" OR director = "Christopher Nolan")' - ) - ); -filtering_guide_3: |- - await client.index('movie_ratings').search('Planet of the Apes', - SearchQuery(filter: 'release_date > 1577884550 AND (NOT director = "Tim Burton")')); +field_properties_guide_searchable_1: "await client\n .index('movies')\n .updateSearchableAttributes(['title', 'overview', 'genres']);" +field_properties_guide_displayed_1: "await client.index('movies').updateDisplayedAttributes([\n 'title',\n 'overview',\n 'genres',\n 'release_date',\n]);" +filtering_guide_1: "await client.index('movie_ratings').search(\n 'Avengers',\n SearchQuery(\n filterExpression: Meili.gt(\n Meili.attr('release_date'),\n DateTime.utc(1995, 3, 18).toMeiliValue(),\n ),\n ),\n );" +filtering_guide_2: "await client.index('movie_ratings').search(\n 'Batman',\n SearchQuery(\n filterExpression: Meili.and([\n Meili.attr('release_date')\n .gt(DateTime.utc(1995, 3, 18).toMeiliValue()),\n Meili.or([\n 'director'.toMeiliAttribute().eq('Tim Burton'.toMeiliValue()),\n 'director'\n .toMeiliAttribute()\n .eq('Christopher Nolan'.toMeiliValue()),\n ]),\n ]),\n ),\n );" +filtering_guide_3: "await client.index('movie_ratings').search(\n 'Planet of the Apes',\n SearchQuery(\n filterExpression: Meili.and([\n Meili.attr('release_date')\n .gt(DateTime.utc(2020, 1, 1, 13, 15, 50).toMeiliValue()),\n Meili.not(\n Meili.attr('director').eq(\"Tim Burton\".toMeiliValue()),\n ),\n ]),\n ),\n );" search_parameter_guide_query_1: |- await client.index('movies').search('shifu'); search_parameter_guide_offset_1: |- await client.index('movies').search('shifu', SearchQuery(offset: 1)); search_parameter_guide_limit_1: |- await client.index('movies').search('shifu', SearchQuery(limit: 2)); -search_parameter_guide_matching_strategy_1: |- - await client.index('movies').search('big fat liar', SearchQuery(matchingStrategy: MatchingStrategy.last)); -search_parameter_guide_matching_strategy_2: |- - await client.index('movies').search('big fat liar', SearchQuery(matchingStrategy: MatchingStrategy.all)); -search_parameter_guide_retrieve_1: |- - await client - .index('movies') - .search('shifu', SearchQuery(attributesToRetrieve: ['overview', 'title'])); -search_parameter_guide_crop_1: |- - await client - .index('movies') - .search('shifu', SearchQuery(attributesToCrop: ['overview'], cropLength: 5)); -search_parameter_guide_highlight_1: |- - await client - .index('movies') - .search('winter feast', SearchQuery(attributesToHighlight: ['overview'])); -search_parameter_guide_show_matches_position_1: |- - await client.index('movies').search('winter feast', SearchQuery(showMatchesPosition: true)); -add_movies_json_1: |- - import 'dart:io'; - import 'dart:convert'; - - final json = await File('movies.json').readAsString(); - - await client.index('movies').addDocumentsJson(json); +search_parameter_guide_matching_strategy_1: "await client.index('movies').search(\n 'big fat liar', SearchQuery(matchingStrategy: MatchingStrategy.last));" +search_parameter_guide_matching_strategy_2: "await client.index('movies').search(\n 'big fat liar', SearchQuery(matchingStrategy: MatchingStrategy.all));" +search_parameter_guide_retrieve_1: "await client.index('movies').search(\n 'shifu', SearchQuery(attributesToRetrieve: ['overview', 'title']));" +search_parameter_guide_crop_1: "await client.index('movies').search(\n 'shifu', SearchQuery(attributesToCrop: ['overview'], cropLength: 5));" +search_parameter_guide_highlight_1: "await client.index('movies').search(\n 'winter feast', SearchQuery(attributesToHighlight: ['overview']));" +search_parameter_guide_show_matches_position_1: "await client\n .index('movies')\n .search('winter feast', SearchQuery(showMatchesPosition: true));" +add_movies_json_1: "\/\/ import 'dart:io';\n\/\/ import 'dart:convert';\nfinal json = await File('movies.json').readAsString();\nawait client.index('movies').addDocumentsJson(json);" primary_field_guide_create_index_primary_key: |- await client.createIndex('books', primaryKey: 'reference_number'); primary_field_guide_update_document_primary_key: |- @@ -370,142 +205,58 @@ typo_tolerance_guide_4: |- ), ); await client.index('movies').updateTypoTolerance(toUpdate); -getting_started_add_documents_md: |- - ```bash - dart pub add meilisearch - ``` - - ```dart - import 'package:meilisearch/meilisearch.dart'; - import 'dart:io'; - import 'dart:convert'; - - var client = MeiliSearchClient('http://localhost:7700', 'aSampleMasterKey'); - - final json = await File('movies.json').readAsString(); - - await client.index('movies').addDocumentsJson(json); - ``` - - [About this SDK](https://github.com/meilisearch/meilisearch-dart/) -getting_started_search_md: |- - ```dart - await client.index('movies').search('botman'); - ``` - - [About this SDK](https://github.com/meilisearch/meilisearch-dart/) -getting_started_update_ranking_rules: |- - await client.index('movies').updateRankingRules([ - 'exactness', - 'words', - 'typo', - 'proximity', - 'attribute', - 'sort', - 'release_date:asc', - 'rank:desc' - ]); -getting_started_update_searchable_attributes: |- - await client.index('movies').updateSearchableAttributes([ - 'title' - ]); +getting_started_add_documents_md: "```bash\ndart pub add meilisearch\n```\n```dart\nimport 'package:meilisearch\/meilisearch.dart';\nimport 'dart:io';\nimport 'dart:convert';\nvar client = MeiliSearchClient('http:\/\/localhost:7700', 'aSampleMasterKey');\nfinal json = await File('movies.json').readAsString();\nawait client.index('movies').addDocumentsJson(json);\n```\n[About this SDK](https:\/\/github.com\/meilisearch\/meilisearch-dart\/)" +getting_started_search_md: "```dart\nawait client.index('movies').search('botman');\n```\n[About this SDK](https:\/\/github.com\/meilisearch\/meilisearch-dart\/)" +getting_started_update_ranking_rules: "await client.index('movies').updateRankingRules([\n 'exactness',\n 'words',\n 'typo',\n 'proximity',\n 'attribute',\n 'sort',\n 'release_date:asc',\n 'rank:desc',\n]);" +getting_started_update_searchable_attributes: "await client.index('movies').updateSearchableAttributes(['title']);" getting_started_update_stop_words: |- await client.index('movies').updateStopWords(['the']); getting_started_check_task_status: |- await client.getTask(0); -getting_started_synonyms: |- - await client.index('movies').updateSynonyms({ - 'winnie': ['piglet'], - 'piglet': ['winnie'] - }); -getting_started_update_displayed_attributes: |- - await client.index('movies').updateDisplayedAttributes([ - 'title', - 'overview', - 'poster' - ]); -getting_started_add_meteorites: |- - final json = await File('meteorites.json').readAsString(); - - await client.index('meteorites').addDocumentsJson(json); -getting_started_configure_settings: |- - await client.index('meteorites').updateSettings(IndexSettings( - filterableAttributes: ['mass', '_geo'], - sortableAttributes: ['mass', '_geo'] - )); -getting_started_geo_radius: |- - await client.index('meteorites').search('', SearchQuery(filter: '_geoRadius(46.9480, 7.4474, 210000)')); -getting_started_geo_point: |- - await client.index('meteorites').search('', SearchQuery(sort: ['_geoPoint(48.8583701, 2.2922926):asc'])); -getting_started_sorting: |- - await client.index('meteorites').search('', SearchQuery(sort: ['mass:asc'], filter: 'mass < 200')); -getting_started_filtering: |- - await client.index('meteorites').search('', SearchQuery(filter: 'mass < 200')); -getting_started_faceting: |- - await client.index('books').updateFaceting(Faceting(maxValuesPerFacet: 2, sortFacetValuesBy: {'*': 'count'})); +getting_started_synonyms: "await client.index('movies').updateSynonyms({\n 'winnie': ['piglet'],\n 'piglet': ['winnie'],\n});" +getting_started_update_displayed_attributes: "await client\n .index('movies')\n .updateDisplayedAttributes(['title', 'overview', 'poster']);" +getting_started_add_meteorites: "final json = await File('meteorites.json').readAsString();\nawait client.index('meteorites').addDocumentsJson(json);" +getting_started_configure_settings: "await client.index('meteorites').updateSettings(IndexSettings(\n filterableAttributes: ['mass', '_geo'],\n sortableAttributes: ['mass', '_geo']));" +getting_started_geo_radius: "await client.index('meteorites').search(\n '',\n SearchQuery(\n filterExpression: Meili.geoRadius(\n (lat: 46.9480, lng: 7.4474),\n 210000,\n ),\n ),\n );" +getting_started_geo_point: "await client.index('meteorites').search(\n '', SearchQuery(sort: ['_geoPoint(48.8583701, 2.2922926):asc']));" +getting_started_sorting: "await client.index('meteorites').search(\n '',\n SearchQuery(\n sort: ['mass:asc'],\n filterExpression: Meili.attr('mass').lt(200.toMeiliValue()),\n ),\n );" +getting_started_filtering: "await client\n .index('meteorites')\n .search('', SearchQuery(filter: 'mass < 200'));" +getting_started_faceting: "await client.index('books').updateFaceting(Faceting(\n maxValuesPerFacet: 2,\n sortFacetValuesBy: {'*': FacetingSortTypes.count}));" getting_started_typo_tolerance: |- final toUpdate = TypoTolerance( minWordSizeForTypos: MinWordSizeForTypos(oneTypo: 4), ); await client.index('movies').updateTypoTolerance(toUpdate); -getting_started_pagination: |- - await client.index('books').updatePagination(Pagination(maxTotalHits: 500)); +getting_started_pagination: "await client\n .index('books')\n .updatePagination(Pagination(maxTotalHits: 500));" filtering_update_settings_1: |- await client.index('movies').updateFilterableAttributes([ 'director', 'genres', ]); -faceted_search_walkthrough_filter_1: |- - await client.index('movies').search('thriller', SearchQuery(filter: [ - ['genres = Horror', 'genres = Mystery'], - 'director = "Jordan Peele"' - ])); +faceted_search_walkthrough_filter_1: "await client.index('movies').search(\n 'thriller',\n SearchQuery(filter: [\n ['genres = Horror', 'genres = Mystery'],\n 'director = \"Jordan Peele\"'\n ]));" post_dump_1: |- await client.createDump(); phrase_search_1: |- await client.index('movies').search('"african american" horror'); sorting_guide_update_sortable_attributes_1: |- await client.index('books').updateSortableAttributes(['author', 'price']); -sorting_guide_update_ranking_rules_1: |- - await client.index('books').updateRankingRules([ - 'words', - 'sort', - 'typo', - 'proximity', - 'attribute', - 'exactness' - ]); -sorting_guide_sort_parameter_1: |- - await client.index('books').search('science fiction', SearchQuery(sort: ['price:asc'])); -sorting_guide_sort_parameter_2: |- - await client.index('books').search('butler', SearchQuery(sort: ['author:desc'])); +sorting_guide_update_ranking_rules_1: "await client.index('books').updateRankingRules(\n ['words', 'sort', 'typo', 'proximity', 'attribute', 'exactness']);" +sorting_guide_sort_parameter_1: "await client\n .index('books')\n .search('science fiction', SearchQuery(sort: ['price:asc']));" +sorting_guide_sort_parameter_2: "await client\n .index('books')\n .search('butler', SearchQuery(sort: ['author:desc']));" get_sortable_attributes_1: |- await client.index('books').getSortableAttributes(); -update_sortable_attributes_1: |- - await client.index('books').updateSortableAttributes([ - 'price', - 'author' - ]); +update_sortable_attributes_1: "await client.index('books').updateSortableAttributes(['price', 'author']);" reset_sortable_attributes_1: |- await client.index('books').resetSortableAttributes(); -search_parameter_guide_sort_1: |- - await client.index('books').search('science fiction', SearchQuery(sort: ['price:asc'])); +search_parameter_guide_sort_1: "await client\n .index('books')\n .search('science fiction', SearchQuery(sort: ['price:asc']));" geosearch_guide_filter_settings_1: |- await client.index('restaurants').updateFilterableAttributes(['_geo']); -geosearch_guide_filter_usage_1: |- - await await client - .index('restaurants') - .search('', SearchQuery(filter: '_geoRadius(45.472735, 9.184019, 2000)')); -geosearch_guide_filter_usage_2: |- - await await client - .index('restaurants') - .search('', SearchQuery(filter: '_geoRadius(45.472735, 9.184019, 2000) AND type = pizza')); +geosearch_guide_filter_usage_1: "await client.index('restaurants').search(\n '',\n SearchQuery(\n filterExpression: Meili.geoRadius(\n (lat: 45.472735, lng: 9.184019),\n 2000,\n ),\n ),\n );" +geosearch_guide_filter_usage_2: "await client.index('restaurants').search(\n '',\n SearchQuery(\n filterExpression: Meili.and([\n Meili.geoRadius(\n (lat: 45.472735, lng: 9.184019),\n 2000,\n ),\n Meili.attr('type').eq('pizza'.toMeiliValue())\n ]),\n ),\n );" geosearch_guide_sort_settings_1: |- await client.index('restaurants').updateSortableAttributes(['_geo']); -geosearch_guide_sort_usage_1: |- - await client.index('restaurants').search('', SearchQuery(sort: ['_geoPoint(48.8561446, 2.2978204):asc'])); -geosearch_guide_sort_usage_2: |- - await client.index('restaurants').search('', SearchQuery(sort: ['_geoPoint(48.8561446, 2.2978204):asc', 'rating:desc'])); +geosearch_guide_sort_usage_1: "await client.index('restaurants').search(\n '', SearchQuery(sort: ['_geoPoint(48.8561446, 2.2978204):asc']));" +geosearch_guide_sort_usage_2: "await client.index('restaurants').search(\n '',\n SearchQuery(\n sort: ['_geoPoint(48.8561446, 2.2978204):asc', 'rating:desc']));" authorization_header_1: |- var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); await client.getKeys(); @@ -513,27 +264,14 @@ get_one_key_1: |- await client.getKey('6062abda-a5aa-4414-ac91-ecd7944c0f8d'); get_all_keys_1: |- await client.getKeys(params: KeysQuery(limit: 3)); -create_a_key_1: |- - await client.createKey( - description: 'Add documents: Products API key', - actions: ['documents.add'], - indexes: ['products'], - expiresAt: DateTime(2042, 04, 02) - ); -update_a_key_1: |- - await client.updateKey( - '6062abda-a5aa-4414-ac91-ecd7944c0f8d', - description: 'Manage documents: Products/Reviews API key', - name: 'Products/Reviews API key' - ); +create_a_key_1: "await client.createKey(\n description: 'Add documents: Products API key',\n actions: ['documents.add'],\n indexes: ['products'],\n expiresAt: DateTime(2042, 04, 02));" +update_a_key_1: "await client.updateKey(\n '6062abda-a5aa-4414-ac91-ecd7944c0f8d',\n description: 'Manage documents: Products\/Reviews API key',\n name: 'Products\/Reviews API key',\n);" delete_a_key_1: |- await client.deleteKey('6062abda-a5aa-4414-ac91-ecd7944c0f8d'); security_guide_search_key_1: |- var client = MeiliSearchClient('http://localhost:7700', 'apiKey'); await client.index('patient_medical_records').search(''); -security_guide_update_key_1: |- - var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); - await client.updateKey('74c9c733-3368-4738-bbe5-1d18a5fecb37', description: 'Default Search API Key'); +security_guide_update_key_1: "var client = MeiliSearchClient('http:\/\/localhost:7700', 'masterKey');\nawait client.updateKey(\n '74c9c733-3368-4738-bbe5-1d18a5fecb37',\n description: 'Default Search API Key',\n);" security_guide_create_key_1: |- var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); await client.createKey( @@ -548,48 +286,7 @@ security_guide_list_keys_1: |- security_guide_delete_key_1: |- var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); await client.deleteKey('ac5cd97d-5a4b-4226-a868-2d0eb6d197ab'); -tenant_token_guide_generate_sdk_1: |- - final uid = '85c3c2f9-bdd6-41f1-abd8-11fcf80e0f76'; - final apiKey = 'B5KdX2MY2jV6EXfUs6scSfmC...'; - final expiresAt = DateTime.utc(2025, 12, 20); - final searchRules = { - 'patient_medical_records': { - 'filter': 'user_id = 1' - } - }; - - final token = client.generateTenantToken( - uid, - searchRules, - apiKey: apiKey, // optional - expiresAt: expiresAt // optional - ); -tenant_token_guide_search_sdk_1: |- - final frontEndClient = MeiliSearchClient('http://localhost:7700', token); - await frontEndClient.index('patient_medical_records').search('blood test'); -landing_getting_started_1: |- - var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); - - await client.index('movies').addDocuments([ - { 'id': 1, 'title': 'Carol' }, - { 'id': 2, 'title': 'Wonder Woman' }, - { 'id': 3, 'title': 'Life of Pi' }, - { 'id': 4, 'title': 'Mad Max: Fury Road' }, - { 'id': 5, 'title': 'Moana' }, - { 'id': 6, 'title': 'Philadelphia'} - ]); -search_parameter_guide_crop_marker_1: |- - await client - .index('movies') - .search('shifu', SearchQuery(attributesToCrop: ['overview'], cropMarker: '[…]')); -search_parameter_guide_highlight_tag_1: |- - await client.index('movies').search( - 'winter feast', - SearchQuery( - attributesToHighlight: ['overview'], - highlightPreTag: '', - highlightPostTag: '')); -geosearch_guide_filter_usage_3: |- - await client.index('restaurants') - .search('', SearchQuery(filter: '_geoBoundingBox([45.494181, 9.214024], [45.449484, 9.179175])')), - }); +search_parameter_guide_crop_marker_1: "await client.index('movies').search(\n 'shifu',\n SearchQuery(\n attributesToCrop: ['overview'],\n cropMarker: '[…]',\n ),\n );" +search_parameter_guide_highlight_tag_1: "await client.index('movies').search(\n 'winter feast',\n SearchQuery(\n attributesToHighlight: ['overview'],\n highlightPreTag: '',\n highlightPostTag: '<\/span>',\n ),\n );" +geosearch_guide_filter_usage_3: "await client.index('restaurants').search(\n '',\n SearchQuery(\n filter:\n '_geoBoundingBox([45.494181, 9.214024], [45.449484, 9.179175])'));\n});" +search_get_1: await client.index('movies').search('American ninja'); diff --git a/.github/release-draft-template.yml b/.github/release-draft-template.yml index e8adefa2..c1c4ac77 100644 --- a/.github/release-draft-template.yml +++ b/.github/release-draft-template.yml @@ -16,6 +16,10 @@ categories: label: 'bug' - title: '🔒 Security' label: 'security' + - title: '⚙️ Maintenance/misc' + label: + - 'maintenance' + - 'documentation' template: | $CHANGES diff --git a/.github/workflows/pre-release-tests.yml b/.github/workflows/pre-release-tests.yml index a0803ffc..5d7e43ec 100644 --- a/.github/workflows/pre-release-tests.yml +++ b/.github/workflows/pre-release-tests.yml @@ -27,4 +27,4 @@ jobs: - name: Meilisearch (${{ env.MEILISEARCH_VERSION }}) setup with Docker run: docker run -d -p 7700:7700 getmeili/meilisearch:${{ env.MEILISEARCH_VERSION }} meilisearch --master-key=masterKey --no-analytics - name: Run integration tests - run: docker run --net="host" -v $PWD:/package -w /package dart:${{ matrix.version }} /bin/sh -c 'dart pub get && dart run test' + run: docker run --net="host" -v $PWD:/package -w /package dart:${{ matrix.version }} /bin/sh -c 'dart pub get && dart pub get -C tool && dart run test' diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index dbe52386..e10934b0 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -41,6 +41,7 @@ jobs: - name: Install dependencies run: | dart pub get + dart pub get -C tool dart pub global activate coverage - name: Run integration tests run: dart test --concurrency=4 --reporter=github --coverage=./coverage/reports @@ -61,7 +62,9 @@ jobs: steps: - uses: actions/checkout@v4 - name: Install dependencies - run: dart pub get + run: | + dart pub get + dart pub get -C tool - name: Run linter run: dart analyze --fatal-infos && dart format . --set-exit-if-changed @@ -75,6 +78,19 @@ jobs: with: config_file: .yamllint.yml + check-code-samples: + name: check .code-samples.meilisearch.yaml + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v3 + - uses: dart-lang/setup-dart@v1 + with: + sdk: '3.0.0' + - name: check if samples changed + run: | + dart pub get + dart pub get -C tool + dart run ./tool/bin/meili.dart update-samples --fail-on-change pana: runs-on: ubuntu-latest timeout-minutes: 10 diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index c7903028..eecd17b4 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -9,6 +9,7 @@ First of all, thank you for contributing to Meilisearch! The goal of this docume - [Requirements ](#requirements-) - [Setup ](#setup-) - [Tests and Linter ](#tests-and-linter-) + - [Updating code samples](#updating-code-samples) - [Git Guidelines](#git-guidelines) - [Git Branches ](#git-branches-) - [Git Commits ](#git-commits-) @@ -76,6 +77,31 @@ dart test dart analyze ``` +### Updating code samples + +Some PRs require updating the code samples (found in `.code-samples.meilisearch.yaml`), this is done automatically using code excerpts, which are actual pieces of code subject to testing and linting. + +A lot of them are placed in `test/code_samples.dart`. + +Also most of the tests in that file are skipped, since they are mostly duplicated in other test files. + +The process to define a new code sample is as follows: +1. Add the piece of code in `test/code_samples.dart` +2. surround it with `#docregion key` and `#enddocregion`, e.g. + ``` + // #docregion meilisearch_contributing_1 + final client = MeilisearchClient(); + anything(); + // #enddocregion + ``` +3. run this command to update the code samples + ```bash + dart run ./tool/bin/meili.dart update-samples + ``` +4. to test if the code samples are updated correctly, run: + ```bash + dart run ./tool/bin/meili.dart update-samples --fail-on-change + ``` ## Git Guidelines ### Git Branches diff --git a/README.md b/README.md index a1723d24..85e4a39c 100644 --- a/README.md +++ b/README.md @@ -28,7 +28,7 @@ **Meilisearch** is an open-source search engine. [Learn more about Meilisearch.](https://github.com/meilisearch/meilisearch) -## Table of Contents +## Table of Contents - [📖 Documentation](#-documentation) - [⚡ Supercharge your Meilisearch experience](#-supercharge-your-meilisearch-experience) diff --git a/lib/src/filter_builder/values.dart b/lib/src/filter_builder/values.dart index c1a8a19f..db1f6d42 100644 --- a/lib/src/filter_builder/values.dart +++ b/lib/src/filter_builder/values.dart @@ -29,8 +29,10 @@ class MeiliDateTimeValueExpression extends MeiliValueExpressionBase { "DateTime passed to Meili must be in UTC to avoid inconsistency accross multiple devices", ); + /// Unix epoch time is seconds since epoch @override - String transform() => value.millisecondsSinceEpoch.toString(); + String transform() => + (value.millisecondsSinceEpoch / 1000).floor().toString(); @override bool operator ==(Object other) { diff --git a/test/code_samples.dart b/test/code_samples.dart new file mode 100644 index 00000000..3de892be --- /dev/null +++ b/test/code_samples.dart @@ -0,0 +1,888 @@ +// ignore_for_file: unused_element + +import 'dart:io'; + +import 'package:meilisearch/meilisearch.dart'; +import 'package:test/test.dart'; + +import 'utils/client.dart'; + +void main() { + // this file hosts some code samples referenced in + // .code-samples.meilisearch.yaml + // it's subject to tests, lint rules, deprecation notices, etc... + group('code samples', () { + setUpClient(); + + test('excerpts', () async { + void a1() async { + // #docregion typo_tolerance_guide_1 + final toUpdate = TypoTolerance(enabled: false); + await client.index('movies').updateTypoTolerance(toUpdate); + // #enddocregion + } + + void a2() async { + // #docregion typo_tolerance_guide_2 + final toUpdate = TypoTolerance( + disableOnAttributes: ['title'], + ); + await client.index('movies').updateTypoTolerance(toUpdate); + // #enddocregion + } + + void a3() async { + // #docregion typo_tolerance_guide_3 + final toUpdate = TypoTolerance( + disableOnWords: ['shrek'], + ); + await client.index('movies').updateTypoTolerance(toUpdate); + // #enddocregion + } + + void a4() async { + // #docregion typo_tolerance_guide_4 + final toUpdate = TypoTolerance( + minWordSizeForTypos: MinWordSizeForTypos( + oneTypo: 4, + twoTypos: 10, + ), + ); + await client.index('movies').updateTypoTolerance(toUpdate); + // #enddocregion + } + + void a8() async { + // #docregion getting_started_add_meteorites + final json = await File('meteorites.json').readAsString(); + + await client.index('meteorites').addDocumentsJson(json); + // #enddocregion + } + + void a10() async { + // #docregion add_movies_json_1 + // import 'dart:io'; + // import 'dart:convert'; + final json = await File('movies.json').readAsString(); + await client.index('movies').addDocumentsJson(json); + // #enddocregion + } + + void a11() async { + // #docregion security_guide_delete_key_1 + var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); + await client.deleteKey('ac5cd97d-5a4b-4226-a868-2d0eb6d197ab'); + // #enddocregion + } + + void a12() async { + // #docregion security_guide_list_keys_1 + var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); + await client.getKeys(); + // #enddocregion + } + + void a13() async { + // #docregion security_guide_create_key_1 + var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); + await client.createKey( + description: 'Search patient records key', + actions: ['search'], + indexes: ['patient_medical_records'], + expiresAt: DateTime(2023, 01, 01), + ); + // #enddocregion + } + + void a14() async { + // #docregion authorization_header_1 + var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); + await client.getKeys(); + // #enddocregion + } + + void a15() async { + // #docregion security_guide_search_key_1 + var client = MeiliSearchClient('http://localhost:7700', 'apiKey'); + await client.index('patient_medical_records').search(''); + // #enddocregion + } + + void a16() async { + // #docregion security_guide_update_key_1 + var client = MeiliSearchClient('http://localhost:7700', 'masterKey'); + await client.updateKey( + '74c9c733-3368-4738-bbe5-1d18a5fecb37', + description: 'Default Search API Key', + ); + // #enddocregion + } + + // #docregion date_guide_index_1 + //import 'dart:io'; + //import 'dart:convert'; + + final json = await File('games.json').readAsString(); + + await client.index('games').addDocumentsJson(json); + // #enddocregion + + // #docregion date_guide_filterable_attributes_1 + await client + .index('games') + .updateFilterableAttributes(['release_timestamp']); + // #enddocregion + + // #docregion date_guide_filter_1 + await client.index('games').search( + '', + SearchQuery( + filterExpression: Meili.and([ + Meili.gte( + 'release_timestamp'.toMeiliAttribute(), + Meili.value(DateTime(2017, 12, 31, 23, 0)), + ), + Meili.lt( + 'release_timestamp'.toMeiliAttribute(), + Meili.value(DateTime(2022, 12, 31, 23, 0)), + ), + ]), + ), + ); + // #enddocregion + + // #docregion date_guide_sortable_attributes_1 + await client + .index('games') + .updateSortableAttributes(['release_timestamp']); + // #enddocregion + + // #docregion date_guide_sort_1 + await client + .index('games') + .search('', SearchQuery(sort: ['release_timestamp:desc'])); + // #enddocregion + + // #docregion get_all_tasks_paginating_1 + await client.getTasks(params: TasksQuery(limit: 2, from: 10)); + // #enddocregion + + // #docregion get_all_tasks_paginating_2 + await client.getTasks(params: TasksQuery(limit: 2, from: 8)); + // #enddocregion + + // #docregion get_pagination_settings_1 + await client.index('movies').getPagination(); + // #enddocregion + + // #docregion update_pagination_settings_1 + await client + .index('books') + .updatePagination(Pagination(maxTotalHits: 100)); + // #enddocregion + + // #docregion reset_pagination_settings_1 + await client.index('movies').resetPagination(); + // #enddocregion + + // #docregion get_faceting_settings_1 + await client.index('movies').getFaceting(); + // #enddocregion + + // #docregion update_faceting_settings_1 + await client.index('books').updateFaceting(Faceting( + maxValuesPerFacet: 2, + sortFacetValuesBy: { + '*': FacetingSortTypes.alpha, + 'genres': FacetingSortTypes.count + })); + // #enddocregion + + // #docregion reset_faceting_settings_1 + await client.index('movies').resetFaceting(); + // #enddocregion + + // #docregion get_one_index_1 + await client.getIndex('movies'); + // #enddocregion + + // #docregion list_all_indexes_1 + await client.getIndexes(params: IndexesQuery(limit: 3)); + // #enddocregion + + // #docregion create_an_index_1 + await client.createIndex('movies', primaryKey: 'id'); + // #enddocregion + + // #docregion update_an_index_1 + await client.index('movies').update(primaryKey: 'id'); + // #enddocregion + + // #docregion delete_an_index_1 + await client.index('movies').delete(); + // #enddocregion + + // #docregion get_one_document_1 + await client.index('movies').getDocument(25684, + fields: ['id', 'title', 'poster', 'release_date']); + // #enddocregion + + // #docregion add_or_replace_documents_1 + await client.index('movies').addDocuments([ + { + 'id': 287947, + 'title': 'Shazam', + 'poster': + 'https://image.tmdb.org/t/p/w1280/xnopI5Xtky18MPhK40cZAGAOVeV.jpg', + 'overview': + 'A boy is given the ability to become an adult superhero in times of need with a single magic word.', + 'release_date': '2019-03-23' + } + ]); + // #enddocregion + + // #docregion add_or_update_documents_1 + await client.index('movies').updateDocuments([ + { + 'id': 287947, + 'title': 'Shazam ⚡️', + 'genres': 'comedy', + } + ]); + // #enddocregion + + // #docregion delete_all_documents_1 + await client.index('movies').deleteAllDocuments(); + // #enddocregion + + // #docregion delete_one_document_1 + await client.index('movies').deleteDocument(25684); + // #enddocregion + + // #docregion delete_documents_by_batch_1 + await client.index('movies').deleteDocuments( + DeleteDocumentsQuery( + ids: [23488, 153738, 437035, 363869], + ), + ); + // #enddocregion + + // #docregion search_post_1 + await client.index('movies').search('American ninja'); + // #enddocregion + + // #docregion get_task_1 + await client.getTask(1); + // #enddocregion + + // #docregion get_all_tasks_1 + await client.getTasks(); + // #enddocregion + + // #docregion get_settings_1 + await client.index('movies').getSettings(); + // #enddocregion + + // #docregion update_settings_1 + await client.index('movies').updateSettings( + IndexSettings( + rankingRules: [ + 'words', + 'typo', + 'proximity', + 'attribute', + 'sort', + 'exactness', + 'release_date:desc', + 'rank:desc' + ], + distinctAttribute: 'movie_id', + searchableAttributes: ['title', 'overview', 'genres'], + displayedAttributes: [ + 'title', + 'overview', + 'genres', + 'release_date' + ], + stopWords: ['the', 'a', 'an'], + sortableAttributes: ['title', 'release_date'], + synonyms: { + 'wolverine': ['xmen', 'logan'], + 'logan': ['wolverine'], + }, + ), + ); + // #enddocregion + + // #docregion reset_settings_1 + await client.index('movies').resetSettings(); + // #enddocregion + + // #docregion get_synonyms_1 + await client.index('movies').getSynonyms(); + // #enddocregion + + // #docregion update_synonyms_1 + await client.index('movies').updateSynonyms({ + 'wolverine': ['xmen', 'logan'], + 'logan': ['wolverine', 'xmen'], + 'wow': ['world of warcraft'], + }); + // #enddocregion + + // #docregion reset_synonyms_1 + await client.index('movies').resetSynonyms(); + // #enddocregion + + // #docregion get_stop_words_1 + await client.index('movies').getStopWords(); + // #enddocregion + + // #docregion update_stop_words_1 + await client.index('movies').updateStopWords(['of', 'the', 'to']); + // #enddocregion + + // #docregion reset_stop_words_1 + await client.index('movies').resetStopWords(); + // #enddocregion + + // #docregion get_ranking_rules_1 + await client.index('movies').getRankingRules(); + // #enddocregion + + // #docregion update_ranking_rules_1 + await client.index('movies').updateRankingRules([ + 'words', + 'typo', + 'proximity', + 'attribute', + 'sort', + 'exactness', + 'release_date:asc', + 'rank:desc', + ]); + // #enddocregion + + // #docregion reset_ranking_rules_1 + await client.index('movies').resetRankingRules(); + // #enddocregion + + // #docregion get_distinct_attribute_1 + await client.index('shoes').getDistinctAttribute(); + // #enddocregion + + // #docregion update_distinct_attribute_1 + await client.index('shoes').updateDistinctAttribute('skuid'); + // #enddocregion + + // #docregion reset_distinct_attribute_1 + await client.index('shoes').resetDistinctAttribute(); + // #enddocregion + + // #docregion get_filterable_attributes_1 + await client.index('movies').getFilterableAttributes(); + // #enddocregion + + // #docregion update_filterable_attributes_1 + await client + .index('movies') + .updateFilterableAttributes(['genres', 'director']); + // #enddocregion + + // #docregion reset_filterable_attributes_1 + await client.index('movies').resetFilterableAttributes(); + // #enddocregion + + // #docregion get_searchable_attributes_1 + await client.index('movies').getSearchableAttributes(); + // #enddocregion + + // #docregion update_searchable_attributes_1 + await client + .index('movies') + .updateSearchableAttributes(['title', 'overview', 'genres']); + // #enddocregion + + // #docregion reset_searchable_attributes_1 + await client.index('movies').resetSearchableAttributes(); + // #enddocregion + + // #docregion get_displayed_attributes_1 + await client.index('movies').getDisplayedAttributes(); + // #enddocregion + + // #docregion update_displayed_attributes_1 + await client.index('movies').updateDisplayedAttributes([ + 'title', + 'overview', + 'genres', + 'release_date', + ]); + // #enddocregion + + // #docregion reset_displayed_attributes_1 + await client.index('movies').resetDisplayedAttributes(); + // #enddocregion + + // #docregion get_typo_tolerance_1 + await client.index('books').getTypoTolerance(); + // #enddocregion + + // #docregion update_typo_tolerance_1 + final toUpdate = TypoTolerance( + minWordSizeForTypos: MinWordSizeForTypos( + oneTypo: 4, + twoTypos: 10, + ), + disableOnAttributes: ['title'], + ); + await client.index('books').updateTypoTolerance(toUpdate); + // #enddocregion + + // #docregion reset_typo_tolerance_1 + await client.index('books').resetTypoTolerance(); + // #enddocregion + + // #docregion get_index_stats_1 + await client.index('movies').getStats(); + // #enddocregion + + // #docregion get_indexes_stats_1 + await client.getStats(); + // #enddocregion + + // #docregion get_health_1 + await client.health(); + // #enddocregion + + // #docregion get_version_1 + await client.getVersion(); + // #enddocregion + + // #docregion distinct_attribute_guide_1 + await client.index('jackets').updateDistinctAttribute('product_id'); + // #enddocregion + + // #docregion field_properties_guide_searchable_1 + await client + .index('movies') + .updateSearchableAttributes(['title', 'overview', 'genres']); + // #enddocregion + + // #docregion field_properties_guide_displayed_1 + await client.index('movies').updateDisplayedAttributes([ + 'title', + 'overview', + 'genres', + 'release_date', + ]); + // #enddocregion + + // #docregion filtering_guide_1 + await client.index('movie_ratings').search( + 'Avengers', + SearchQuery( + filterExpression: Meili.gt( + Meili.attr('release_date'), + DateTime.utc(1995, 3, 18).toMeiliValue(), + ), + ), + ); + // #enddocregion + + // #docregion filtering_guide_2 + await client.index('movie_ratings').search( + 'Batman', + SearchQuery( + filterExpression: Meili.and([ + Meili.attr('release_date') + .gt(DateTime.utc(1995, 3, 18).toMeiliValue()), + Meili.or([ + 'director'.toMeiliAttribute().eq('Tim Burton'.toMeiliValue()), + 'director' + .toMeiliAttribute() + .eq('Christopher Nolan'.toMeiliValue()), + ]), + ]), + ), + ); + // #enddocregion + + // #docregion filtering_guide_3 + await client.index('movie_ratings').search( + 'Planet of the Apes', + SearchQuery( + filterExpression: Meili.and([ + Meili.attr('release_date') + .gt(DateTime.utc(2020, 1, 1, 13, 15, 50).toMeiliValue()), + Meili.not( + Meili.attr('director').eq("Tim Burton".toMeiliValue()), + ), + ]), + ), + ); + // #enddocregion + + // #docregion search_parameter_guide_query_1 + await client.index('movies').search('shifu'); + // #enddocregion + + // #docregion search_parameter_guide_offset_1 + await client.index('movies').search('shifu', SearchQuery(offset: 1)); + // #enddocregion + + // #docregion search_parameter_guide_limit_1 + await client.index('movies').search('shifu', SearchQuery(limit: 2)); + // #enddocregion + + // #docregion search_parameter_guide_matching_strategy_1 + await client.index('movies').search( + 'big fat liar', SearchQuery(matchingStrategy: MatchingStrategy.last)); + // #enddocregion + + // #docregion search_parameter_guide_matching_strategy_2 + await client.index('movies').search( + 'big fat liar', SearchQuery(matchingStrategy: MatchingStrategy.all)); + // #enddocregion + + // #docregion search_parameter_guide_retrieve_1 + await client.index('movies').search( + 'shifu', SearchQuery(attributesToRetrieve: ['overview', 'title'])); + // #enddocregion + + // #docregion search_parameter_guide_crop_1 + await client.index('movies').search( + 'shifu', SearchQuery(attributesToCrop: ['overview'], cropLength: 5)); + // #enddocregion + + // #docregion search_parameter_guide_highlight_1 + await client.index('movies').search( + 'winter feast', SearchQuery(attributesToHighlight: ['overview'])); + // #enddocregion + + // #docregion search_parameter_guide_show_matches_position_1 + await client + .index('movies') + .search('winter feast', SearchQuery(showMatchesPosition: true)); + // #enddocregion + + // #docregion primary_field_guide_create_index_primary_key + await client.createIndex('books', primaryKey: 'reference_number'); + // #enddocregion + + // #docregion primary_field_guide_update_document_primary_key + await client.updateIndex('books', 'title'); + // #enddocregion + + // #docregion primary_field_guide_add_document_primary_key + await client.index('movies').addDocuments([ + { + 'reference_number': 287947, + 'title': 'Diary of a Wimpy Kid', + 'author': 'Jeff Kinney', + 'genres': ['comedy', 'humor'], + 'price': 5.00 + } + ], primaryKey: 'reference_number'); + // #enddocregion + + // #docregion getting_started_update_ranking_rules + await client.index('movies').updateRankingRules([ + 'exactness', + 'words', + 'typo', + 'proximity', + 'attribute', + 'sort', + 'release_date:asc', + 'rank:desc', + ]); + // #enddocregion + + // #docregion getting_started_update_searchable_attributes + await client.index('movies').updateSearchableAttributes(['title']); + // #enddocregion + + // #docregion getting_started_update_stop_words + await client.index('movies').updateStopWords(['the']); + // #enddocregion + + // #docregion getting_started_check_task_status + await client.getTask(0); + // #enddocregion + + // #docregion getting_started_synonyms + await client.index('movies').updateSynonyms({ + 'winnie': ['piglet'], + 'piglet': ['winnie'], + }); + // #enddocregion + + // #docregion getting_started_update_displayed_attributes + await client + .index('movies') + .updateDisplayedAttributes(['title', 'overview', 'poster']); + // #enddocregion + + // #docregion getting_started_configure_settings + await client.index('meteorites').updateSettings(IndexSettings( + filterableAttributes: ['mass', '_geo'], + sortableAttributes: ['mass', '_geo'])); + // #enddocregion + + // #docregion getting_started_geo_radius + await client.index('meteorites').search( + '', + SearchQuery( + filterExpression: Meili.geoRadius( + (lat: 46.9480, lng: 7.4474), + 210000, + ), + ), + ); + // #enddocregion + + // #docregion getting_started_geo_point + await client.index('meteorites').search( + '', SearchQuery(sort: ['_geoPoint(48.8583701, 2.2922926):asc'])); + // #enddocregion + + // #docregion getting_started_sorting + await client.index('meteorites').search( + '', + SearchQuery( + sort: ['mass:asc'], + filterExpression: Meili.attr('mass').lt(200.toMeiliValue()), + ), + ); + // #enddocregion + + // #docregion getting_started_filtering + await client + .index('meteorites') + .search('', SearchQuery(filter: 'mass < 200')); + // #enddocregion + + // #docregion getting_started_faceting + await client.index('books').updateFaceting(Faceting( + maxValuesPerFacet: 2, + sortFacetValuesBy: {'*': FacetingSortTypes.count})); + // #enddocregion + + void a9() async { + // #docregion getting_started_typo_tolerance + final toUpdate = TypoTolerance( + minWordSizeForTypos: MinWordSizeForTypos(oneTypo: 4), + ); + await client.index('movies').updateTypoTolerance(toUpdate); + // #enddocregion + } + + // #docregion getting_started_pagination + await client + .index('books') + .updatePagination(Pagination(maxTotalHits: 500)); + // #enddocregion + + // #docregion filtering_update_settings_1 + await client.index('movies').updateFilterableAttributes([ + 'director', + 'genres', + ]); + // #enddocregion + + // #docregion faceted_search_walkthrough_filter_1 + await client.index('movies').search( + 'thriller', + SearchQuery(filter: [ + ['genres = Horror', 'genres = Mystery'], + 'director = "Jordan Peele"' + ])); + // #enddocregion + + // #docregion post_dump_1 + await client.createDump(); + // #enddocregion + + // #docregion phrase_search_1 + await client.index('movies').search('"african american" horror'); + // #enddocregion + + // #docregion sorting_guide_update_sortable_attributes_1 + await client.index('books').updateSortableAttributes(['author', 'price']); + // #enddocregion + + // #docregion sorting_guide_update_ranking_rules_1 + await client.index('books').updateRankingRules( + ['words', 'sort', 'typo', 'proximity', 'attribute', 'exactness']); + // #enddocregion + + // #docregion sorting_guide_sort_parameter_1 + await client + .index('books') + .search('science fiction', SearchQuery(sort: ['price:asc'])); + // #enddocregion + + // #docregion sorting_guide_sort_parameter_2 + await client + .index('books') + .search('butler', SearchQuery(sort: ['author:desc'])); + // #enddocregion + + // #docregion get_sortable_attributes_1 + await client.index('books').getSortableAttributes(); + // #enddocregion + + // #docregion update_sortable_attributes_1 + await client.index('books').updateSortableAttributes(['price', 'author']); + // #enddocregion + + // #docregion reset_sortable_attributes_1 + await client.index('books').resetSortableAttributes(); + // #enddocregion + + // #docregion search_parameter_guide_sort_1 + await client + .index('books') + .search('science fiction', SearchQuery(sort: ['price:asc'])); + // #enddocregion + + // #docregion geosearch_guide_filter_settings_1 + await client.index('restaurants').updateFilterableAttributes(['_geo']); + // #enddocregion + + // #docregion geosearch_guide_filter_usage_1 + await client.index('restaurants').search( + '', + SearchQuery( + filterExpression: Meili.geoRadius( + (lat: 45.472735, lng: 9.184019), + 2000, + ), + ), + ); + // #enddocregion + + // #docregion geosearch_guide_filter_usage_2 + await client.index('restaurants').search( + '', + SearchQuery( + filterExpression: Meili.and([ + Meili.geoRadius( + (lat: 45.472735, lng: 9.184019), + 2000, + ), + Meili.attr('type').eq('pizza'.toMeiliValue()) + ]), + ), + ); + // #enddocregion + + // #docregion geosearch_guide_sort_settings_1 + await client.index('restaurants').updateSortableAttributes(['_geo']); + // #enddocregion + + // #docregion geosearch_guide_sort_usage_1 + await client.index('restaurants').search( + '', SearchQuery(sort: ['_geoPoint(48.8561446, 2.2978204):asc'])); + // #enddocregion + + // #docregion geosearch_guide_sort_usage_2 + await client.index('restaurants').search( + '', + SearchQuery( + sort: ['_geoPoint(48.8561446, 2.2978204):asc', 'rating:desc'])); + // #enddocregion + + // #docregion get_one_key_1 + await client.getKey('6062abda-a5aa-4414-ac91-ecd7944c0f8d'); + // #enddocregion + + // #docregion get_all_keys_1 + await client.getKeys(params: KeysQuery(limit: 3)); + // #enddocregion + + // #docregion create_a_key_1 + await client.createKey( + description: 'Add documents: Products API key', + actions: ['documents.add'], + indexes: ['products'], + expiresAt: DateTime(2042, 04, 02)); + // #enddocregion + + // #docregion update_a_key_1 + await client.updateKey( + '6062abda-a5aa-4414-ac91-ecd7944c0f8d', + description: 'Manage documents: Products/Reviews API key', + name: 'Products/Reviews API key', + ); + // #enddocregion + + // #docregion delete_a_key_1 + await client.deleteKey('6062abda-a5aa-4414-ac91-ecd7944c0f8d'); + // #enddocregion + + // #docregion search_parameter_guide_crop_marker_1 + await client.index('movies').search( + 'shifu', + SearchQuery( + attributesToCrop: ['overview'], + cropMarker: '[…]', + ), + ); + // #enddocregion + + // #docregion search_parameter_guide_highlight_tag_1 + await client.index('movies').search( + 'winter feast', + SearchQuery( + attributesToHighlight: ['overview'], + highlightPreTag: '', + highlightPostTag: '', + ), + ); + // #enddocregion + + // #docregion geosearch_guide_filter_usage_3 + await client.index('restaurants').search( + '', + SearchQuery( + filter: + '_geoBoundingBox([45.494181, 9.214024], [45.449484, 9.179175])')); + }); + // #enddocregion + // skip this test, since it's only used for generating code samples + }, skip: true); + +// unformatted examples +/* + // #docregion getting_started_search_md + ```dart + await client.index('movies').search('botman'); + ``` + + [About this SDK](https://github.com/meilisearch/meilisearch-dart/) + // #enddocregion + + // #docregion getting_started_add_documents_md + ```bash + dart pub add meilisearch + ``` + + ```dart + import 'package:meilisearch/meilisearch.dart'; + import 'dart:io'; + import 'dart:convert'; + + var client = MeiliSearchClient('http://localhost:7700', 'aSampleMasterKey'); + + final json = await File('movies.json').readAsString(); + + await client.index('movies').addDocumentsJson(json); + ``` + + [About this SDK](https://github.com/meilisearch/meilisearch-dart/) + // #enddocregion +*/ +} diff --git a/test/documents_test.dart b/test/documents_test.dart index b0c7a8ee..8023a8f1 100644 --- a/test/documents_test.dart +++ b/test/documents_test.dart @@ -557,4 +557,49 @@ void main() { }); }); }); + + test( + 'documents code samples', + () async { + // #docregion delete_documents_by_filter_1 + await client.index('movies').deleteDocuments( + DeleteDocumentsQuery( + filterExpression: Meili.or([ + Meili.attr('genres').eq(Meili.value('action')), + Meili.attr('genres').eq(Meili.value('adventure')), + ]), + ), + ); + // #enddocregion + + // #docregion get_documents_1 + await client.index('movies').getDocuments( + params: DocumentsQuery( + limit: 2, + filter: Meili.attr('genres').eq('action'.toMeiliValue()), + ), + ); + // #enddocregion + + // #docregion get_documents_post_1 + await client.index('movies').getDocuments( + params: DocumentsQuery( + filterExpression: Meili.and([ + 'language'.toMeiliAttribute().eq('English'.toMeiliValue()), + Meili.and([ + 'rating'.toMeiliAttribute().gt(3.toMeiliValue()), + Meili.or([ + 'genres'.toMeiliAttribute().eq('Adventure'.toMeiliValue()), + 'genres'.toMeiliAttribute().eq('Fiction'.toMeiliValue()), + ]), + ]), + ]), + fields: ['title', 'genres', 'rating', 'language'], + limit: 3, + ), + ); + // #enddocregion + }, + skip: true, + ); } diff --git a/test/exceptions_test.dart b/test/exceptions_test.dart index 3981a7d1..6975cbff 100644 --- a/test/exceptions_test.dart +++ b/test/exceptions_test.dart @@ -25,7 +25,7 @@ void main() { throwsA(isA().having( (error) => error.toString(), // Actual 'toString() method', // Description of the check - 'MeiliSearchApiError - message: The request returned an invalid status code of 404.', // Expected + contains('404'), // Expected )), ); }); diff --git a/test/filter_builder_test.dart b/test/filter_builder_test.dart index 86431fce..fe962c83 100644 --- a/test/filter_builder_test.dart +++ b/test/filter_builder_test.dart @@ -75,7 +75,7 @@ void main() { }); test("Dates", () { final testData = [ - [DateTime.utc(1999, 12, 14, 18, 53, 56), '945197636000'], + [DateTime.utc(1999, 12, 14, 18, 53, 56), '945197636'], ]; for (var element in testData) { diff --git a/test/multi_index_search_test.dart b/test/multi_index_search_test.dart index 7aabbc7b..64abd3f7 100644 --- a/test/multi_index_search_test.dart +++ b/test/multi_index_search_test.dart @@ -46,4 +46,14 @@ void main() { expect(result.results.last.hits.length, 2); }); }); + + test('code samples', () async { + // #docregion multi_search_1 + await client.multiSearch(MultiSearchQuery(queries: [ + IndexSearchQuery(query: 'pooh', indexUid: 'movies', limit: 5), + IndexSearchQuery(query: 'nemo', indexUid: 'movies', limit: 5), + IndexSearchQuery(query: 'us', indexUid: 'movies_ratings'), + ])); + // #enddocregion + }, skip: true); } diff --git a/test/search_test.dart b/test/search_test.dart index 0fbb14f5..45a17688 100644 --- a/test/search_test.dart +++ b/test/search_test.dart @@ -604,4 +604,102 @@ void main() { ); }); }); + + test('search code samples', () async { + // #docregion search_get_1 + await client.index('movies').search('American ninja'); + // #enddocregion + + // #docregion search_parameter_guide_show_ranking_score_1 + await client + .index('movies') + .search('dragon', SearchQuery(showRankingScore: true)); + // #enddocregion + }, skip: true); + + test('facet search code samples', () async { + // #docregion facet_search_1 + await client.index('books').facetSearch( + FacetSearchQuery( + facetQuery: 'fiction', + facetName: 'genres', + filter: 'rating > 3', + ), + ); + // #enddocregion + + // #docregion facet_search_2 + await client.index('books').updateFaceting( + Faceting( + sortFacetValuesBy: { + 'genres': FacetingSortTypes.count, + }, + ), + ); + // #enddocregion + + // #docregion facet_search_3 + await client.index('books').facetSearch( + FacetSearchQuery( + facetQuery: 'c', + facetName: 'genres', + ), + ); + // #enddocregion + + // #docregion search_parameter_guide_attributes_to_search_on_1 + await client.index('books').facetSearch( + FacetSearchQuery( + facetQuery: 'c', + facetName: 'genres', + ), + ); + // #enddocregion + + // #docregion search_parameter_guide_facet_stats_1 + await client + .index('movie_ratings') + .search('Batman', SearchQuery(facets: ['genres', 'rating'])); + // #enddocregion + + // #docregion faceted_search_1 + await client + .index('books') + .search('', SearchQuery(facets: ['genres', 'rating', 'language'])); + // #enddocregion + + // #docregion filtering_guide_nested_1 + await client.index('movie_ratings').search( + 'thriller', + SearchQuery( + filterExpression: Meili.gte( + //or Meili.attr('rating.users') + //or 'rating.users'.toMeiliAttribute() + Meili.attrFromParts(['rating', 'users']), + Meili.value(90), + ), + ), + ); + // #enddocregion + + // #docregion sorting_guide_sort_nested_1 + await client + .index('movie_ratings') + .search('thriller', SearchQuery(sort: ['rating.users:asc'])); + // #enddocregion + + // #docregion search_parameter_guide_page_1 + await client + .index('movies') + .search('', SearchQuery(page: 2)) + .asPaginatedResult(); + // #enddocregion + + // #docregion search_parameter_guide_hitsperpage_1 + await client + .index('movies') + .search('', SearchQuery(hitsPerPage: 15)) + .asPaginatedResult(); + // #enddocregion + }, skip: true); } diff --git a/test/settings_test.dart b/test/settings_test.dart index 0c457e49..c6d98621 100644 --- a/test/settings_test.dart +++ b/test/settings_test.dart @@ -400,4 +400,19 @@ void main() { }); }); }); + + test('code samples', () async { + // #docregion faceted_search_update_settings_1 + await client + .index('movie_ratings') + .updateFilterableAttributes(['genres', 'rating', 'language']); + // #enddocregion + + // #docregion synonyms_guide_1 + await client.index('movies').updateSynonyms({ + 'great': ['fantastic'], + 'fantastic': ['great'], + }); + // #enddocregion + }); } diff --git a/test/swaps_test.dart b/test/swaps_test.dart index f7f4cbda..58ba3260 100644 --- a/test/swaps_test.dart +++ b/test/swaps_test.dart @@ -30,4 +30,13 @@ void main() { ]); }); }); + + test('code samples', () async { + // #docregion swap_indexes_1 + await client.swapIndexes([ + SwapIndex(['indexA', 'indexB']), + SwapIndex(['indexX', 'indexY']), + ]); + // #enddocregion + }, skip: true); } diff --git a/test/tasks_test.dart b/test/tasks_test.dart index 7a94e869..f521aeba 100644 --- a/test/tasks_test.dart +++ b/test/tasks_test.dart @@ -66,4 +66,60 @@ void main() { ); }); }); + + test( + 'code samples', + () async { + // #docregion async_guide_filter_by_date_1 + await client.getTasks( + params: TasksQuery( + afterEnqueuedAt: DateTime(2020, 10, 11, 11, 49, 53), + ), + ); + // #enddocregion + // #docregion async_guide_multiple_filters_1 + await client.getTasks( + params: TasksQuery( + indexUids: ['movies'], + types: ['documentAdditionOrUpdate', 'documentDeletion'], + statuses: ['processing'], + ), + ); + // #enddocregion + // #docregion async_guide_filter_by_ids_1 + await client.getTasks( + params: TasksQuery( + uids: [5, 10, 13], + ), + ); + // #enddocregion + // #docregion async_guide_filter_by_statuses_1 + await client.getTasks( + params: TasksQuery( + statuses: ['failed', 'canceled'], + ), + ); + // #enddocregion + // #docregion async_guide_filter_by_types_1 + await client.getTasks( + params: TasksQuery( + types: ['dumpCreation', 'indexSwap'], + ), + ); + // #enddocregion + // #docregion async_guide_filter_by_index_uids_1 + await client.getTasks(params: TasksQuery(indexUids: ['movies'])); + // #enddocregion + // #docregion delete_tasks_1 + await client.deleteTasks(params: DeleteTasksQuery(uids: [1, 2])); + // #enddocregion + // #docregion cancel_tasks_1 + await client.cancelTasks(params: CancelTasksQuery(uids: [1, 2])); + // #enddocregion + // #docregion async_guide_canceled_by_1 + await client.getTasks(params: TasksQuery(canceledBy: [9, 15])); + // #enddocregion + }, + skip: true, + ); } diff --git a/tool/.gitignore b/tool/.gitignore new file mode 100644 index 00000000..3a857904 --- /dev/null +++ b/tool/.gitignore @@ -0,0 +1,3 @@ +# https://dart.dev/guides/libraries/private-files +# Created by `dart pub` +.dart_tool/ diff --git a/tool/analysis_options.yaml b/tool/analysis_options.yaml new file mode 100644 index 00000000..c0a95325 --- /dev/null +++ b/tool/analysis_options.yaml @@ -0,0 +1,18 @@ +# This file configures the static analysis results for your project (errors, +# warnings, and lints). +# +# This enables the 'recommended' set of lints from `package:lints`. +# This set helps identify many issues that may lead to problems when running +# or consuming Dart code, and enforces writing Dart using a single, idiomatic +# style and format. +# +# If you want a smaller set of lints you can change this to specify +# 'package:lints/core.yaml'. These are just the most critical lints +# (the recommended set includes the core lints). +# The core lints are also what is used by pub.dev for scoring packages. + +include: package:lints/recommended.yaml + +linter: + rules: + avoid_print: false diff --git a/tool/bin/meili.dart b/tool/bin/meili.dart new file mode 100644 index 00000000..201457bf --- /dev/null +++ b/tool/bin/meili.dart @@ -0,0 +1 @@ +export 'package:meili_tool/src/main.dart'; diff --git a/tool/lib/src/command_base.dart b/tool/lib/src/command_base.dart new file mode 100644 index 00000000..bd7853dc --- /dev/null +++ b/tool/lib/src/command_base.dart @@ -0,0 +1,37 @@ +import 'package:args/command_runner.dart'; +import 'package:file/file.dart'; +import 'package:meili_tool/src/result.dart'; +import 'package:platform/platform.dart'; +import 'package:path/path.dart' as p; + +abstract class MeiliCommandBase extends Command { + final Directory packageDirectory; + + MeiliCommandBase( + this.packageDirectory, { + this.platform = const LocalPlatform(), + }); + + /// The current platform. + /// + /// This can be overridden for testing. + final Platform platform; + + /// A context that matches the default for [platform]. + p.Context get path => platform.isWindows ? p.windows : p.posix; + // Returns the relative path from [from] to [entity] in Posix style. + /// + /// This should be used when, for example, printing package-relative paths in + /// status or error messages. + String getRelativePosixPath( + FileSystemEntity entity, { + required Directory from, + }) => + p.posix.joinAll(path.split(path.relative(entity.path, from: from.path))); + + String get indentation => ' '; + + bool getBoolArg(String key) { + return (argResults![key] as bool?) ?? false; + } +} diff --git a/tool/lib/src/core.dart b/tool/lib/src/core.dart new file mode 100644 index 00000000..9835d0a0 --- /dev/null +++ b/tool/lib/src/core.dart @@ -0,0 +1,16 @@ +/// Error thrown when a command needs to exit with a non-zero exit code. +/// +/// While there is no specific definition of the meaning of different non-zero +/// exit codes for this tool, commands should follow the general convention: +/// 1: The command ran correctly, but found errors. +/// 2: The command failed to run because the arguments were invalid. +/// >2: The command failed to run correctly for some other reason. Ideally, +/// each such failure should have a unique exit code within the context of +/// that command. +class ToolExit extends Error { + /// Creates a tool exit with the given [exitCode]. + ToolExit(this.exitCode); + + /// The code that the process should exit with. + final int exitCode; +} diff --git a/tool/lib/src/main.dart b/tool/lib/src/main.dart new file mode 100644 index 00000000..ce3dc16e --- /dev/null +++ b/tool/lib/src/main.dart @@ -0,0 +1,53 @@ +import 'dart:io' as io; + +import 'package:args/command_runner.dart'; +import 'package:file/file.dart'; +import 'package:file/local.dart'; +import 'package:meili_tool/src/output_utils.dart'; +import 'package:meili_tool/src/result.dart'; + +import 'core.dart'; +import 'update_samples_command.dart'; + +void main(List arguments) { + const FileSystem fileSystem = LocalFileSystem(); + final Directory scriptDir = + fileSystem.file(io.Platform.script.toFilePath()).parent; + final Directory toolsDir = + scriptDir.basename == 'bin' ? scriptDir.parent : scriptDir.parent.parent; + + final Directory meilisearchDirectory = toolsDir.parent; + + final commandRunner = CommandRunner( + 'dart run ./tool/bin/meili.dart', 'Productivity utils for meilisearch.') + ..addCommand(UpdateSamplesCommand(meilisearchDirectory)); + + commandRunner.run(arguments).then((value) { + if (value == null) { + print('MUST output either a success or fail.'); + assert(false); + io.exit(255); + } + switch (value.state) { + case RunState.succeeded: + printSuccess('Success!'); + break; + case RunState.failed: + printError('Failed!'); + if (value.details.isNotEmpty) { + printError(value.details.join('\n')); + } + io.exit(255); + } + }).catchError((Object e) { + final ToolExit toolExit = e as ToolExit; + int exitCode = toolExit.exitCode; + // This should never happen; this check is here to guarantee that a ToolExit + // never accidentally has code 0 thus causing CI to pass. + if (exitCode == 0) { + assert(false); + exitCode = 255; + } + io.exit(exitCode); + }, test: (Object e) => e is ToolExit); +} diff --git a/tool/lib/src/output_utils.dart b/tool/lib/src/output_utils.dart new file mode 100644 index 00000000..7fd39f68 --- /dev/null +++ b/tool/lib/src/output_utils.dart @@ -0,0 +1,44 @@ +// Copyright 2013 The Flutter Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +import 'dart:io'; + +import 'package:colorize/colorize.dart'; +import 'package:meta/meta.dart'; + +export 'package:colorize/colorize.dart' show Styles; + +/// True if color should be applied. +/// +/// Defaults to autodetecting stdout. +@visibleForTesting +bool useColorForOutput = stdout.supportsAnsiEscapes; + +String _colorizeIfAppropriate(String string, Styles color) { + if (!useColorForOutput) { + return string; + } + return Colorize(string).apply(color).toString(); +} + +/// Prints [message] in green, if the environment supports color. +void printSuccess(String message) { + print(_colorizeIfAppropriate(message, Styles.GREEN)); +} + +/// Prints [message] in yellow, if the environment supports color. +void printWarning(String message) { + print(_colorizeIfAppropriate(message, Styles.YELLOW)); +} + +/// Prints [message] in red, if the environment supports color. +void printError(String message) { + print(_colorizeIfAppropriate(message, Styles.RED)); +} + +/// Returns [message] with escapes to print it in [color], if the environment +/// supports color. +String colorizeString(String message, Styles color) { + return _colorizeIfAppropriate(message, color); +} diff --git a/tool/lib/src/result.dart b/tool/lib/src/result.dart new file mode 100644 index 00000000..331bf433 --- /dev/null +++ b/tool/lib/src/result.dart @@ -0,0 +1,34 @@ +/// Possible outcomes of a command run for a package. +enum RunState { + /// The command succeeded for the package. + succeeded, + + /// The command failed for the package. + failed, +} + +/// The result of a [runForPackage] call. +class PackageResult { + /// A successful result. + PackageResult.success() : this._(RunState.succeeded); + + /// A run that failed. + /// + /// If [errors] are provided, they will be listed in the summary, otherwise + /// the summary will simply show that the package failed. + PackageResult.fail([List errors = const []]) + : this._(RunState.failed, errors); + + const PackageResult._(this.state, [this.details = const []]); + + /// The state the package run completed with. + final RunState state; + + /// Information about the result: + /// - For `succeeded`, this is empty. + /// - For `skipped`, it contains a single entry describing why the run was + /// skipped. + /// - For `failed`, it contains zero or more specific error details to be + /// shown in the summary. + final List details; +} diff --git a/tool/lib/src/update_samples_command.dart b/tool/lib/src/update_samples_command.dart new file mode 100644 index 00000000..54620088 --- /dev/null +++ b/tool/lib/src/update_samples_command.dart @@ -0,0 +1,269 @@ +// Source: https://github.com/flutter/packages/blob/d0411e450a8d94fcb221e8d8eacd3b1f8ca0e2fc/script/tool/lib/src/update_excerpts_command.dart +// but modified to accept yaml files. + +import 'dart:async'; +import 'package:file/file.dart'; +import 'package:http/http.dart' as http; +import 'package:meili_tool/src/command_base.dart'; +import 'package:meili_tool/src/result.dart'; +import 'package:yaml/yaml.dart'; +import 'package:yaml_edit/yaml_edit.dart'; + +class _SourceFile { + final File file; + final List contents; + Map? result; + + _SourceFile({ + required this.file, + required this.contents, + }); +} + +class UpdateSamplesCommand extends MeiliCommandBase { + static const String _failOnChangeFlag = 'fail-on-change'; + static const String _checkRemoteRepoFlag = 'check-remote-repository'; + static const String _generateMissingExcerpts = 'generate-missing-excerpts'; + + UpdateSamplesCommand( + super.packageDirectory, { + super.platform, + }) { + argParser.addFlag( + _failOnChangeFlag, + help: 'Fail if the command does anything. ' + '(Used in CI to ensure excerpts are up to date.)', + ); + argParser.addFlag( + _checkRemoteRepoFlag, + hide: true, + help: + 'Check the remote code samples to see if there are missing/useless keys', + ); + argParser.addFlag( + _generateMissingExcerpts, + hide: true, + help: 'Generate entries that are found in code samples, but not in code', + ); + } + + @override + String get description => + 'Updates .code-samples.meilisearch.yaml, based on code from code files'; + + @override + String get name => 'update-samples'; + + static const docregion = '#docregion'; + static const enddocregion = '#enddocregion'; + final startRegionRegex = RegExp(RegExp.escape(docregion) + r'\s+(?\w+)'); + + @override + Future run() async { + try { + final failOnChange = getBoolArg(_failOnChangeFlag); + final checkRemoteRepo = getBoolArg(_checkRemoteRepoFlag); + final generateMissingExcerpts = getBoolArg(_generateMissingExcerpts); + //read the samples yaml file + final changedKeys = {}; + final File samplesFile = + packageDirectory.childFile('.code-samples.meilisearch.yaml'); + final samplesContentRaw = await samplesFile.readAsString(); + final samplesYaml = loadYaml(samplesContentRaw); + if (samplesYaml is! YamlMap) { + print(samplesYaml.runtimeType); + return PackageResult.fail(['samples yaml must be an YamlMap']); + } + + final newSamplesYaml = YamlEditor(samplesContentRaw); + final foundCodeSamples = {}; + final missingSamples = {}; + final sourceFiles = await _discoverSourceFiles(); + for (var sourceFile in sourceFiles) { + final newValues = _runInFile(sourceFile); + foundCodeSamples.addAll(newValues); + sourceFile.result = newValues; + for (var element in newValues.entries) { + final existingValue = samplesYaml[element.key]; + if (existingValue != null) { + if (existingValue == element.value) { + continue; + } else { + changedKeys[element.key] = element.value; + } + } else { + changedKeys[element.key] = element.value; + } + } + if (failOnChange && changedKeys.isNotEmpty) { + return PackageResult.fail([ + 'found changed keys: ${changedKeys.keys.toList()}', + ]); + } + + if (!failOnChange) { + for (var changedEntry in changedKeys.entries) { + newSamplesYaml.update([changedEntry.key], changedEntry.value); + } + } + } + + for (var entry in samplesYaml.entries) { + if (foundCodeSamples.containsKey(entry.key)) { + continue; + } + missingSamples[entry.key] = entry.value; + } + if (generateMissingExcerpts) { + final targetFile = packageDirectory + .childDirectory('test') + .childFile('missing_samples.dart'); + final sb = StringBuffer(); + + sb.writeln(r"import 'package:meilisearch/meilisearch.dart';"); + sb.writeln('late MeiliSearchClient client;'); + sb.writeln('void main() async {'); + for (var element in missingSamples.entries) { + sb.writeln('// #docregion ${element.key}'); + sb.writeln(element.value); + sb.writeln('// #enddocregion'); + sb.writeln(); + } + sb.writeln('}'); + await targetFile.writeAsString(sb.toString()); + } + + // for now don't check remote repository + if (checkRemoteRepo) { + final fullSamplesYaml = await getFullCorrectSamples(); + final missingEntries = fullSamplesYaml.entries + .where((element) => !samplesYaml.containsKey(element.key)); + final oldEntries = samplesYaml.entries + .where((element) => !fullSamplesYaml.containsKey(element.key)); + if (failOnChange) { + if (missingEntries.isNotEmpty || oldEntries.isNotEmpty) { + return PackageResult.fail([ + if (missingEntries.isNotEmpty) + 'found the following missing entries: ${missingEntries.map((e) => e.key).join('\n')}', + if (oldEntries.isNotEmpty) + 'found the following useless entries: ${oldEntries.map((e) => e.key).join('\n')}', + ]); + } + } else { + for (var element in missingEntries) { + newSamplesYaml.update([element.key], element.value); + } + for (var element in oldEntries) { + newSamplesYaml.remove([element.key]); + } + } + } + + if (!failOnChange && !generateMissingExcerpts) { + await samplesFile.writeAsString(newSamplesYaml.toString()); + } + return PackageResult.success(); + } on PackageResult catch (e) { + return e; + } + } + + Future getFullCorrectSamples() async { + final uri = Uri.parse( + 'https://raw.githubusercontent.com/meilisearch/documentation/main/.code-samples.meilisearch.yaml'); + final data = await http.get(uri); + final parsed = loadYaml(data.body, sourceUrl: uri); + return parsed as YamlMap; + } + + Map _runInFile(_SourceFile file) { + int lineNumber = 0; + String? currentKey; + final keys = []; + final res = {}; + final currentKeyLines = >[]; + for (var line in file.contents) { + lineNumber++; + if (currentKey == null) { + final capture = startRegionRegex.firstMatch(line); + if (capture == null) { + continue; + } + final key = capture.namedGroup('key'); + if (key == null) { + throw PackageResult.fail(['found a #docregion with no key']); + } + if (keys.contains(key)) { + throw PackageResult.fail(['found duplicate keys $key']); + } + keys.add(key); + currentKey = key; + } else { + if (line.contains(enddocregion)) { + final sb = StringBuffer(); + final unindentedLines = + unindentLines(currentKeyLines.map((e) => e.value).toList()) + .join('\n'); + sb.write(unindentedLines); + //add to results. + res[currentKey] = sb.toString(); + + currentKey = null; + currentKeyLines.clear(); + } else { + currentKeyLines.add(MapEntry(lineNumber, line)); + } + } + } + return res; + } + + List unindentLines(List src) { + if (src.isEmpty) { + return src; + } + final ogFirst = src.first; + final trimmedFirst = ogFirst.trimLeft(); + final firstIndentation = ogFirst.length - trimmedFirst.length; + final res = []; + for (var element in src) { + final trimmedLine = element.trimLeft(); + if (trimmedLine.isEmpty) { + continue; + } + var indentation = element.length - trimmedLine.length; + indentation -= firstIndentation; + res.add('${" " * indentation}$trimmedLine'); + } + + return res; + } + + Future> _discoverSourceFiles() async { + final libDir = packageDirectory.childDirectory('lib'); + final testsDir = packageDirectory.childDirectory('test'); + //look in dart files and generate a new yaml file based on the referenced code. + final allDartFiles = [ + ...libDir.listSync(recursive: true), + ...testsDir.listSync(recursive: true), + ].where((element) => element.basename.toLowerCase().endsWith('.dart')); + + final sourceFiles = <_SourceFile>[]; + for (var dartFile in allDartFiles) { + if (dartFile is! File) { + continue; + } + final fileContents = await dartFile.readAsLines(); + if (!fileContents.any((line) => line.contains(docregion))) { + continue; + } + sourceFiles.add( + _SourceFile( + file: dartFile, + contents: fileContents, + ), + ); + } + return sourceFiles; + } +} diff --git a/tool/pubspec.yaml b/tool/pubspec.yaml new file mode 100644 index 00000000..be5bf8b8 --- /dev/null +++ b/tool/pubspec.yaml @@ -0,0 +1,30 @@ +name: meili_tool +description: | + Productivity tools for meilisearch dart repository, + most of this is inspired from the flutter packages repository https://github.com/flutter/packages/. +version: 1.0.0 +repository: https://github.com/meilisearch/meilisearch-dart + +environment: + sdk: '>=3.0.0 <4.0.0' + +# Add regular dependencies here. +dependencies: + lints: ^2.0.0 + test: ^1.21.0 + args: ^2.4.2 + cli_util: ^0.4.0 + file: ^7.0.0 + path: ^1.8.3 + platform: ^3.1.2 + collection: ^1.15.0 + colorize: ^3.0.0 + meta: ^1.10.0 + yaml: ^3.1.2 + yaml_edit: ^2.1.1 + http: ^1.1.0 + +dev_dependencies: + build_runner: ^2.0.3 + matcher: ^0.12.10 + mockito: '>=5.3.2 <=5.4.0'