rsnext/packages/next-server/lib/utils.ts

258 lines
5.7 KiB
TypeScript
Raw Normal View History

import { format, UrlObject, URLFormatOptions } from 'url'
import { ServerResponse, IncomingMessage } from 'http'
import { ComponentType } from 'react'
import { ParsedUrlQuery } from 'querystring'
import { ManifestItem } from '../server/get-dynamic-import-bundles'
2019-04-26 09:37:57 +02:00
import { BaseRouter } from './router/router'
/**
* Types used by both next and next-server
*/
export type NextComponentType<
C extends BaseContext = NextPageContext,
IP = {},
P = {}
> = ComponentType<P> & {
getInitialProps?(context: C): Promise<IP>,
}
export type DocumentType = NextComponentType<
DocumentContext,
DocumentInitialProps,
DocumentProps
>
export type AppType = NextComponentType<
AppContextType,
AppInitialProps,
AppPropsType
>
export type Enhancer<C> = (Component: C) => C
export type ComponentsEnhancer =
| {
enhanceApp?: Enhancer<AppType>
enhanceComponent?: Enhancer<NextComponentType>,
}
| Enhancer<NextComponentType>
export type RenderPageResult = {
html: string
head?: Array<JSX.Element | null>
dataOnly?: true,
}
export type RenderPage = (
options?: ComponentsEnhancer,
) => RenderPageResult | Promise<RenderPageResult>
2019-04-26 09:37:57 +02:00
export type BaseContext = {
res?: ServerResponse
2019-04-26 09:37:57 +02:00
[k: string]: any,
}
export type NEXT_DATA = {
dataManager: string
props: any
page: string
query: ParsedUrlQuery
buildId: string
dynamicBuildId: boolean
assetPrefix?: string
runtimeConfig?: { [key: string]: any }
nextExport?: boolean
dynamicIds?: string[]
2019-04-26 09:37:57 +02:00
err?: Error & { statusCode?: number },
}
/**
* `Next` context
*/
// tslint:disable-next-line interface-name
export interface NextPageContext {
/**
* Error object if encountered during rendering
*/
err?: Error & { statusCode?: number } | null
/**
* `HTTP` request object.
*/
req?: IncomingMessage
/**
* `HTTP` response object.
*/
res?: ServerResponse
/**
* Path section of `URL`.
*/
pathname: string
/**
* Query string section of `URL` parsed as an object.
*/
query: ParsedUrlQuery
/**
* `String` of the actual path including query.
*/
asPath?: string
}
2019-04-26 09:37:57 +02:00
export type AppContextType<R extends BaseRouter = BaseRouter> = {
Component: NextComponentType<NextPageContext>
router: R
ctx: NextPageContext,
}
2019-04-26 09:37:57 +02:00
export type AppInitialProps = {
pageProps: any,
}
export type AppPropsType<
R extends BaseRouter = BaseRouter,
P = {}
> = AppInitialProps & {
Component: NextComponentType<NextPageContext, any, P>
2019-04-26 09:37:57 +02:00
router: R,
}
export type DocumentContext = NextPageContext & {
2019-04-26 09:37:57 +02:00
renderPage: RenderPage,
}
2019-04-26 09:37:57 +02:00
export type DocumentInitialProps = RenderPageResult & {
styles?: React.ReactElement[],
}
2019-04-26 09:37:57 +02:00
export type DocumentProps = DocumentInitialProps & {
__NEXT_DATA__: NEXT_DATA
dangerousAsPath: string
ampPath: string
amphtml: boolean
hasAmp: boolean
staticMarkup: boolean
devFiles: string[]
files: string[]
dynamicImports: ManifestItem[]
2019-04-26 09:37:57 +02:00
assetPrefix?: string,
canonicalBase: string,
}
/**
* Utils
*/
export function execOnce(this: any, fn: () => any) {
let used = false
return (...args: any) => {
if (!used) {
used = true
fn.apply(this, args)
}
}
}
export function getLocationOrigin() {
const { protocol, hostname, port } = window.location
return `${protocol}//${hostname}${port ? ':' + port : ''}`
}
export function getURL() {
const { href } = window.location
const origin = getLocationOrigin()
return href.substring(origin.length)
}
export function getDisplayName(Component: ComponentType<any>) {
return typeof Component === 'string'
? Component
: Component.displayName || Component.name || 'Unknown'
}
export function isResSent(res: ServerResponse) {
return res.finished || res.headersSent
}
export async function loadGetInitialProps<
C extends BaseContext,
IP = {},
P = {}
>(Component: NextComponentType<C, IP, P>, ctx: C): Promise<IP | null> {
if (process.env.NODE_ENV !== 'production') {
if (Component.prototype && Component.prototype.getInitialProps) {
const message = `"${getDisplayName(
Component,
)}.getInitialProps()" is defined as an instance method - visit https://err.sh/zeit/next.js/get-initial-props-as-an-instance-method for more information.`
throw new Error(message)
}
}
// when called from _app `ctx` is nested in `ctx`
const res = ctx.res || (ctx.ctx && ctx.ctx.res)
if (!Component.getInitialProps) {
return null
}
const props = await Component.getInitialProps(ctx)
if (res && isResSent(res)) {
return props
}
// if page component doesn't have getInitialProps
// set cache-control header to stale-while-revalidate
if (ctx.Component && !ctx.Component.getInitialProps) {
const customAppGetInitialProps = (Component as any).origGetInitialProps && (
(Component as any).origGetInitialProps !== Component.getInitialProps
)
if (!customAppGetInitialProps && res && res.setHeader) {
res.setHeader(
'Cache-Control', 's-maxage=86400, stale-while-revalidate',
)
}
}
if (!props) {
const message = `"${getDisplayName(
Component,
)}.getInitialProps()" should resolve to an object. But found "${props}" instead.`
throw new Error(message)
}
return props
}
export const urlObjectKeys = [
'auth',
'hash',
'host',
'hostname',
'href',
'path',
'pathname',
'port',
'protocol',
'query',
'search',
'slashes',
]
export function formatWithValidation(
url: UrlObject,
options?: URLFormatOptions,
) {
if (process.env.NODE_ENV === 'development') {
if (url !== null && typeof url === 'object') {
Object.keys(url).forEach((key) => {
if (urlObjectKeys.indexOf(key) === -1) {
console.warn(
`Unknown key passed via urlObject into url.format: ${key}`,
)
}
})
}
}
return format(url as any, options)
}