2021-10-20 19:52:11 +02:00
|
|
|
import type { I18NConfig } from '../../config-shared'
|
2021-11-19 18:09:52 +01:00
|
|
|
import type { RequestData } from '../types'
|
2021-10-20 19:52:11 +02:00
|
|
|
import { NextURL } from '../next-url'
|
|
|
|
import { isBot } from '../../utils'
|
2022-05-30 14:01:36 +02:00
|
|
|
import { toNodeHeaders, validateURL } from '../utils'
|
2021-10-20 19:52:11 +02:00
|
|
|
import parseua from 'next/dist/compiled/ua-parser-js'
|
|
|
|
|
feat: better cookies API for Edge Functions (#36478)
This PR introduces a more predictable API to manipulate cookies in an Edge Function context.
```js
const response = new NextResponse()
// set a cookie
response.cookies.set('foo, 'bar') // => set-cookie: 'foo=bar; Path=/'`
// set another cookie
response.cookies.set('fooz, 'barz') // => set-cookie: 'foo=bar; Path=/, fooz=barz; Path=/'`
// delete a cookie means mark it as expired
response.cookies.delete('foo') // => set-cookie: 'foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, fooz=barz; Path=/'`
// clear all cookies means mark all of them as expired
response.cookies.clear() // => set-cookie: 'fooz=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT'`
```
This new cookies API uses [Map](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) interface, and it's available for `NextRequest` and `NextResponse`.
Additionally, you can pass a specific cookies option as a third argument in `set` method:
```js
response.cookies.set('foo', 'bar', {
path: '/',
maxAge: 60 * 60 * 24 * 7,
httpOnly: true,
sameSite: 'strict',
domain: 'example.com'
}
```
**Note**: `maxAge` it's in seconds rather than milliseconds.
Any cookie manipulation will be reflected over the `set-cookie` header, transparently.
closes #31719
2022-05-09 11:50:32 +02:00
|
|
|
import { NextCookies } from './cookies'
|
|
|
|
|
2021-10-20 19:52:11 +02:00
|
|
|
export const INTERNALS = Symbol('internal request')
|
|
|
|
|
|
|
|
export class NextRequest extends Request {
|
|
|
|
[INTERNALS]: {
|
feat: better cookies API for Edge Functions (#36478)
This PR introduces a more predictable API to manipulate cookies in an Edge Function context.
```js
const response = new NextResponse()
// set a cookie
response.cookies.set('foo, 'bar') // => set-cookie: 'foo=bar; Path=/'`
// set another cookie
response.cookies.set('fooz, 'barz') // => set-cookie: 'foo=bar; Path=/, fooz=barz; Path=/'`
// delete a cookie means mark it as expired
response.cookies.delete('foo') // => set-cookie: 'foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, fooz=barz; Path=/'`
// clear all cookies means mark all of them as expired
response.cookies.clear() // => set-cookie: 'fooz=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT'`
```
This new cookies API uses [Map](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) interface, and it's available for `NextRequest` and `NextResponse`.
Additionally, you can pass a specific cookies option as a third argument in `set` method:
```js
response.cookies.set('foo', 'bar', {
path: '/',
maxAge: 60 * 60 * 24 * 7,
httpOnly: true,
sameSite: 'strict',
domain: 'example.com'
}
```
**Note**: `maxAge` it's in seconds rather than milliseconds.
Any cookie manipulation will be reflected over the `set-cookie` header, transparently.
closes #31719
2022-05-09 11:50:32 +02:00
|
|
|
cookies: NextCookies
|
2021-11-19 18:09:52 +01:00
|
|
|
geo: RequestData['geo']
|
2021-10-20 19:52:11 +02:00
|
|
|
ip?: string
|
2022-04-14 01:08:53 +02:00
|
|
|
page?: { name?: string; params?: { [key: string]: string | string[] } }
|
2021-10-20 19:52:11 +02:00
|
|
|
ua?: UserAgent | null
|
|
|
|
url: NextURL
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor(input: Request | string, init: RequestInit = {}) {
|
2022-05-30 14:01:36 +02:00
|
|
|
const url = typeof input === 'string' ? input : input.url
|
|
|
|
validateURL(url)
|
2021-10-20 19:52:11 +02:00
|
|
|
super(input, init)
|
|
|
|
this[INTERNALS] = {
|
feat: better cookies API for Edge Functions (#36478)
This PR introduces a more predictable API to manipulate cookies in an Edge Function context.
```js
const response = new NextResponse()
// set a cookie
response.cookies.set('foo, 'bar') // => set-cookie: 'foo=bar; Path=/'`
// set another cookie
response.cookies.set('fooz, 'barz') // => set-cookie: 'foo=bar; Path=/, fooz=barz; Path=/'`
// delete a cookie means mark it as expired
response.cookies.delete('foo') // => set-cookie: 'foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, fooz=barz; Path=/'`
// clear all cookies means mark all of them as expired
response.cookies.clear() // => set-cookie: 'fooz=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT'`
```
This new cookies API uses [Map](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) interface, and it's available for `NextRequest` and `NextResponse`.
Additionally, you can pass a specific cookies option as a third argument in `set` method:
```js
response.cookies.set('foo', 'bar', {
path: '/',
maxAge: 60 * 60 * 24 * 7,
httpOnly: true,
sameSite: 'strict',
domain: 'example.com'
}
```
**Note**: `maxAge` it's in seconds rather than milliseconds.
Any cookie manipulation will be reflected over the `set-cookie` header, transparently.
closes #31719
2022-05-09 11:50:32 +02:00
|
|
|
cookies: new NextCookies(this),
|
2021-10-20 19:52:11 +02:00
|
|
|
geo: init.geo || {},
|
|
|
|
ip: init.ip,
|
|
|
|
page: init.page,
|
2022-05-30 14:01:36 +02:00
|
|
|
url: new NextURL(url, {
|
2021-10-20 19:52:11 +02:00
|
|
|
headers: toNodeHeaders(this.headers),
|
2022-05-27 20:29:04 +02:00
|
|
|
nextConfig: init.nextConfig,
|
2021-10-20 19:52:11 +02:00
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public get cookies() {
|
feat: better cookies API for Edge Functions (#36478)
This PR introduces a more predictable API to manipulate cookies in an Edge Function context.
```js
const response = new NextResponse()
// set a cookie
response.cookies.set('foo, 'bar') // => set-cookie: 'foo=bar; Path=/'`
// set another cookie
response.cookies.set('fooz, 'barz') // => set-cookie: 'foo=bar; Path=/, fooz=barz; Path=/'`
// delete a cookie means mark it as expired
response.cookies.delete('foo') // => set-cookie: 'foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, fooz=barz; Path=/'`
// clear all cookies means mark all of them as expired
response.cookies.clear() // => set-cookie: 'fooz=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT, foo=; Path=/; Expires=Thu, 01 Jan 1970 00:00:00 GMT'`
```
This new cookies API uses [Map](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) interface, and it's available for `NextRequest` and `NextResponse`.
Additionally, you can pass a specific cookies option as a third argument in `set` method:
```js
response.cookies.set('foo', 'bar', {
path: '/',
maxAge: 60 * 60 * 24 * 7,
httpOnly: true,
sameSite: 'strict',
domain: 'example.com'
}
```
**Note**: `maxAge` it's in seconds rather than milliseconds.
Any cookie manipulation will be reflected over the `set-cookie` header, transparently.
closes #31719
2022-05-09 11:50:32 +02:00
|
|
|
return this[INTERNALS].cookies
|
2021-10-20 19:52:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public get geo() {
|
|
|
|
return this[INTERNALS].geo
|
|
|
|
}
|
|
|
|
|
|
|
|
public get ip() {
|
|
|
|
return this[INTERNALS].ip
|
|
|
|
}
|
|
|
|
|
|
|
|
public get preflight() {
|
|
|
|
return this.headers.get('x-middleware-preflight')
|
|
|
|
}
|
|
|
|
|
|
|
|
public get nextUrl() {
|
|
|
|
return this[INTERNALS].url
|
|
|
|
}
|
|
|
|
|
|
|
|
public get page() {
|
|
|
|
return {
|
|
|
|
name: this[INTERNALS].page?.name,
|
|
|
|
params: this[INTERNALS].page?.params,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public get ua() {
|
|
|
|
if (typeof this[INTERNALS].ua !== 'undefined') {
|
|
|
|
return this[INTERNALS].ua || undefined
|
|
|
|
}
|
|
|
|
|
|
|
|
const uaString = this.headers.get('user-agent')
|
|
|
|
if (!uaString) {
|
|
|
|
this[INTERNALS].ua = null
|
|
|
|
return this[INTERNALS].ua || undefined
|
|
|
|
}
|
|
|
|
|
|
|
|
this[INTERNALS].ua = {
|
|
|
|
...parseua(uaString),
|
|
|
|
isBot: isBot(uaString),
|
|
|
|
}
|
|
|
|
|
|
|
|
return this[INTERNALS].ua
|
|
|
|
}
|
|
|
|
|
|
|
|
public get url() {
|
|
|
|
return this[INTERNALS].url.toString()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-26 00:59:41 +02:00
|
|
|
export interface RequestInit extends globalThis.RequestInit {
|
2021-10-20 19:52:11 +02:00
|
|
|
geo?: {
|
|
|
|
city?: string
|
|
|
|
country?: string
|
|
|
|
region?: string
|
|
|
|
}
|
|
|
|
ip?: string
|
|
|
|
nextConfig?: {
|
|
|
|
basePath?: string
|
|
|
|
i18n?: I18NConfig | null
|
|
|
|
trailingSlash?: boolean
|
|
|
|
}
|
|
|
|
page?: {
|
|
|
|
name?: string
|
2022-04-14 01:08:53 +02:00
|
|
|
params?: { [key: string]: string | string[] }
|
2021-10-20 19:52:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-30 22:59:02 +01:00
|
|
|
interface UserAgent {
|
2021-10-20 19:52:11 +02:00
|
|
|
isBot: boolean
|
2021-11-30 22:59:02 +01:00
|
|
|
ua: string
|
|
|
|
browser: {
|
|
|
|
name?: string
|
|
|
|
version?: string
|
|
|
|
}
|
|
|
|
device: {
|
|
|
|
model?: string
|
|
|
|
type?: string
|
|
|
|
vendor?: string
|
|
|
|
}
|
|
|
|
engine: {
|
|
|
|
name?: string
|
|
|
|
version?: string
|
|
|
|
}
|
|
|
|
os: {
|
|
|
|
name?: string
|
|
|
|
version?: string
|
|
|
|
}
|
|
|
|
cpu: {
|
|
|
|
architecture?: string
|
|
|
|
}
|
2021-10-20 19:52:11 +02:00
|
|
|
}
|