diff --git a/.cspell/en-words.txt b/.cspell/en-words.txt index aa514245f43a..78a77f7d49b9 100644 --- a/.cspell/en-words.txt +++ b/.cspell/en-words.txt @@ -108,6 +108,7 @@ quantile quantiles quarkus quoteservice +react-native-app recommendationservice redis relref diff --git a/content/en/docs/demo/_index.md b/content/en/docs/demo/_index.md index 5eaecbffd280..d21b43875c50 100644 --- a/content/en/docs/demo/_index.md +++ b/content/en/docs/demo/_index.md @@ -29,7 +29,8 @@ here. | C++ | | | [Currency Service](services/currency/) | | Go | | [Checkout Service](services/checkout/), [Product Catalog Service](services/product-catalog/) | [Checkout Service](services/checkout/), [Product Catalog Service](services/product-catalog/) | | Java | [Ad Service](services/ad/) | | [Ad Service](services/ad/) | -| JavaScript | | [Frontend](services/frontend/) | [Frontend](services/frontend/), [Payment Service](services/payment/) | +| JavaScript | | | [Payment Service](services/payment/) | +| TypeScript | | [Frontend](services/frontend/), [React Native App](services/react-native-app/) | [Frontend](services/frontend/) | | Kotlin | | [Fraud Detection Service](services/fraud-detection/) | | | PHP | | [Quote Service](services/quote/) | [Quote Service](services/quote/) | | Python | [Recommendation Service](services/recommendation/) | | [Recommendation Service](services/recommendation/) | @@ -54,6 +55,7 @@ found here: - [Recommendation Service](services/recommendation/) - [Shipping Service](services/shipping/) - [Image Provider Service](services/imageprovider/) +- [React Native App](services/react-native-app/) ## Scenarios diff --git a/content/en/docs/demo/architecture.md b/content/en/docs/demo/architecture.md index 245ff539c225..b3fd68c5cabe 100644 --- a/content/en/docs/demo/architecture.md +++ b/content/en/docs/demo/architecture.md @@ -32,6 +32,7 @@ quoteservice(Quote Service):::php recommendationservice(Recommendation Service):::python shippingservice(Shipping Service):::rust queue[(queue
(Kafka))]:::java +react-native-app(React Native App):::typescript adservice ---->|gRPC| flagd @@ -73,6 +74,8 @@ recommendationservice -->|gRPC| productcatalogservice recommendationservice -->|gRPC| flagd shippingservice -->|HTTP| quoteservice + +react-native-app -->|HTTP| frontendproxy end classDef dotnet fill:#178600,color:white; diff --git a/content/en/docs/demo/services/_index.md b/content/en/docs/demo/services/_index.md index b09a35f1c016..12ae06ed9b98 100644 --- a/content/en/docs/demo/services/_index.md +++ b/content/en/docs/demo/services/_index.md @@ -15,10 +15,11 @@ To visualize request flows, see the [Service Diagram](../architecture/). | [currencyservice](currency/) | C++ | Converts one money amount to another currency. Uses real values fetched from European Central Bank. It's the highest QPS service. | | [emailservice](email/) | Ruby | Sends users an order confirmation email (mock/). | | [frauddetectionservice](fraud-detection/) | Kotlin | Analyzes incoming orders and detects fraud attempts (mock/). | -| [frontend](frontend/) | JavaScript | Exposes an HTTP server to serve the website. Does not require sign up / login and generates session IDs for all users automatically. | +| [frontend](frontend/) | TypeScript | Exposes an HTTP server to serve the website. Does not require sign up / login and generates session IDs for all users automatically. | | [loadgenerator](load-generator/) | Python/Locust | Continuously sends requests imitating realistic user shopping flows to the frontend. | | [paymentservice](payment/) | JavaScript | Charges the given credit card info (mock/) with the given amount and returns a transaction ID. | | [productcatalogservice](product-catalog/) | Go | Provides the list of products from a JSON file and ability to search products and get individual products. | | [quoteservice](quote/) | PHP | Calculates the shipping costs, based on the number of items to be shipped. | | [recommendationservice](recommendation/) | Python | Recommends other products based on what's given in the cart. | | [shippingservice](shipping/) | Rust | Gives shipping cost estimates based on the shopping cart. Ships items to the given address (mock/). | +| [react-native-app](react-native-app/) | TypeScript | React Native mobile application that provides a UI on top of the shopping services. | diff --git a/content/en/docs/demo/services/react-native-app.md b/content/en/docs/demo/services/react-native-app.md new file mode 100644 index 000000000000..6b4f9ca8d261 --- /dev/null +++ b/content/en/docs/demo/services/react-native-app.md @@ -0,0 +1,149 @@ +--- +title: React Native App +cSpell:ignore: typeof +--- + +The React Native app provides a mobile UI for users on Android and iOS devices +to interact with the demo's services. It is built with +[Expo](https://docs.expo.dev/get-started/introduction/) and uses Expo's +file-based routing to layout the screens for the app. + +[React Native app source](https://github.com/open-telemetry/opentelemetry-demo/blob/main/src/react-native-app/) + +## Instrumentation + +The application uses the OpenTelemetry packages to instrument the application at +the JS layer. + +{{% alert title="Important" color="warning" %}} + +The JS OTel packages are supported for node and web environments. While they +work for React Native as well, they are not explicitly supported for that +environment, where they might break compatibility with minor version updates or +require workarounds. Building JS OTel package support for React Native is an +area of active development. + +{{% /alert %}} + +The main entry point for the application is `app/_layout.tsx` where a hook is +used to initialize the instrumentation and make sure it is loaded before +displaying the UI: + +```typescript +import { useTracer } from '@/hooks/useTracer'; + +const { loaded: tracerLoaded } = useTracer(); +``` + +`hooks/useTracer.ts` contains all the code for setting up instrumentation +including initializing a TracerProvider, establishing an OTLP export, +registering trace context propagators, and registering auto-instrumentation of +network requests. + +```typescript +import { + CompositePropagator, + W3CBaggagePropagator, + W3CTraceContextPropagator, +} from '@opentelemetry/core'; +import { WebTracerProvider } from '@opentelemetry/sdk-trace-web'; +import { BatchSpanProcessor } from '@opentelemetry/sdk-trace-base'; +import { XMLHttpRequestInstrumentation } from '@opentelemetry/instrumentation-xml-http-request'; +import { FetchInstrumentation } from '@opentelemetry/instrumentation-fetch'; +import { registerInstrumentations } from '@opentelemetry/instrumentation'; +import { Resource } from '@opentelemetry/resources'; +import { + ATTR_DEVICE_ID, + ATTR_OS_NAME, + ATTR_OS_VERSION, + ATTR_SERVICE_NAME, + ATTR_SERVICE_VERSION, +} from '@opentelemetry/semantic-conventions/incubating'; +import { OTLPTraceExporter } from '@opentelemetry/exporter-trace-otlp-http'; +import getLocalhost from '@/utils/Localhost'; +import { useEffect, useState } from 'react'; +import { + getDeviceId, + getSystemVersion, + getVersion, +} from 'react-native-device-info'; +import { Platform } from 'react-native'; +import { SessionIdProcessor } from '@/utils/SessionIdProcessor'; + +const Tracer = async () => { + const localhost = await getLocalhost(); + + const resource = new Resource({ + [ATTR_SERVICE_NAME]: 'react-native-app', + [ATTR_OS_NAME]: Platform.OS, + [ATTR_OS_VERSION]: getSystemVersion(), + [ATTR_SERVICE_VERSION]: getVersion(), + [ATTR_DEVICE_ID]: getDeviceId(), + }); + + const provider = new WebTracerProvider({ + resource, + spanProcessors: [ + new BatchSpanProcessor( + new OTLPTraceExporter({ + url: `http://${localhost}:${process.env.EXPO_PUBLIC_FRONTEND_PROXY_PORT}/otlp-http/v1/traces`, + }), + { + scheduledDelayMillis: 500, + }, + ), + new SessionIdProcessor(), + ], + }); + + provider.register({ + propagator: new CompositePropagator({ + propagators: [ + new W3CBaggagePropagator(), + new W3CTraceContextPropagator(), + ], + }), + }); + + registerInstrumentations({ + instrumentations: [ + // Some tiptoeing required here, propagateTraceHeaderCorsUrls is required to make the instrumentation + // work in the context of a mobile app even though we are not making CORS requests. `clearTimingResources` must + // be turned off to avoid using the web-only Performance API + new FetchInstrumentation({ + propagateTraceHeaderCorsUrls: /.*/, + clearTimingResources: false, + }), + + // The React Native implementation of fetch is simply a polyfill on top of XMLHttpRequest: + // https://github.com/facebook/react-native/blob/7ccc5934d0f341f9bc8157f18913a7b340f5db2d/packages/react-native/Libraries/Network/fetch.js#L17 + // Because of this when making requests using `fetch` there will an additional span created for the underlying + // request made with XMLHttpRequest. Since in this demo calls to /api/ are made using fetch, turn off + // instrumentation for that path to avoid the extra spans. + new XMLHttpRequestInstrumentation({ + ignoreUrls: [/\/api\/.*/], + }), + ], + }); +}; + +export interface TracerResult { + loaded: boolean; +} + +export const useTracer = (): TracerResult => { + const [loaded, setLoaded] = useState(false); + + useEffect(() => { + if (!loaded) { + Tracer() + .catch(() => console.warn('failed to setup tracer')) + .finally(() => setLoaded(true)); + } + }, [loaded]); + + return { + loaded, + }; +}; +```