diff --git a/.changeset/rotten-nails-complain.md b/.changeset/rotten-nails-complain.md new file mode 100644 index 0000000000..f3eb3d0689 --- /dev/null +++ b/.changeset/rotten-nails-complain.md @@ -0,0 +1,44 @@ +--- +'@shopify/hydrogen': minor +--- + +The `setLogger` and `setLoggerOptions` utilities have been removed. The same information can now be passed under the `logger` property in Hydrogen config: + +```diff +// App.server.jsx + +-import {setLogger, setLoggerOptions} from '@shopify/hydrogen'; + +-setLogger({ +- trace() {}, +- error() {}, +- // ... +-}); + +-setLoggerOptions({ +- showQueryTiming: true, +- showCacheControlHeader: true, +- // ... +-}); + +function App() { + // ... +} + +export default renderHydrogen(App); +``` + +```diff +// hydrogen.config.js + +export default defineConfig({ + // ... ++ logger: { ++ trace() {}, ++ error() {}, ++ showQueryTiming: true, ++ showCacheControlHeader: true, ++ // ... ++ }, +}); +``` diff --git a/docs/framework/cache.md b/docs/framework/cache.md index 14aabd0082..231d6af3ff 100644 --- a/docs/framework/cache.md +++ b/docs/framework/cache.md @@ -152,25 +152,7 @@ export default defineConfig({ {% endcodeblock %} -To enable logging for the cache API status, call `setLoggerOptions` and set `showCacheApiStatus` to `true`: - -{% codeblock file, filename: '/src/App.server.jsx' %} - -```js -import renderHydrogen from '@shopify/hydrogen/entry-server'; -import {setLoggerOptions} from '@shopify/hydrogen'; - -setLoggerOptions({showCacheApiStatus: true}); - -function App() { - /* ... */ -} -// ... -``` - -{% endcodeblock %} - -The status of the cache updates on each query: +To enable logging for the cache API status, set `logger.showCacheApiStatus` to `true` in your [Hydrogen configuration file](https://shopify.dev/custom-storefronts/hydrogen/framework/hydrogen-config#logger). The status of the cache updates on each query: ```sh [Cache] MISS query shopInfo @@ -179,25 +161,7 @@ The status of the cache updates on each query: [Cache] MISS query Localization ``` -To enable logging for cache control headers, call `setLoggerOptions` and set `showCacheControlHeader` to `true`: - -{% codeblock file, filename: '/src/App.server.jsx' %} - -```js -import renderHydrogen from '@shopify/hydrogen/entry-server'; -import {setLoggerOptions} from '@shopify/hydrogen'; - -setLoggerOptions({showCacheControlHeader: true}); - -function App() { - /* ... */ -} -// ... -``` - -{% endcodeblock %} - -A cache control header report displays for each page request. The report includes the associated queries +To enable logging for cache control headers, set `logger.showCacheControlHeader` to `true` in your [Hydrogen configuration file](https://shopify.dev/custom-storefronts/hydrogen/framework/hydrogen-config#logger). A cache control header report displays for each page request. The report includes the associated queries that built the request and the cache control headers: ```sh diff --git a/docs/framework/hydrogen-config.md b/docs/framework/hydrogen-config.md index 0478f6c308..4a8f726b74 100644 --- a/docs/framework/hydrogen-config.md +++ b/docs/framework/hydrogen-config.md @@ -44,6 +44,7 @@ The following groupings of configuration properties can exist in Hydrogen: - [`session`](#session) - [`serverAnalyticsConnectors`](#serveranalyticsconnectors) - [`enableStreaming`](#enablestreaming) +- [`logger`](#logger) ### `routes` @@ -209,6 +210,37 @@ export default defineConfig({ > Tip: > There are [performance benefits](https://shopify.dev/custom-storefronts/hydrogen/best-practices/performance) to streaming. You shouldn't completely disable streaming for all of your storefront's routes. +## Logger + +The default behavior of the [`log` utility](https://shopify.dev/api/hydrogen/utilities/log) maps to the global `console` object. However, you can also customize this behavior in the configuration object. + +You can pass [any method](https://shopify.dev/api/hydrogen/utilities/log#methods) of the `log` utility in the `logger` object to override the default behavior. The first argument of each log method contains a `request` object if the log was called in the same context as a request. The following Boolean options are also available: + +{% codeblock file, filename: 'hydrogen.config.ts' %} + +```tsx +export default defineConfig({ + logger: { + /* Overrides the default `log.trace` behavior. */ + trace: (request, ...args) => console.log(request.url, ...args), + /* Overrides the default `log.error` behavior. */ + error: (request, error) => myErrorTrackingService.send(error, {request}), + /* ... */ + + /* Logs the cache status of each stored entry: `PUT`, `HIT`, `MISS` or `STALE`. */ + showCacheApiStatus: true, + /* Logs the cache control headers of the main document and its sub queries. */ + showCacheControlHeader: true, + /* Logs the timeline of when queries are being requested, resolved, and rendered. */ + showQueryTiming: true, + /* Logs warnings in your app if you're over-fetching data from the Storefront API. */ + showUnusedQueryProperties: true, + } +}); +``` + +{% endcodeblock %} + ## Changing the configuration file location If you don't want the Hydrogen configuration file located at the root of your project, then you can provide the new path to the file in the Hydrogen Vite plugin (`vite.config.js`): diff --git a/docs/framework/preloaded-queries.md b/docs/framework/preloaded-queries.md index 17270e319d..113eac890d 100644 --- a/docs/framework/preloaded-queries.md +++ b/docs/framework/preloaded-queries.md @@ -101,21 +101,9 @@ const data = fetchSync('https://my.api.com/data.json', { ## Test a preloaded query -To test a preloaded query, enable the `showQueryTiming` property in `App.server.js`. The [`showQueryTiming`](https://shopify.dev/api/hydrogen/utilities/log#logger-options) property logs the timeline of when queries are being requested, resolved, and rendered. +To test a preloaded query, enable the `logger.showQueryTiming` property in your [Hydrogen configuration file](https://shopify.dev/custom-storefronts/hydrogen/framework/hydrogen-config#logger). -{% codeblock file, filename: "App.server.js" %} - -```js -import {setLoggerOptions} from '@shopify/hydrogen'; -... -setLoggerOptions({ - showQueryTiming: true -}) -``` - -{% endcodeblock %} - -If a query is preloaded, but isn't being used, then a warning displays in the server log: +The [`showQueryTiming`](https://shopify.dev/api/hydrogen/utilities/log#logger-options) property logs the timeline of when queries are being requested, resolved, and rendered. If a query is preloaded, but isn't being used, then a warning displays in the server log: ![Shows a screenshot of preloaded query warning](/assets/custom-storefronts/hydrogen/preload-query-warning.png) diff --git a/docs/utilities/log.md b/docs/utilities/log.md index 487d408d48..f1604a388e 100644 --- a/docs/utilities/log.md +++ b/docs/utilities/log.md @@ -22,13 +22,9 @@ export default function Product({country = {isoCode: 'US'}, log}) { } ``` -## Arguments +## Methods -None - -## Return type - -Return an object with methods for logging information at different priorities: +The `log` utility exposes the following methods for logging information at different priorities: | Log method | Description | | ------------- | --------------------------------------------------------------------------------- | @@ -38,52 +34,7 @@ Return an object with methods for logging information at different priorities: | `log.error()` | The logging used for errors or invalid application state. | | `log.fatal()` | The logging used just prior to the process exiting. | -## Logger options - -Logger has the following Boolean options: +## Swap logger implementation and options -| Option | Description | -| --------------------------- | ------------------------------------------------------------------------------- | -| `showCacheApiStatus` | Logs the cache status of each stored entry: `PUT`, `HIT`, `MISS` or `STALE`. | -| `showCacheControlHeader` | Logs the cache control headers of the main document and its sub queries. | -| `showQueryTiming` | Logs the timeline of when queries are being requested, resolved, and rendered. | -| `showUnusedQueryProperties` | Logs warnings in your app if you're over-fetching data from the Storefront API. | +Hydrogen includes a default logger implementation that can be swapped for a logger of your choice. You can also show debugging information for cache and queries by providing extra options. For more information, refer to [Hydrogen configuration](https://shopify.dev/custom-storefronts/hydrogen/framework/hydrogen-config#logger). -### Example - -```js -import {setLoggerOptions} from '@shopify/hydrogen'; - -setLoggerOptions({ - showCacheApiStatus: true, - showCacheControlHeader: true, - showQueryTiming: true, - showUnusedQueryProperties: true, -}); -``` - -## Swap logger implementation - -Hydrogen includes a default logger implementation that can be swapped for a logger of your choice. You can call `setLogger` with your own implementation. The first argument of each log method will contain a `request` object if the log was called in the same context as a request: - -```js -import {setLogger} from '@shopify/hydrogen'; - -setLogger({ - trace(request, ...args) { - // Call your own logger. - }, - debug(request, ...args) { - // Call your own logger. - }, - warn(request, ...args) { - // Call your own logger. - }, - error(request, ...args) { - // Call your own logger. - }, - fatal(request, ...args) { - // Call your own logger. - }, -}); -``` diff --git a/packages/hydrogen/src/entry-server.tsx b/packages/hydrogen/src/entry-server.tsx index 9ac4aae2c4..3be00b2ee6 100644 --- a/packages/hydrogen/src/entry-server.tsx +++ b/packages/hydrogen/src/entry-server.tsx @@ -41,7 +41,7 @@ import { } from './streaming.server'; import {RSC_PATHNAME, EVENT_PATHNAME, EVENT_PATHNAME_REGEX} from './constants'; import {stripScriptsFromTemplate} from './utilities/template'; -import {RenderType} from './utilities/log/log'; +import {setLogger, RenderType} from './utilities/log/log'; import {Analytics} from './foundation/Analytics/Analytics.server'; import {ServerAnalyticsRoute} from './foundation/Analytics/ServerAnalyticsRoute.server'; import {getSyncSessionApi} from './foundation/session/session'; @@ -112,8 +112,10 @@ export const renderHydrogen = (App: any) => { request.ctx.hydrogenConfig = hydrogenConfig; request.ctx.buyerIpHeader = buyerIpHeader; - const response = new ServerComponentResponse(); + setLogger(hydrogenConfig.logger); const log = getLoggerWithContext(request); + + const response = new ServerComponentResponse(); const sessionApi = hydrogenConfig.session ? hydrogenConfig.session(log) : undefined; diff --git a/packages/hydrogen/src/index.ts b/packages/hydrogen/src/index.ts index 7587edf92e..9a26f038ba 100644 --- a/packages/hydrogen/src/index.ts +++ b/packages/hydrogen/src/index.ts @@ -17,7 +17,7 @@ export * from './foundation/useServerProps'; export {FileRoutes} from './foundation/FileRoutes/FileRoutes.server'; export {Route} from './foundation/Route/Route.server'; export {Router} from './foundation/Router/Router.server'; -export {log, setLogger, setLoggerOptions, Logger} from './utilities/log'; +export {log, type Logger} from './utilities/log'; export {LocalizationProvider} from './components/LocalizationProvider/LocalizationProvider.server'; export {ShopifyProvider} from './foundation/ShopifyProvider/ShopifyProvider.server'; export { diff --git a/packages/hydrogen/src/types.ts b/packages/hydrogen/src/types.ts index 427d23587a..c081625789 100644 --- a/packages/hydrogen/src/types.ts +++ b/packages/hydrogen/src/types.ts @@ -1,5 +1,5 @@ import type {ServerResponse} from 'http'; -import type {Logger} from './utilities/log/log'; +import type {Logger, LoggerConfig} from './utilities/log/log'; import type {ServerComponentRequest} from './framework/Hydration/ServerComponentRequest.server'; import type {ServerComponentResponse} from './framework/Hydration/ServerComponentResponse.server'; import type { @@ -84,6 +84,7 @@ export type InlineHydrogenConfig = { routes?: InlineHydrogenRoutes; shopify?: ShopifyConfig | ShopifyConfigFetcher; serverAnalyticsConnectors?: Array; + logger?: LoggerConfig; session?: (log: Logger) => SessionStorageAdapter; enableStreaming?: (request: ServerComponentRequest) => boolean; }; diff --git a/packages/hydrogen/src/utilities/log/__tests__/log-cache-api-status.test.ts b/packages/hydrogen/src/utilities/log/__tests__/log-cache-api-status.test.ts index 6b6a1ee2ae..6da35b1dbf 100644 --- a/packages/hydrogen/src/utilities/log/__tests__/log-cache-api-status.test.ts +++ b/packages/hydrogen/src/utilities/log/__tests__/log-cache-api-status.test.ts @@ -1,10 +1,4 @@ -import { - Logger, - setLogger, - logCacheApiStatus, - resetLogger, - setLoggerOptions, -} from '../index'; +import {Logger, setLogger, logCacheApiStatus} from '../index'; let mockLogger: jest.Mocked; @@ -19,14 +13,11 @@ describe('cache header log', () => { options: jest.fn(() => ({})), }; - setLogger(mockLogger); - setLoggerOptions({ - showCacheApiStatus: true, - }); + setLogger({...mockLogger, showCacheApiStatus: true}); }); afterEach(() => { - resetLogger(); + setLogger(undefined); }); it('should log cache api status', () => { diff --git a/packages/hydrogen/src/utilities/log/__tests__/log-cache-header.test.ts b/packages/hydrogen/src/utilities/log/__tests__/log-cache-header.test.ts index d22a9fe5dd..1590b5bec5 100644 --- a/packages/hydrogen/src/utilities/log/__tests__/log-cache-header.test.ts +++ b/packages/hydrogen/src/utilities/log/__tests__/log-cache-header.test.ts @@ -3,8 +3,6 @@ import { setLogger, logCacheControlHeaders, collectQueryCacheControlHeaders, - resetLogger, - setLoggerOptions, } from '../index'; import {ServerComponentRequest} from '../../../framework/Hydration/ServerComponentRequest.server'; import {ServerComponentResponse} from '../../../framework/Hydration/ServerComponentResponse.server'; @@ -26,14 +24,11 @@ describe('cache header log', () => { options: jest.fn(() => ({})), }; - setLogger(mockLogger); - setLoggerOptions({ - showCacheControlHeader: true, - }); + setLogger({...mockLogger, showCacheControlHeader: true}); }); afterEach(() => { - resetLogger(); + setLogger(undefined); }); it('should log cache control header for main request', () => { diff --git a/packages/hydrogen/src/utilities/log/__tests__/log-query-timeline.test.ts b/packages/hydrogen/src/utilities/log/__tests__/log-query-timeline.test.ts index c138f26e8c..77b46e5970 100644 --- a/packages/hydrogen/src/utilities/log/__tests__/log-query-timeline.test.ts +++ b/packages/hydrogen/src/utilities/log/__tests__/log-query-timeline.test.ts @@ -1,5 +1,5 @@ import {ServerComponentRequest} from '../../../framework/Hydration/ServerComponentRequest.server'; -import {Logger, setLogger, resetLogger, setLoggerOptions} from '../index'; +import {Logger, setLogger} from '../log'; import {collectQueryTimings, logQueryTimings} from '../log-query-timeline'; let mockLogger: jest.Mocked; @@ -36,14 +36,11 @@ describe('cache header log', () => { options: jest.fn(() => ({})), }; - setLogger(mockLogger); - setLoggerOptions({ - showQueryTiming: true, - }); + setLogger({...mockLogger, showQueryTiming: true}); }); afterEach(() => { - resetLogger(); + setLogger(undefined); }); it('should log query timing', () => { diff --git a/packages/hydrogen/src/utilities/log/__tests__/log.test.ts b/packages/hydrogen/src/utilities/log/__tests__/log.test.ts index d11c64a958..03302a51ee 100644 --- a/packages/hydrogen/src/utilities/log/__tests__/log.test.ts +++ b/packages/hydrogen/src/utilities/log/__tests__/log.test.ts @@ -4,10 +4,8 @@ import { Logger, logServerResponse, getLoggerWithContext, - resetLogger, } from '../log'; import {ServerComponentRequest} from '../../../framework/Hydration/ServerComponentRequest.server'; -import {setLoggerOptions} from '..'; let mockLogger: jest.Mocked; @@ -29,7 +27,7 @@ describe('log', () => { }); afterEach(() => { - resetLogger(); + setLogger(undefined); }); it('should return the wrapped mockLogger instance when log is called', () => { @@ -47,11 +45,10 @@ describe('log', () => { warn: jest.fn(), error: jest.fn(), fatal: jest.fn(), - options: jest.fn(() => ({ - showCacheControlHeader: true, - })), + options: jest.fn(() => ({})), }; - setLogger(mockLogger2); + + setLogger({...mockLogger2, showCacheControlHeader: true}); log.debug('test'); expect(mockLogger2.debug).toHaveBeenCalled(); @@ -62,17 +59,15 @@ describe('log', () => { expect(mockLogger2.debug.mock.calls[0][1]).toEqual('test'); }); - it('should set showCacheControlHeader option when setLoggerOptions is called', () => { - setLoggerOptions({ - showCacheControlHeader: true, - }); + it('should set showCacheControlHeader option correctly', () => { + setLogger({showCacheControlHeader: true}); expect(log.options()).toEqual({ showCacheControlHeader: true, }); }); - it('should set showCacheApiStatus option when setLoggerOptions is called', () => { - setLoggerOptions({ + it('should set showCacheApiStatus option correctly', () => { + setLogger({ showCacheApiStatus: true, }); expect(log.options()).toEqual({ @@ -80,14 +75,14 @@ describe('log', () => { }); }); - it('should return the correct options when setLoggerOptions is set multiple times', () => { - setLoggerOptions({ + it('should set multiple options correctly', () => { + setLogger({ showCacheControlHeader: true, }); expect(log.options()).toEqual({ showCacheControlHeader: true, }); - setLoggerOptions({ + setLogger({ showCacheApiStatus: true, showCacheControlHeader: true, }); diff --git a/packages/hydrogen/src/utilities/log/index.ts b/packages/hydrogen/src/utilities/log/index.ts index 85d52a95d6..c5821679b3 100644 --- a/packages/hydrogen/src/utilities/log/index.ts +++ b/packages/hydrogen/src/utilities/log/index.ts @@ -1,11 +1,9 @@ export { log, setLogger, - setLoggerOptions, getLoggerWithContext, Logger, logServerResponse, - resetLogger, } from './log'; export { collectQueryCacheControlHeaders, diff --git a/packages/hydrogen/src/utilities/log/log.ts b/packages/hydrogen/src/utilities/log/log.ts index b87370b7e0..dc72ec1b85 100644 --- a/packages/hydrogen/src/utilities/log/log.ts +++ b/packages/hydrogen/src/utilities/log/log.ts @@ -9,10 +9,6 @@ import {parseUrl} from './utils'; * current request in progress. */ -declare namespace globalThis { - let __logger: Logger; -} - export interface Logger { trace: (...args: Array) => void; debug: (...args: Array) => void; @@ -29,57 +25,60 @@ export type LoggerOptions = { showUnusedQueryProperties?: boolean; }; +export type LoggerConfig = Partial> & LoggerOptions; + export type RenderType = 'str' | 'rsc' | 'ssr' | 'api'; -const defaultLogger = { - trace(context: {[key: string]: any}, ...args: Array) { +const defaultLogger: Logger = { + trace(context, ...args) { // Re-enable following line to show trace debugging information // console.log(context.id, ...args); }, - debug(context: {[key: string]: any}, ...args: Array) { + debug(context, ...args) { console.log(...args); }, - warn(context: {[key: string]: any}, ...args: Array) { + warn(context, ...args) { console.warn(yellow('WARN: '), ...args); }, - error(context: {[key: string]: any}, ...args: Array) { + error(context, ...args) { console.error(red('ERROR: '), ...args); }, - fatal(context: {[key: string]: any}, ...args: Array) { + fatal(context, ...args) { console.error(red('FATAL: '), ...args); }, - options: () => ({}), + options: () => ({} as LoggerOptions), }; -globalThis.__logger = defaultLogger as Logger; +let currentLogger = defaultLogger as Logger; -function buildLogger(this: any): Logger { +export function getLoggerWithContext(context: any): Logger { return { - trace: (...args) => globalThis.__logger.trace(this, ...args), - debug: (...args) => globalThis.__logger.debug(this, ...args), - warn: (...args) => globalThis.__logger.warn(this, ...args), - error: (...args) => globalThis.__logger.error(this, ...args), - fatal: (...args) => globalThis.__logger.fatal(this, ...args), - options: () => globalThis.__logger.options(), + trace: (...args) => currentLogger.trace(context, ...args), + debug: (...args) => currentLogger.debug(context, ...args), + warn: (...args) => currentLogger.warn(context, ...args), + error: (...args) => currentLogger.error(context, ...args), + fatal: (...args) => currentLogger.fatal(context, ...args), + options: () => currentLogger.options(), }; } -export const log: Logger = buildLogger.call({}); +export const log: Logger = getLoggerWithContext({}); -export function getLoggerWithContext(context: any = {}): Logger { - return buildLogger.call(context); -} +export function setLogger(config?: LoggerConfig) { + if (!config) { + currentLogger = defaultLogger; + return; + } -export function setLogger(newLogger: Logger) { - globalThis.__logger = newLogger; -} - -export function setLoggerOptions(options: LoggerOptions) { - globalThis.__logger.options = () => options; -} + const options = {} as LoggerOptions; + currentLogger = {...defaultLogger, ...config, options: () => options}; -export function resetLogger() { - globalThis.__logger = defaultLogger; + for (const key of Object.keys(config) as (keyof LoggerOptions)[]) { + if (!(key in defaultLogger)) { + delete currentLogger[key as keyof Logger]; + options[key] = config[key]; + } + } } const SERVER_RESPONSE_MAP: Record = { diff --git a/packages/playground/async-config/hydrogen.config.js b/packages/playground/async-config/hydrogen.config.js index 5c0b1ea803..8a27083968 100644 --- a/packages/playground/async-config/hydrogen.config.js +++ b/packages/playground/async-config/hydrogen.config.js @@ -16,4 +16,8 @@ export default defineConfig({ storefrontApiVersion: '2022-07', }; }, + logger: { + trace() {}, + debug() {}, + }, }); diff --git a/packages/playground/async-config/src/App.server.jsx b/packages/playground/async-config/src/App.server.jsx index 0a45c54cd5..b1cd8b62c5 100644 --- a/packages/playground/async-config/src/App.server.jsx +++ b/packages/playground/async-config/src/App.server.jsx @@ -1,28 +1,7 @@ import renderHydrogen from '@shopify/hydrogen/entry-server'; -import { - Router, - FileRoutes, - ShopifyProvider, - setLogger, - useUrl, -} from '@shopify/hydrogen'; +import {Router, FileRoutes, ShopifyProvider, useUrl} from '@shopify/hydrogen'; import {Suspense} from 'react'; -setLogger({ - trace() {}, - debug() {}, - warn(context, ...args) { - console.warn(...args); - }, - error(context, ...args) { - console.error(...args); - }, - fatal(context, ...args) { - console.error(...args); - }, - options: () => ({}), -}); - let localeRedirects; const handleRequest = renderHydrogen(({response}) => { diff --git a/packages/playground/server-components/hydrogen.config.js b/packages/playground/server-components/hydrogen.config.js index 532110a3b8..fb8e045c47 100644 --- a/packages/playground/server-components/hydrogen.config.js +++ b/packages/playground/server-components/hydrogen.config.js @@ -14,4 +14,8 @@ export default defineConfig({ enableStreaming: (req) => { return req.headers.get('user-agent') !== 'custom bot'; }, + logger: { + trace() {}, + debug() {}, + }, }); diff --git a/packages/playground/server-components/src/App.server.jsx b/packages/playground/server-components/src/App.server.jsx index 0daa8e181b..5627804a2f 100644 --- a/packages/playground/server-components/src/App.server.jsx +++ b/packages/playground/server-components/src/App.server.jsx @@ -1,32 +1,11 @@ import renderHydrogen from '@shopify/hydrogen/entry-server'; -import { - Route, - Router, - FileRoutes, - ShopifyProvider, - setLogger, -} from '@shopify/hydrogen'; +import {Route, Router, FileRoutes, ShopifyProvider} from '@shopify/hydrogen'; import {Suspense} from 'react'; import Custom1 from './customRoutes/custom1.server'; import Custom2 from './customRoutes/custom2.server'; import LazyRoute from './customRoutes/lazyRoute.server'; import ServerParams from './customRoutes/params.server'; -setLogger({ - trace() {}, - debug() {}, - warn(context, ...args) { - console.warn(...args); - }, - error(context, ...args) { - console.error(...args); - }, - fatal(context, ...args) { - console.error(...args); - }, - options: () => ({}), -}); - export default renderHydrogen(() => { return (