diff --git a/CHANGELOG.md b/CHANGELOG.md index b461f67de..f5d191e00 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,9 @@ ## 2.6.1 +### New features +- Added experimental Details API that provides access to POI metadata, boundary details, addresses and places. See `DetailsApi` type. + ### Bug fixes - Fixed `OpenHours` parsing for the Search Box Api type. diff --git a/MapboxSearch/base/src/main/java/com/mapbox/search/base/ExperimentalMapboxSearchAPI.kt b/MapboxSearch/base/src/main/java/com/mapbox/search/base/ExperimentalMapboxSearchAPI.kt new file mode 100644 index 000000000..c5b7dfec9 --- /dev/null +++ b/MapboxSearch/base/src/main/java/com/mapbox/search/base/ExperimentalMapboxSearchAPI.kt @@ -0,0 +1,19 @@ +package com.mapbox.search.base + +/** + * This annotation marks the experimental state of the Mapbox Search SDK API. + * This API is stable in nature, but it's likely that properties might be added or removed in the + * future. + * Any usage of a declaration annotated with `@ExperimentalMapboxSearchAPI` must be accepted + * either by annotating that usage with the [OptIn] annotation, + * e.g. `@OptIn(ExperimentalMapboxSearchAPI::class)`, or by using the compiler argument + * `-Xopt-in=com.mapbox.search.base.ExperimentalMapboxSearchAPI`. + */ +@Retention(value = AnnotationRetention.BINARY) +@RequiresOptIn(level = RequiresOptIn.Level.ERROR) +@Target( + AnnotationTarget.CLASS, + AnnotationTarget.FUNCTION, + AnnotationTarget.PROPERTY, +) +annotation class ExperimentalMapboxSearchAPI diff --git a/MapboxSearch/sample/src/main/AndroidManifest.xml b/MapboxSearch/sample/src/main/AndroidManifest.xml index e42852470..bccc82d09 100644 --- a/MapboxSearch/sample/src/main/AndroidManifest.xml +++ b/MapboxSearch/sample/src/main/AndroidManifest.xml @@ -68,6 +68,7 @@ + diff --git a/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/MainActivity.kt b/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/MainActivity.kt index b4502cfa2..ae1c8073a 100644 --- a/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/MainActivity.kt +++ b/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/MainActivity.kt @@ -72,6 +72,7 @@ import com.mapbox.search.sample.api.OfflineSearchAlongRouteExampleActivity import com.mapbox.search.sample.api.OfflineSearchJavaExampleActivity import com.mapbox.search.sample.api.OfflineSearchKotlinExampleActivity import com.mapbox.search.sample.api.PlaceAutocompleteKotlinExampleActivity +import com.mapbox.search.sample.api.DetailsApiKotlinExampleActivity import com.mapbox.search.sample.api.ReverseGeocodingJavaExampleActivity import com.mapbox.search.sample.api.ReverseGeocodingKotlinExampleActivity import com.mapbox.search.ui.adapter.engines.SearchEngineUiAdapter @@ -477,6 +478,10 @@ class MainActivity : AppCompatActivity() { startActivity(Intent(this, JapanSearchKotlinExampleActivity::class.java)) true } + R.id.open_details_api_kt_example -> { + startActivity(Intent(this, DetailsApiKotlinExampleActivity::class.java)) + true + } R.id.open_japan_search_java_example -> { startActivity(Intent(this, JapanSearchJavaExampleActivity::class.java)) true diff --git a/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/api/DetailsApiKotlinExampleActivity.kt b/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/api/DetailsApiKotlinExampleActivity.kt new file mode 100644 index 000000000..b373ed578 --- /dev/null +++ b/MapboxSearch/sample/src/main/java/com/mapbox/search/sample/api/DetailsApiKotlinExampleActivity.kt @@ -0,0 +1,53 @@ +package com.mapbox.search.sample.api + +import android.os.Bundle +import com.mapbox.search.AttributeSet +import com.mapbox.search.ResponseInfo +import com.mapbox.search.details.RetrieveDetailsOptions +import com.mapbox.search.SearchResultCallback +import com.mapbox.search.base.ExperimentalMapboxSearchAPI +import com.mapbox.search.common.AsyncOperationTask +import com.mapbox.search.details.DetailsApi +import com.mapbox.search.details.DetailsApiSettings +import com.mapbox.search.result.SearchResult +import com.mapbox.search.sample.R + +@OptIn(ExperimentalMapboxSearchAPI::class) +class DetailsApiKotlinExampleActivity : BaseKotlinExampleActivity() { + + override val titleResId: Int = R.string.action_open_details_api_kt_example + + private lateinit var detailsApi: DetailsApi + private var task: AsyncOperationTask? = null + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + + // Set your Access Token here if it's not already set in some other way + // MapboxOptions.accessToken = "" + detailsApi = DetailsApi.create(DetailsApiSettings()) + } + + override fun onDestroy() { + task?.cancel() + super.onDestroy() + } + + override fun startExample() { + task = detailsApi.retrieveDetails( + mapboxId = "dXJuOm1ieHBvaTowZGY2MzE4Yi0wNGNjLTRkOTYtYTZmMy0yNmJmM2ZiODUyODU", + options = RetrieveDetailsOptions(attributeSets = AttributeSet.values().toList()), + callback = object : SearchResultCallback { + override fun onResult(result: SearchResult, responseInfo: ResponseInfo) { + logI("SearchApiExample", "Retrieve result:", result) + onFinished() + } + + override fun onError(e: Exception) { + logE("SearchApiExample", "Retrieve error", e) + onFinished() + } + } + ) + } +} \ No newline at end of file diff --git a/MapboxSearch/sample/src/main/res/menu/main_activity_options_menu.xml b/MapboxSearch/sample/src/main/res/menu/main_activity_options_menu.xml index 2d41f218b..50a0e0315 100644 --- a/MapboxSearch/sample/src/main/res/menu/main_activity_options_menu.xml +++ b/MapboxSearch/sample/src/main/res/menu/main_activity_options_menu.xml @@ -94,6 +94,11 @@ android:title="@string/action_open_japan_search_kt_example" /> + + Favorites Data Provider Custom data provider Custom data provider + Details API Offline Search Along Route diff --git a/MapboxSearch/sdk/api/api-metalava.txt b/MapboxSearch/sdk/api/api-metalava.txt index 2925cea4d..0d3994f76 100644 --- a/MapboxSearch/sdk/api/api-metalava.txt +++ b/MapboxSearch/sdk/api/api-metalava.txt @@ -604,6 +604,48 @@ package com.mapbox.search.analytics { } +package com.mapbox.search.details { + + @com.mapbox.search.base.ExperimentalMapboxSearchAPI public interface DetailsApi { + method public default static com.mapbox.search.details.DetailsApi create(com.mapbox.search.details.DetailsApiSettings settings); + method public default com.mapbox.search.common.AsyncOperationTask retrieveDetails(String mapboxId, com.mapbox.search.details.RetrieveDetailsOptions options, com.mapbox.search.SearchResultCallback callback); + method public com.mapbox.search.common.AsyncOperationTask retrieveDetails(String mapboxId, com.mapbox.search.details.RetrieveDetailsOptions options, java.util.concurrent.Executor executor, com.mapbox.search.SearchResultCallback callback); + field public static final com.mapbox.search.details.DetailsApi.Companion Companion; + } + + public static final class DetailsApi.Companion { + method public com.mapbox.search.details.DetailsApi create(com.mapbox.search.details.DetailsApiSettings settings); + } + + @com.mapbox.search.base.ExperimentalMapboxSearchAPI public final class DetailsApiSettings { + ctor public DetailsApiSettings(com.mapbox.common.location.LocationProvider? locationProvider = (), com.mapbox.search.ViewportProvider? viewportProvider = null, String? baseUrl = null); + ctor public DetailsApiSettings(com.mapbox.common.location.LocationProvider? locationProvider = (), com.mapbox.search.ViewportProvider? viewportProvider = null); + ctor public DetailsApiSettings(com.mapbox.common.location.LocationProvider? locationProvider = ()); + method public String? getBaseUrl(); + method public com.mapbox.common.location.LocationProvider? getLocationProvider(); + method public com.mapbox.search.ViewportProvider? getViewportProvider(); + property public final String? baseUrl; + property public final com.mapbox.common.location.LocationProvider? locationProvider; + property public final com.mapbox.search.ViewportProvider? viewportProvider; + } + + @com.mapbox.search.base.ExperimentalMapboxSearchAPI @kotlinx.parcelize.Parcelize public final class RetrieveDetailsOptions implements android.os.Parcelable { + ctor public RetrieveDetailsOptions(java.util.List? attributeSets = null, com.mapbox.search.common.IsoLanguageCode language = (), com.mapbox.search.common.IsoCountryCode? worldview = null); + ctor public RetrieveDetailsOptions(java.util.List? attributeSets = null, com.mapbox.search.common.IsoLanguageCode language = ()); + ctor public RetrieveDetailsOptions(java.util.List? attributeSets = null); + method public java.util.List? getAttributeSets(); + method public com.mapbox.search.common.IsoLanguageCode getLanguage(); + method public com.mapbox.search.common.IsoCountryCode? getWorldview(); + property public final java.util.List? attributeSets; + property public final com.mapbox.search.common.IsoLanguageCode language; + property public final com.mapbox.search.common.IsoCountryCode? worldview; + } + + public final class RetrieveDetailsOptionsKt { + } + +} + package com.mapbox.search.record { @kotlinx.parcelize.Parcelize public final class FavoriteRecord implements com.mapbox.search.record.IndexableRecord android.os.Parcelable { diff --git a/MapboxSearch/sdk/api/sdk.api b/MapboxSearch/sdk/api/sdk.api index a57965180..1e69a4a45 100644 --- a/MapboxSearch/sdk/api/sdk.api +++ b/MapboxSearch/sdk/api/sdk.api @@ -736,6 +736,60 @@ public final class com/mapbox/search/analytics/events/SearchResultsInfo$Creator public synthetic fun newArray (I)[Ljava/lang/Object; } +public abstract interface class com/mapbox/search/details/DetailsApi { + public static final field Companion Lcom/mapbox/search/details/DetailsApi$Companion; + public static fun create (Lcom/mapbox/search/details/DetailsApiSettings;)Lcom/mapbox/search/details/DetailsApi; + public abstract fun retrieveDetails (Ljava/lang/String;Lcom/mapbox/search/details/RetrieveDetailsOptions;Lcom/mapbox/search/SearchResultCallback;)Lcom/mapbox/search/common/AsyncOperationTask; + public abstract fun retrieveDetails (Ljava/lang/String;Lcom/mapbox/search/details/RetrieveDetailsOptions;Ljava/util/concurrent/Executor;Lcom/mapbox/search/SearchResultCallback;)Lcom/mapbox/search/common/AsyncOperationTask; +} + +public final class com/mapbox/search/details/DetailsApi$Companion { + public final fun create (Lcom/mapbox/search/details/DetailsApiSettings;)Lcom/mapbox/search/details/DetailsApi; +} + +public final class com/mapbox/search/details/DetailsApi$DefaultImpls { + public static fun retrieveDetails (Lcom/mapbox/search/details/DetailsApi;Ljava/lang/String;Lcom/mapbox/search/details/RetrieveDetailsOptions;Lcom/mapbox/search/SearchResultCallback;)Lcom/mapbox/search/common/AsyncOperationTask; +} + +public final class com/mapbox/search/details/DetailsApiSettings { + public fun ()V + public fun (Lcom/mapbox/common/location/LocationProvider;)V + public fun (Lcom/mapbox/common/location/LocationProvider;Lcom/mapbox/search/ViewportProvider;)V + public fun (Lcom/mapbox/common/location/LocationProvider;Lcom/mapbox/search/ViewportProvider;Ljava/lang/String;)V + public synthetic fun (Lcom/mapbox/common/location/LocationProvider;Lcom/mapbox/search/ViewportProvider;Ljava/lang/String;ILkotlin/jvm/internal/DefaultConstructorMarker;)V + public fun equals (Ljava/lang/Object;)Z + public final fun getBaseUrl ()Ljava/lang/String; + public final fun getLocationProvider ()Lcom/mapbox/common/location/LocationProvider; + public final fun getViewportProvider ()Lcom/mapbox/search/ViewportProvider; + public fun hashCode ()I + public fun toString ()Ljava/lang/String; +} + +public final class com/mapbox/search/details/RetrieveDetailsOptions : android/os/Parcelable { + public static final field CREATOR Landroid/os/Parcelable$Creator; + public fun ()V + public fun (Ljava/util/List;)V + public fun (Ljava/util/List;Lcom/mapbox/search/common/IsoLanguageCode;)V + public fun (Ljava/util/List;Lcom/mapbox/search/common/IsoLanguageCode;Lcom/mapbox/search/common/IsoCountryCode;)V + public synthetic fun (Ljava/util/List;Lcom/mapbox/search/common/IsoLanguageCode;Lcom/mapbox/search/common/IsoCountryCode;ILkotlin/jvm/internal/DefaultConstructorMarker;)V + public fun describeContents ()I + public fun equals (Ljava/lang/Object;)Z + public final fun getAttributeSets ()Ljava/util/List; + public final fun getLanguage ()Lcom/mapbox/search/common/IsoLanguageCode; + public final fun getWorldview ()Lcom/mapbox/search/common/IsoCountryCode; + public fun hashCode ()I + public fun toString ()Ljava/lang/String; + public fun writeToParcel (Landroid/os/Parcel;I)V +} + +public final class com/mapbox/search/details/RetrieveDetailsOptions$Creator : android/os/Parcelable$Creator { + public fun ()V + public final fun createFromParcel (Landroid/os/Parcel;)Lcom/mapbox/search/details/RetrieveDetailsOptions; + public synthetic fun createFromParcel (Landroid/os/Parcel;)Ljava/lang/Object; + public final fun newArray (I)[Lcom/mapbox/search/details/RetrieveDetailsOptions; + public synthetic fun newArray (I)[Ljava/lang/Object; +} + public final class com/mapbox/search/record/FavoriteRecord : android/os/Parcelable, com/mapbox/search/record/IndexableRecord { public static final field CREATOR Landroid/os/Parcelable$Creator; public fun (Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Lcom/mapbox/search/result/SearchAddress;Ljava/util/List;Ljava/util/List;Ljava/lang/String;Lcom/mapbox/geojson/Point;Lcom/mapbox/search/result/SearchResultType;Lcom/mapbox/search/SearchResultMetadata;)V diff --git a/MapboxSearch/sdk/src/androidTest/assets/details_api/response_successful.json b/MapboxSearch/sdk/src/androidTest/assets/details_api/response_successful.json new file mode 100644 index 000000000..5ae6aba68 --- /dev/null +++ b/MapboxSearch/sdk/src/androidTest/assets/details_api/response_successful.json @@ -0,0 +1,186 @@ +{ + "geometry": { + "coordinates": [ + -77.029129, + 38.902309 + ], + "type": "Point" + }, + "properties": { + "address": "925 13th St NW", + "context": { + "address": { + "address_number": "925", + "id": "", + "name": "925 13th St NW, Washington DC, DC 20005-4005", + "street_name": "13th st nw" + }, + "country": { + "country_code": "US", + "country_code_alpha_3": "USA", + "id": "", + "name": "United States" + }, + "neighborhood": { + "id": "dXJuOm1ieHBsYzpDaXpzN0E", + "name": "Downtown" + }, + "place": { + "id": "dXJuOm1ieHBsYzpGSmlvN0E", + "name": "Washington" + }, + "postcode": { + "id": "dXJuOm1ieHBsYzpBNTd1N0E", + "name": "20005" + }, + "region": { + "id": "", + "name": "District of Columbia", + "region_code": "DC", + "region_code_full": "US-DC" + }, + "street": { + "id": "", + "name": "13th st nw" + } + }, + "coordinates": { + "latitude": 38.902309, + "longitude": -77.029129, + "routable_points": [ + { + "latitude": 38.9023073713156, + "longitude": -77.0296257487584, + "name": "POI" + } + ] + }, + "external_ids": { + "tripadvisor": "20145439" + }, + "feature_type": "poi", + "full_address": "925 13th St NW, Washington, District of Columbia 20005, United States", + "language": "en", + "maki": "marker", + "mapbox_id": "dXJuOm1ieHBvaTo0ZTg2ZWFkNS1jOWMwLTQ3OWEtOTA5Mi1kMDVlNDQ3NDdlODk", + "metadata": { + "open_hours": { + "periods": [ + { + "close": { + "day": 0, + "time": "1800" + }, + "open": { + "day": 0, + "time": "1000" + } + }, + { + "close": { + "day": 1, + "time": "1700" + }, + "open": { + "day": 1, + "time": "1000" + } + }, + { + "close": { + "day": 3, + "time": "1700" + }, + "open": { + "day": 3, + "time": "1000" + } + }, + { + "close": { + "day": 4, + "time": "1700" + }, + "open": { + "day": 4, + "time": "1000" + } + }, + { + "close": { + "day": 5, + "time": "1700" + }, + "open": { + "day": 5, + "time": "1000" + } + }, + { + "close": { + "day": 6, + "time": "1800" + }, + "open": { + "day": 6, + "time": "1000" + } + } + ], + "weekday_text": [ + "Monday: 10:00 AM - 5:00 PM", + "Tuesday: Closed", + "Wednesday: 10:00 AM - 5:00 PM", + "Thursday: 10:00 AM - 5:00 PM", + "Friday: 10:00 AM - 5:00 PM", + "Saturday: 10:00 AM - 6:00 PM", + "Sunday: 10:00 AM - 6:00 PM" + ] + }, + "phone": "+12029313139", + "photos": [ + { + "height": 768, + "url": "https://media-cdn.tripadvisor.com/media/photo-o/2a/08/ad/cd/caption.jpg", + "width": 1024 + }, + { + "height": 1024, + "url": "https://media-cdn.tripadvisor.com/media/photo-o/2a/08/ad/fe/caption.jpg", + "width": 768 + }, + { + "height": 975, + "url": "https://media-cdn.tripadvisor.com/media/photo-o/28/b6/2d/cc/caption.jpg", + "width": 2006 + }, + { + "height": 3888, + "url": "https://media-cdn.tripadvisor.com/media/photo-o/2a/45/ec/0a/caption.jpg", + "width": 5184 + }, + { + "height": 975, + "url": "https://media-cdn.tripadvisor.com/media/photo-o/28/b6/2d/ca/caption.jpg", + "width": 2006 + } + ], + "primary_photo": "https://media-cdn.tripadvisor.com/media/photo-o/2a/08/ad/cd/caption.jpg", + "rating": 5, + "review_count": 69, + "website": "http://www.planetwordmuseum.org" + }, + "name": "Planet Word", + "operational_status": "active", + "place_formatted": "Washington, District of Columbia 20005, United States", + "poi_category": [ + "museum", + "tourist attraction" + ], + "poi_category_ids": [ + "museum", + "tourist_attraction" + ] + }, + "type": "Feature" +} \ No newline at end of file diff --git a/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/SearchEngineIntegrationTest.kt b/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/SearchEngineIntegrationTest.kt index 0fda778da..be67b215c 100644 --- a/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/SearchEngineIntegrationTest.kt +++ b/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/SearchEngineIntegrationTest.kt @@ -1300,9 +1300,9 @@ internal class SearchEngineIntegrationTest : BaseTest() { searchEngine.retrieve(mapboxId, callback) val result = callback.getResultBlocking() - assertTrue(result is SearchResult) + assertTrue(result.isSuccess) - val searchResult = result as SearchResult + val searchResult = result.getSuccess().result assertEquals(mapboxId, searchResult.mapboxId) } diff --git a/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/details/DetailsApiIntegrationTest.kt b/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/details/DetailsApiIntegrationTest.kt new file mode 100644 index 000000000..e25896118 --- /dev/null +++ b/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/details/DetailsApiIntegrationTest.kt @@ -0,0 +1,115 @@ +package com.mapbox.search.details + +import com.mapbox.common.MapboxOptions +import com.mapbox.geojson.Point +import com.mapbox.search.AttributeSet +import com.mapbox.search.BaseTest +import com.mapbox.search.MapboxSearchSdk +import com.mapbox.search.base.ExperimentalMapboxSearchAPI +import com.mapbox.search.common.IsoCountryCode +import com.mapbox.search.common.IsoLanguageCode +import com.mapbox.search.common.SearchRequestException +import com.mapbox.search.common.metadata.OpenHours +import com.mapbox.search.tests_support.BlockingSearchResultCallback +import com.mapbox.search.utils.assertEqualsIgnoreCase +import okhttp3.mockwebserver.MockResponse +import okhttp3.mockwebserver.MockWebServer +import org.junit.Assert.assertEquals +import org.junit.Assert.assertTrue +import org.junit.Before +import org.junit.Test + +@OptIn(ExperimentalMapboxSearchAPI::class) +internal class DetailsApiIntegrationTest : BaseTest() { + + private lateinit var mockServer: MockWebServer + private lateinit var detailsApiSettings: DetailsApiSettings + private lateinit var detailsApi: DetailsApi + + @Before + override fun setUp() { + super.setUp() + MapboxOptions.accessToken = TEST_ACCESS_TOKEN + + mockServer = MockWebServer() + + MapboxSearchSdk.initialize( + application = targetApplication, + ) + + detailsApiSettings = DetailsApiSettings( + baseUrl = mockServer.url("").toString(), + ) + + detailsApi = DetailsApi.create(detailsApiSettings) + } + + @Test + fun testRequestParametersAreCorrect() { + mockServer.enqueue(MockResponse().setResponseCode(500)) + + val options = RetrieveDetailsOptions( + attributeSets = listOf(AttributeSet.BASIC, AttributeSet.VISIT, AttributeSet.PHOTOS, AttributeSet.VENUE), + language = IsoLanguageCode.FRENCH, + worldview = IsoCountryCode.FRANCE, + ) + + val callback = BlockingSearchResultCallback() + detailsApi.retrieveDetails(TEST_MAPBOX_ID, options, callback) + + val request = mockServer.takeRequest() + assertEqualsIgnoreCase("get", request.method!!) + + val url = request.requestUrl!! + assertEqualsIgnoreCase("//search/details/v1/retrieve/$TEST_MAPBOX_ID", url.encodedPath) + assertEquals(TEST_ACCESS_TOKEN, url.queryParameter("access_token")) + assertEquals(options.language.code, url.queryParameter("language")) + assertEquals(options.worldview!!.code, url.queryParameter("worldview")) + assertEquals( + options.attributeSets!!.joinToString(separator = ",") { it.name.lowercase() }, + url.queryParameter("attribute_sets") + ) + } + + @Test + fun testSuccessfulResponse() { + mockServer.enqueue(createSuccessfulResponse("details_api/response_successful.json")) + + val callback = BlockingSearchResultCallback() + detailsApi.retrieveDetails(TEST_MAPBOX_ID, RetrieveDetailsOptions(), callback) + + val requestResult = callback.getResultBlocking() + assertTrue(requestResult.isSuccess) + + val searchResult = requestResult.getSuccess().result + + assertEquals("Planet Word", searchResult.name) + assertEquals( + "dXJuOm1ieHBvaTo0ZTg2ZWFkNS1jOWMwLTQ3OWEtOTA5Mi1kMDVlNDQ3NDdlODk", + searchResult.mapboxId, + ) + assertEquals(Point.fromLngLat(-77.029129, 38.902309), searchResult.coordinate) + assertTrue(searchResult.metadata?.openHours is OpenHours.Scheduled) + assertEquals("+12029313139", searchResult.metadata?.phone) + assertEquals(5, searchResult.metadata?.otherPhotos?.size) + } + + @Test + fun testErrorResponse() { + mockServer.enqueue(MockResponse().setResponseCode(404)) + + val callback = BlockingSearchResultCallback() + detailsApi.retrieveDetails(TEST_MAPBOX_ID, RetrieveDetailsOptions(), callback) + + val requestResult = callback.getResultBlocking() + assertTrue(requestResult.isError) + + val e = requestResult.getError().e + assertTrue(e is SearchRequestException && e.code == 404) + } + + private companion object { + const val TEST_ACCESS_TOKEN = "pk.test" + const val TEST_MAPBOX_ID = "test-id" + } +} diff --git a/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/tests_support/BlockingSearchResultCallback.kt b/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/tests_support/BlockingSearchResultCallback.kt index b362ec354..2d4359a4b 100644 --- a/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/tests_support/BlockingSearchResultCallback.kt +++ b/MapboxSearch/sdk/src/androidTest/java/com/mapbox/search/tests_support/BlockingSearchResultCallback.kt @@ -5,13 +5,30 @@ import com.mapbox.search.SearchResultCallback import com.mapbox.search.common.tests.BaseBlockingCallback import com.mapbox.search.result.SearchResult -internal class BlockingSearchResultCallback : SearchResultCallback, BaseBlockingCallback() { +internal class BlockingSearchResultCallback : + SearchResultCallback, + BaseBlockingCallback() { override fun onResult(result: SearchResult, responseInfo: ResponseInfo) { - publishResult(result) + publishResult(Result.Success(result, responseInfo)) } override fun onError(e: Exception) { - publishResult(e) + publishResult(Result.Error(e)) + } + + sealed class Result { + + val isSuccess: Boolean + get() = this is Success + + val isError: Boolean + get() = !isSuccess + + fun getSuccess() = this as Success + fun getError() = this as Error + + data class Success(val result: SearchResult, val responseInfo: ResponseInfo) : Result() + data class Error(val e: Exception) : Result() } } diff --git a/MapboxSearch/sdk/src/main/java/com/mapbox/search/SearchEngineFactory.kt b/MapboxSearch/sdk/src/main/java/com/mapbox/search/SearchEngineFactory.kt index e8f0a3f9b..da48b8ec4 100644 --- a/MapboxSearch/sdk/src/main/java/com/mapbox/search/SearchEngineFactory.kt +++ b/MapboxSearch/sdk/src/main/java/com/mapbox/search/SearchEngineFactory.kt @@ -1,5 +1,6 @@ package com.mapbox.search +import com.mapbox.common.location.LocationProvider import com.mapbox.search.analytics.AnalyticsService import com.mapbox.search.base.BaseSearchSdkInitializerImpl import com.mapbox.search.base.StubCompletionCallback @@ -93,18 +94,32 @@ internal class SearchEngineFactory { else -> null } + return createCoreEngineByApiType( + apiType = apiType, + baseUrl = baseUrl, + locationProvider = settings.locationProvider, + viewportProvider = settings.viewportProvider, + ) + } + + fun createCoreEngineByApiType( + apiType: ApiType, + baseUrl: String?, + locationProvider: LocationProvider?, + viewportProvider: ViewportProvider?, + ): CoreSearchEngineInterface { // Workaround for sync location provider in test environment. // Needed while https://github.com/mapbox/mapbox-search-sdk/issues/671 not fixed - val coreLocationProvider = if (settings.locationProvider is CoreLocationProvider) { - settings.locationProvider + val coreLocationProvider = if (locationProvider is CoreLocationProvider) { + locationProvider } else { WrapperLocationProvider( LocationEngineAdapter( BaseSearchSdkInitializerImpl.app, - settings.locationProvider + locationProvider ) ) { - settings.viewportProvider?.getViewport()?.mapToCore() + viewportProvider?.getViewport()?.mapToCore() } } diff --git a/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApi.kt b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApi.kt new file mode 100644 index 000000000..c12cc3171 --- /dev/null +++ b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApi.kt @@ -0,0 +1,93 @@ +package com.mapbox.search.details + +import com.mapbox.search.ApiType +import com.mapbox.search.MapboxSearchSdk +import com.mapbox.search.SearchEngineFactory +import com.mapbox.search.SearchResultCallback +import com.mapbox.search.base.ExperimentalMapboxSearchAPI +import com.mapbox.search.base.core.getUserActivityReporter +import com.mapbox.search.common.AsyncOperationTask +import com.mapbox.search.common.concurrent.SearchSdkMainThreadWorker +import java.util.concurrent.Executor + +/** + * The Details API provides access to POI metadata, boundary details, addresses and places. + * For more information, visit [Details API page](https://docs.mapbox.com/api/search/details/). + * + * Instance of the [DetailsApi] can be obtained with [DetailsApi.create]. + */ +@ExperimentalMapboxSearchAPI +public interface DetailsApi { + + /** + * Request basic metadata for a POI, which includes attributes such as POI name, address, + * coordinates, primary photo and category classification. + * + * To retrieve additional attributes beyond the basic data for a POI, + * specify [RetrieveDetailsOptions.attributeSets] in the provided [options]. + * + * @param mapboxId A unique identifier for the geographic feature. + * @param options Retrieve options. + * @param callback Search result callback. Events are dispatched on the main thread. + * @return [AsyncOperationTask] object representing pending completion of the request + */ + public fun retrieveDetails( + mapboxId: String, + options: RetrieveDetailsOptions, + callback: SearchResultCallback, + ): AsyncOperationTask = retrieveDetails( + mapboxId = mapboxId, + options = options, + executor = SearchSdkMainThreadWorker.mainExecutor, + callback = callback, + ) + + /** + * Request basic metadata for a POI, which includes attributes such as POI name, address, + * coordinates, primary photo and category classification. + * + * To retrieve additional attributes beyond the basic data for a POI, + * specify [RetrieveDetailsOptions.attributeSets] in the provided [options]. + * + * @param mapboxId A unique identifier for the geographic feature. + * @param options Retrieve options. + * @param executor [Executor] used for events dispatching, default is the main thread. + * @param callback Search result callback. + * @return [AsyncOperationTask] object representing pending completion of the request + */ + public fun retrieveDetails( + mapboxId: String, + options: RetrieveDetailsOptions, + executor: Executor, + callback: SearchResultCallback, + ): AsyncOperationTask + + /** + * Companion object. + */ + public companion object { + + /** + * Creates a new instance of the [DetailsApi]. + * + * @param settings [DetailsApiSettings] settings. + * @return a new instance instance of the [DetailsApi]. + */ + @JvmStatic + public fun create(settings: DetailsApiSettings): DetailsApi { + val coreEngine = SearchEngineFactory().createCoreEngineByApiType( + apiType = ApiType.SEARCH_BOX, + baseUrl = settings.baseUrl, + locationProvider = settings.locationProvider, + viewportProvider = settings.viewportProvider, + ) + + return DetailsApiImpl( + coreEngine, + getUserActivityReporter(), + MapboxSearchSdk.searchRequestContextProvider, + MapboxSearchSdk.searchResultFactory, + ) + } + } +} diff --git a/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApiImpl.kt b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApiImpl.kt new file mode 100644 index 000000000..80ae5a63b --- /dev/null +++ b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApiImpl.kt @@ -0,0 +1,62 @@ +package com.mapbox.search.details + +import com.mapbox.search.SearchResultCallback +import com.mapbox.search.adapter.SearchResultCallbackAdapter +import com.mapbox.search.base.ExperimentalMapboxSearchAPI +import com.mapbox.search.base.SearchRequestContextProvider +import com.mapbox.search.base.core.CoreApiType +import com.mapbox.search.base.core.CoreSearchEngineInterface +import com.mapbox.search.base.engine.BaseSearchEngine +import com.mapbox.search.base.engine.OneStepRequestCallbackWrapper +import com.mapbox.search.base.result.SearchResultFactory +import com.mapbox.search.common.AsyncOperationTask +import com.mapbox.search.internal.bindgen.UserActivityReporterInterface +import java.util.concurrent.Executor +import java.util.concurrent.ExecutorService +import java.util.concurrent.Executors + +@OptIn(ExperimentalMapboxSearchAPI::class) +internal class DetailsApiImpl( + private val coreEngine: CoreSearchEngineInterface, + private val activityReporter: UserActivityReporterInterface, + private val requestContextProvider: SearchRequestContextProvider, + private val searchResultFactory: SearchResultFactory, + private val engineExecutorService: ExecutorService = DEFAULT_EXECUTOR, +) : BaseSearchEngine(), DetailsApi { + + override fun retrieveDetails( + mapboxId: String, + options: RetrieveDetailsOptions, + executor: Executor, + callback: SearchResultCallback + ): AsyncOperationTask { + activityReporter.reportActivity("details-api-retrieve") + + val baseCallback = SearchResultCallbackAdapter(callback) + + return makeRequest(baseCallback) { task -> + val requestId = coreEngine.retrieveDetails( + mapboxId, + options.mapToCore(), + OneStepRequestCallbackWrapper( + searchResultFactory = searchResultFactory, + callbackExecutor = executor, + workerExecutor = engineExecutorService, + searchRequestTask = task, + searchRequestContext = requestContextProvider.provide(CoreApiType.SEARCH_BOX), + isOffline = false, + ) + ) + + task.addOnCancelledCallback { + coreEngine.cancel(requestId) + } + } + } + + private companion object { + val DEFAULT_EXECUTOR: ExecutorService = Executors.newSingleThreadExecutor { runnable -> + Thread(runnable, "DetailsApi executor") + } + } +} diff --git a/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApiSettings.kt b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApiSettings.kt new file mode 100644 index 000000000..ed2728237 --- /dev/null +++ b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/DetailsApiSettings.kt @@ -0,0 +1,72 @@ +package com.mapbox.search.details + +import android.Manifest +import com.mapbox.common.location.LocationProvider +import com.mapbox.common.location.LocationServiceFactory +import com.mapbox.search.ViewportProvider +import com.mapbox.search.base.ExperimentalMapboxSearchAPI +import com.mapbox.search.base.location.defaultLocationProvider + +/** + * Settings used for [DetailsApi] configuration. + * @see DetailsApi + */ +@ExperimentalMapboxSearchAPI +public class DetailsApiSettings @JvmOverloads constructor( + + /** + * The mechanism responsible for providing location approximations to the SDK. + * By default [LocationProvider] is provided by [LocationServiceFactory]. + * Note that this class requires [Manifest.permission.ACCESS_COARSE_LOCATION] or + * [Manifest.permission.ACCESS_FINE_LOCATION] to work properly. + */ + public val locationProvider: LocationProvider? = defaultLocationProvider(), + + /** + * Viewport provider instance. + */ + public val viewportProvider: ViewportProvider? = null, + + /** + * Base endpoint URL. + */ + public val baseUrl: String? = null, +) { + + /** + * @suppress + */ + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as DetailsApiSettings + + if (locationProvider != other.locationProvider) return false + if (viewportProvider != other.viewportProvider) return false + if (baseUrl != other.baseUrl) return false + + return true + } + + /** + * @suppress + */ + override fun hashCode(): Int { + var result = locationProvider?.hashCode() ?: 0 + result = 31 * result + (viewportProvider?.hashCode() ?: 0) + result = 31 * result + (baseUrl?.hashCode() ?: 0) + return result + } + + /** + * @suppress + */ + override fun toString(): String { + return "DetailsApiSettings(" + + "locationProvider=$locationProvider, " + + "viewportProvider=$viewportProvider, " + + "baseUrl=$baseUrl" + + ")" + } +} diff --git a/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/RetrieveDetailsOptions.kt b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/RetrieveDetailsOptions.kt new file mode 100644 index 000000000..642aed47a --- /dev/null +++ b/MapboxSearch/sdk/src/main/java/com/mapbox/search/details/RetrieveDetailsOptions.kt @@ -0,0 +1,96 @@ +package com.mapbox.search.details + +import android.os.Parcelable +import com.mapbox.search.AttributeSet +import com.mapbox.search.base.ExperimentalMapboxSearchAPI +import com.mapbox.search.base.defaultLocaleLanguage +import com.mapbox.search.common.IsoCountryCode +import com.mapbox.search.common.IsoLanguageCode +import com.mapbox.search.internal.bindgen.DetailsOptions +import com.mapbox.search.mapToCore +import kotlinx.parcelize.Parcelize + +/** + * Options, used for the [DetailsApi.retrieveDetails]. + */ +@ExperimentalMapboxSearchAPI +@Parcelize +public class RetrieveDetailsOptions @JvmOverloads constructor( + + /** + * Besides the basic metadata attributes, developers can request additional + * attributes by setting attribute_sets parameter with attribute set values, + * for example &attribute_sets=basic,photos,visit. + * The requested metadata will be provided in metadata object in the response. + */ + public val attributeSets: List? = null, + + /** + * Specify the user’s language. This parameter controls the language of the text supplied in responses. + * If language is not set explicitly, then language from default system locale will be used. + */ + public val language: IsoLanguageCode = defaultLocaleLanguage(), + + /** + * The ISO country code to requests a worldview for the location data, + * if applicable data is available. + * This parameters will only be applicable for Boundaries and Places feature types. + */ + public val worldview: IsoCountryCode? = null, +) : Parcelable { + + /** + * @suppress + */ + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as RetrieveDetailsOptions + + if (attributeSets != other.attributeSets) return false + if (language != other.language) return false + if (worldview != other.worldview) return false + + return true + } + + /** + * @suppress + */ + override fun hashCode(): Int { + var result = attributeSets?.hashCode() ?: 0 + result = 31 * result + language.hashCode() + result = 31 * result + (worldview?.hashCode() ?: 0) + return result + } + + /** + * @suppress + */ + override fun toString(): String { + return "RetrieveDetailsOptions(" + + "attributeSets=$attributeSets, " + + "language=$language, " + + "worldview=$worldview" + + ")" + } +} + +@OptIn(ExperimentalMapboxSearchAPI::class) +@JvmSynthetic +internal fun RetrieveDetailsOptions.mapToCore(): DetailsOptions { + return DetailsOptions( + attributeSets?.fixedAttributesOption()?.map { it.mapToCore() }, + language.code, + worldview?.code, + ) +} + +private fun List.fixedAttributesOption(): List { + return if (isNotEmpty() && !contains(AttributeSet.BASIC)) { + this + AttributeSet.BASIC + } else { + this + } +} diff --git a/MapboxSearch/sdk/src/test/java/com/mapbox/search/RetrieveDetailsOptionsTest.kt b/MapboxSearch/sdk/src/test/java/com/mapbox/search/RetrieveDetailsOptionsTest.kt new file mode 100644 index 000000000..2c5e18d84 --- /dev/null +++ b/MapboxSearch/sdk/src/test/java/com/mapbox/search/RetrieveDetailsOptionsTest.kt @@ -0,0 +1,40 @@ +package com.mapbox.search + +import com.mapbox.search.base.core.CoreAttributeSet +import com.mapbox.search.common.IsoCountryCode +import com.mapbox.search.common.IsoLanguageCode +import com.mapbox.search.common.tests.ToStringVerifier +import com.mapbox.search.details.RetrieveDetailsOptions +import com.mapbox.search.details.mapToCore +import nl.jqno.equalsverifier.EqualsVerifier +import org.junit.jupiter.api.Assertions.assertEquals +import org.junit.jupiter.api.Test + +internal class RetrieveDetailsOptionsTest { + + @Test + fun `Test generated equals(), hashCode() and toString() methods`() { + EqualsVerifier.forClass(RetrieveDetailsOptions::class.java) + .verify() + + ToStringVerifier(RetrieveDetailsOptions::class).verify() + } + + @Test + fun `Test mapToCore() function`() { + val options = RetrieveDetailsOptions( + attributeSets = listOf(AttributeSet.BASIC, AttributeSet.VISIT), + language = IsoLanguageCode.FRENCH, + worldview = IsoCountryCode.FRANCE, + ) + + val coreOptions = options.mapToCore() + + assertEquals( + listOf(CoreAttributeSet.BASIC, CoreAttributeSet.VISIT), + coreOptions.attributeSets + ) + assertEquals(IsoLanguageCode.FRENCH.code, coreOptions.language) + assertEquals(IsoCountryCode.FRANCE.code, coreOptions.worldview) + } +} diff --git a/MapboxSearch/ui/src/main/java/com/mapbox/search/ui/adapter/engines/SearchEngineUiAdapter.kt b/MapboxSearch/ui/src/main/java/com/mapbox/search/ui/adapter/engines/SearchEngineUiAdapter.kt index bf3a719ad..3e376842c 100644 --- a/MapboxSearch/ui/src/main/java/com/mapbox/search/ui/adapter/engines/SearchEngineUiAdapter.kt +++ b/MapboxSearch/ui/src/main/java/com/mapbox/search/ui/adapter/engines/SearchEngineUiAdapter.kt @@ -12,6 +12,7 @@ import com.mapbox.common.ReachabilityInterface import com.mapbox.common.location.LocationProvider import com.mapbox.common.location.LocationServiceFactory import com.mapbox.geojson.Feature +import com.mapbox.search.AttributeSet import com.mapbox.search.ResponseInfo import com.mapbox.search.SearchEngine import com.mapbox.search.SearchOptions @@ -30,6 +31,9 @@ import com.mapbox.search.common.CompletionCallback import com.mapbox.search.common.OsmIdUtils import com.mapbox.search.common.concurrent.MainThreadWorker import com.mapbox.search.common.concurrent.SearchSdkMainThreadWorker +import com.mapbox.search.details.DetailsApi +import com.mapbox.search.details.DetailsApiSettings +import com.mapbox.search.details.RetrieveDetailsOptions import com.mapbox.search.internal.bindgen.UserActivityReporter import com.mapbox.search.offline.OfflineResponseInfo import com.mapbox.search.offline.OfflineSearchCallback @@ -350,7 +354,13 @@ public class SearchEngineUiAdapter( when (isOnlineSearch) { true -> { val mapboxId = feature.id()?.let { OsmIdUtils.fromPoiId(it) } - searchEngine.retrieve(mapboxId!!, searchCallback) + + DetailsApi.create(DetailsApiSettings()).retrieveDetails( + mapboxId!!, + RetrieveDetailsOptions(AttributeSet.values().toList()), + searchCallback + ) + //searchEngine.retrieve(mapboxId!!, searchCallback) } false -> { offlineSearchEngine.retrieve(feature, offlineSearchCallback)