Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

✨ [RUMF-764] Use new intake domain for US #616

Merged
merged 23 commits into from
Dec 3, 2020
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
569e0aa
🏷️ [RUM] Use enum for endpoint type
webNeat Nov 13, 2020
723bd92
✨ [RUM] Use new intake domain for US
webNeat Nov 13, 2020
7f89562
🎨 [RUM] Refactor isIntakeRequest()
webNeat Nov 16, 2020
3a75832
✨ [RUM] Support legacy endpoints
webNeat Nov 16, 2020
cc99182
🎨 [RUM] Use indexOf() instead of includes() to support IE 11
webNeat Nov 17, 2020
73f4c35
🎨 [RUM] Precompute intake urls
webNeat Nov 20, 2020
7ccedc6
🎨 [RUM] Use .some() instead of for..of
webNeat Nov 23, 2020
87228b9
🎨 [RUM] Remove unused imports
webNeat Nov 23, 2020
9d75102
🎨 [RUM] Replace isIntakeRequest by configuration.isIntakeUrl
webNeat Nov 23, 2020
d187810
✨ [RUM] Add configuration option to use new intake domains
webNeat Nov 25, 2020
ea48e31
🎨 [RUM] Remove uneeded lines
webNeat Nov 26, 2020
16d83b8
🎨 [RUM] Use either new or old intake domains
webNeat Nov 26, 2020
5bee248
🎨 [RUM] Rename configuration option to useAlternateIntakeDomains
webNeat Nov 27, 2020
c843fb3
Merge branch 'master' into amine/use-public-intake
webNeat Nov 27, 2020
927d7df
✅ [RUM] Fix unit tests
webNeat Nov 27, 2020
4fc29bc
👌 Remove list of allowed sites in favor of specifying them in docs
webNeat Nov 27, 2020
7fd2e40
🏷️ Refactor types to account for alternate intake
webNeat Nov 30, 2020
fd3c833
🎨 Refactor code to eliminate duplication
webNeat Dec 1, 2020
0758cb1
👌 Remove uneeded property
webNeat Dec 1, 2020
48a0ffc
👌 Handle sites with subdomains
webNeat Dec 2, 2020
4ba25a4
👌 Support classic intake for sites with subdomains
webNeat Dec 2, 2020
e3fe145
👌 Simplify the code
webNeat Dec 2, 2020
812b230
Merge branch 'master' into amine/use-public-intake
webNeat Dec 2, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions packages/core/src/boot/init.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,8 @@ export const INTAKE_SITE = {
[Datacenter.US]: 'datadoghq.com',
}

export const NEW_INTAKE_DOMAIN_ALLOWED_SITES = [INTAKE_SITE[Datacenter.US], 'datad0g.com']
bcaudan marked this conversation as resolved.
Show resolved Hide resolved

export enum BuildMode {
RELEASE = 'release',
STAGING = 'staging',
Expand Down
4 changes: 2 additions & 2 deletions packages/core/src/domain/automaticErrorCollection.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { resetXhrProxy, startXhrProxy, XhrCompleteContext } from '../browser/xhr
import { ErrorSource, formatUnknownError, RawError, toStackTraceString } from '../tools/error'
import { Observable } from '../tools/observable'
import { jsonStringify, ONE_MINUTE, RequestType } from '../tools/utils'
import { Configuration, isIntakeRequest } from './configuration'
import { Configuration } from './configuration'
import { monitor } from './internalMonitoring'
import { computeStackTrace, Handler, report, StackTrace } from './tracekit'

Expand Down Expand Up @@ -94,7 +94,7 @@ export function trackNetworkError(configuration: Configuration, errorObservable:
startFetchProxy().onRequestComplete((context) => handleCompleteRequest(RequestType.FETCH, context))

function handleCompleteRequest(type: RequestType, request: XhrCompleteContext | FetchCompleteContext) {
if (!isIntakeRequest(request.url, configuration) && (isRejected(request) || isServerError(request))) {
if (!configuration.isIntakeUrl(request.url) && (isRejected(request) || isServerError(request))) {
errorObservable.notify({
message: `${format(type)} error ${request.method} ${request.url}`,
resource: {
Expand Down
77 changes: 58 additions & 19 deletions packages/core/src/domain/configuration.spec.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { BuildEnv, BuildMode, Datacenter } from '../boot/init'
import { buildConfiguration, isIntakeRequest } from './configuration'
import { buildConfiguration } from './configuration'

describe('configuration', () => {
const clientToken = 'some_client_token'
Expand Down Expand Up @@ -52,9 +52,9 @@ describe('configuration', () => {

describe('proxyHost', () => {
it('should replace endpoint host add set it as a query parameter', () => {
const configuration = buildConfiguration({ clientToken, proxyHost: 'proxy.io' }, usEnv)
const configuration = buildConfiguration({ clientToken, site: 'datadoghq.eu', proxyHost: 'proxy.io' }, usEnv)
expect(configuration.rumEndpoint).toMatch(/^https:\/\/proxy\.io\//)
expect(configuration.rumEndpoint).toContain('?ddhost=rum-http-intake.logs.datadoghq.com&')
expect(configuration.rumEndpoint).toContain('?ddhost=rum-http-intake.logs.datadoghq.eu&')
})
})

Expand Down Expand Up @@ -104,44 +104,83 @@ describe('configuration', () => {
})
})

describe('isIntakeRequest', () => {
describe('isIntakeUrl', () => {
it('should not detect non intake request', () => {
const configuration = buildConfiguration({ clientToken }, usEnv)
expect(isIntakeRequest('https://www.foo.com', configuration)).toBe(false)
expect(configuration.isIntakeUrl('https://www.foo.com')).toBe(false)
})

it('should detect intake request', () => {
it('should detect intake request for EU site', () => {
const configuration = buildConfiguration({ clientToken, site: 'datadoghq.eu' }, usEnv)
expect(configuration.isIntakeUrl('https://rum-http-intake.logs.datadoghq.eu/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.datadoghq.eu/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://public-trace-http-intake.logs.datadoghq.eu/v1/input/xxx')).toBe(true)
})

it('should detect intake request for US site', () => {
const configuration = buildConfiguration({ clientToken }, usEnv)
expect(isIntakeRequest('https://rum-http-intake.logs.datadoghq.com/v1/input/xxx', configuration)).toBe(true)
expect(isIntakeRequest('https://browser-http-intake.logs.datadoghq.com/v1/input/xxx', configuration)).toBe(true)
expect(isIntakeRequest('https://public-trace-http-intake.logs.datadoghq.com/v1/input/xxx', configuration)).toBe(
true
)
// expect(configuration.isIntakeUrl('https://rum.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
// expect(configuration.isIntakeUrl('https://logs.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
// expect(configuration.isIntakeUrl('https://trace.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
bcaudan marked this conversation as resolved.
Show resolved Hide resolved

expect(configuration.isIntakeUrl('https://rum-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://public-trace-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
})

it('should detect new intake domains for US site', () => {
bcaudan marked this conversation as resolved.
Show resolved Hide resolved
const configuration = buildConfiguration({ clientToken, useNewIntakeDomains: true }, usEnv)
expect(configuration.isIntakeUrl('https://rum.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://logs.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://trace.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)

expect(configuration.isIntakeUrl('https://rum-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://public-trace-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
})

it('should detect proxy intake request', () => {
let configuration = buildConfiguration({ clientToken, proxyHost: 'www.proxy.com' }, usEnv)
expect(isIntakeRequest('https://www.proxy.com/v1/input/xxx', configuration)).toBe(true)
expect(configuration.isIntakeUrl('https://www.proxy.com/v1/input/xxx')).toBe(true)
configuration = buildConfiguration({ clientToken, proxyHost: 'www.proxy.com', useNewIntakeDomains: true }, usEnv)
expect(configuration.isIntakeUrl('https://www.proxy.com/v1/input/xxx')).toBe(true)
configuration = buildConfiguration({ clientToken, proxyHost: 'www.proxy.com/custom/path' }, usEnv)
expect(isIntakeRequest('https://www.proxy.com/custom/path/v1/input/xxx', configuration)).toBe(true)
expect(configuration.isIntakeUrl('https://www.proxy.com/custom/path/v1/input/xxx')).toBe(true)
})

it('should not detect request done on the same host as the proxy', () => {
const configuration = buildConfiguration({ clientToken, proxyHost: 'www.proxy.com' }, usEnv)
expect(isIntakeRequest('https://www.proxy.com/foo', configuration)).toBe(false)
expect(configuration.isIntakeUrl('https://www.proxy.com/foo')).toBe(false)
})

it('should detect replica intake request', () => {
const configuration = buildConfiguration(
{ clientToken, site: 'foo.com', replica: { clientToken } },
{ ...usEnv, buildMode: BuildMode.STAGING }
)
expect(isIntakeRequest('https://rum-http-intake.logs.foo.com/v1/input/xxx', configuration)).toBe(true)
expect(isIntakeRequest('https://browser-http-intake.logs.foo.com/v1/input/xxx', configuration)).toBe(true)
expect(isIntakeRequest('https://public-trace-http-intake.logs.foo.com/v1/input/xxx', configuration)).toBe(true)
expect(configuration.isIntakeUrl('https://rum-http-intake.logs.foo.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.foo.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://public-trace-http-intake.logs.foo.com/v1/input/xxx')).toBe(true)

// expect(configuration.isIntakeUrl('https://rum.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
// expect(configuration.isIntakeUrl('https://logs.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
bcaudan marked this conversation as resolved.
Show resolved Hide resolved
expect(configuration.isIntakeUrl('https://rum-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
})

expect(isIntakeRequest('https://rum-http-intake.logs.datadoghq.com/v1/input/xxx', configuration)).toBe(true)
expect(isIntakeRequest('https://browser-http-intake.logs.datadoghq.com/v1/input/xxx', configuration)).toBe(true)
it('should detect replica intake request with new domains', () => {
const configuration = buildConfiguration(
{ clientToken, site: 'foo.com', replica: { clientToken }, useNewIntakeDomains: true },
{ ...usEnv, buildMode: BuildMode.STAGING }
)
expect(configuration.isIntakeUrl('https://rum-http-intake.logs.foo.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.foo.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://public-trace-http-intake.logs.foo.com/v1/input/xxx')).toBe(true)

expect(configuration.isIntakeUrl('https://rum.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://logs.browser-intake-datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://rum-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
expect(configuration.isIntakeUrl('https://browser-http-intake.logs.datadoghq.com/v1/input/xxx')).toBe(true)
})
})
})
82 changes: 59 additions & 23 deletions packages/core/src/domain/configuration.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { BuildEnv, BuildMode, Datacenter, INTAKE_SITE } from '../boot/init'
import { BuildEnv, BuildMode, Datacenter, INTAKE_SITE, NEW_INTAKE_DOMAIN_ALLOWED_SITES } from '../boot/init'
import { CookieOptions, getCurrentSite } from '../browser/cookie'
import { getPathName, haveSameOrigin } from '../tools/urlPolyfill'
import { includes, ONE_KILO_BYTE, ONE_SECOND } from '../tools/utils'

export const DEFAULT_CONFIGURATION = {
Expand Down Expand Up @@ -55,6 +54,7 @@ export interface UserConfiguration {
env?: string
version?: string

useNewIntakeDomains?: boolean
bcaudan marked this conversation as resolved.
Show resolved Hide resolved
useCrossSiteSessionCookie?: boolean
useSecureSessionCookie?: boolean
trackSessionAcrossSubdomains?: boolean
Expand All @@ -79,6 +79,7 @@ export type Configuration = typeof DEFAULT_CONFIGURATION & {
service?: string

isEnabled: (feature: string) => boolean
isIntakeUrl: (url: string) => boolean

// only on staging build mode
replica?: ReplicaConfiguration
Expand All @@ -98,12 +99,19 @@ interface TransportConfiguration {
sdkVersion: string
applicationId?: string
proxyHost?: string
useNewIntakeDomains: boolean

service?: string
env?: string
version?: string
}

enum EndpointType {
LOGS = 'logs',
RUM = 'rum',
TRACE = 'trace',
}

export function buildConfiguration(userConfiguration: UserConfiguration, buildEnv: BuildEnv): Configuration {
const transportConfiguration: TransportConfiguration = {
applicationId: userConfiguration.applicationId,
Expand All @@ -114,28 +122,32 @@ export function buildConfiguration(userConfiguration: UserConfiguration, buildEn
sdkVersion: buildEnv.sdkVersion,
service: userConfiguration.service,
site: userConfiguration.site || INTAKE_SITE[userConfiguration.datacenter || buildEnv.datacenter],
useNewIntakeDomains: userConfiguration.useNewIntakeDomains || false,
version: userConfiguration.version,
}

const enableExperimentalFeatures = Array.isArray(userConfiguration.enableExperimentalFeatures)
? userConfiguration.enableExperimentalFeatures
: []

const intakeUrls = getIntakeUrls(transportConfiguration, userConfiguration.replica !== undefined)
const configuration: Configuration = {
cookieOptions: buildCookieOptions(userConfiguration),
isEnabled: (feature: string) => {
return includes(enableExperimentalFeatures, feature)
},
logsEndpoint: getEndpoint('browser', transportConfiguration),
logsEndpoint: getEndpoint(EndpointType.LOGS, transportConfiguration),
proxyHost: userConfiguration.proxyHost,
rumEndpoint: getEndpoint('rum', transportConfiguration),
rumEndpoint: getEndpoint(EndpointType.RUM, transportConfiguration),
service: userConfiguration.service,
traceEndpoint: getEndpoint('public-trace', transportConfiguration),
traceEndpoint: getEndpoint(EndpointType.TRACE, transportConfiguration),

isIntakeUrl: (url) => intakeUrls.some((intakeUrl) => url.indexOf(intakeUrl) === 0),
...DEFAULT_CONFIGURATION,
}
if (userConfiguration.internalMonitoringApiKey) {
configuration.internalMonitoringEndpoint = getEndpoint(
'browser',
EndpointType.LOGS,
transportConfiguration,
'browser-agent-internal-monitoring'
)
Expand Down Expand Up @@ -174,12 +186,12 @@ export function buildConfiguration(userConfiguration: UserConfiguration, buildEn
configuration.replica = {
applicationId: userConfiguration.replica.applicationId,
internalMonitoringEndpoint: getEndpoint(
'browser',
EndpointType.LOGS,
replicaTransportConfiguration,
'browser-agent-internal-monitoring'
),
logsEndpoint: getEndpoint('browser', replicaTransportConfiguration),
rumEndpoint: getEndpoint('rum', replicaTransportConfiguration),
logsEndpoint: getEndpoint(EndpointType.LOGS, replicaTransportConfiguration),
rumEndpoint: getEndpoint(EndpointType.RUM, replicaTransportConfiguration),
}
}
}
Expand All @@ -200,13 +212,13 @@ export function buildCookieOptions(userConfiguration: UserConfiguration) {
return cookieOptions
}

function getEndpoint(type: string, conf: TransportConfiguration, source?: string) {
function getEndpoint(type: EndpointType, conf: TransportConfiguration, source?: string) {
const tags =
`sdk_version:${conf.sdkVersion}` +
`${conf.env ? `,env:${conf.env}` : ''}` +
`${conf.service ? `,service:${conf.service}` : ''}` +
`${conf.version ? `,version:${conf.version}` : ''}`
const datadogHost = `${type}-http-intake.logs.${conf.site}`
const datadogHost = getHost(type, conf)
const host = conf.proxyHost ? conf.proxyHost : datadogHost
const proxyParameter = conf.proxyHost ? `ddhost=${datadogHost}&` : ''
const applicationIdParameter = conf.applicationId ? `_dd.application_id=${conf.applicationId}&` : ''
Expand All @@ -215,18 +227,42 @@ function getEndpoint(type: string, conf: TransportConfiguration, source?: string
return `https://${host}/v1/input/${conf.clientToken}?${parameters}`
}

export function isIntakeRequest(url: string, configuration: Configuration) {
return (
getPathName(url).indexOf('/v1/input/') !== -1 &&
(haveSameOrigin(url, configuration.logsEndpoint) ||
haveSameOrigin(url, configuration.rumEndpoint) ||
haveSameOrigin(url, configuration.traceEndpoint) ||
(!!configuration.internalMonitoringEndpoint && haveSameOrigin(url, configuration.internalMonitoringEndpoint)) ||
(!!configuration.replica &&
(haveSameOrigin(url, configuration.replica.logsEndpoint) ||
haveSameOrigin(url, configuration.replica.rumEndpoint) ||
haveSameOrigin(url, configuration.replica.internalMonitoringEndpoint))))
)
function getHost(type: EndpointType, conf: TransportConfiguration) {
if (conf.useNewIntakeDomains && NEW_INTAKE_DOMAIN_ALLOWED_SITES.indexOf(conf.site) !== -1) {
return `${type}.browser-intake-${conf.site}`
}
const oldTypes = {
[EndpointType.LOGS]: 'browser',
[EndpointType.RUM]: 'rum',
[EndpointType.TRACE]: 'public-trace',
}
bcaudan marked this conversation as resolved.
Show resolved Hide resolved
return `${oldTypes[type]}-http-intake.logs.${conf.site}`
}

function getIntakeUrls(conf: TransportConfiguration, withReplica: boolean) {
if (conf.proxyHost) {
return [`https://${conf.proxyHost}/v1/input/`]
}
const sites = [conf.site]
if (conf.buildMode === BuildMode.STAGING && withReplica) {
sites.push(INTAKE_SITE[Datacenter.US])
}
const urls = []
for (const site of sites) {
urls.push(
`https://rum-http-intake.logs.${site}/v1/input/`,
`https://browser-http-intake.logs.${site}/v1/input/`,
`https://public-trace-http-intake.logs.${site}/v1/input/`
)
if (conf.useNewIntakeDomains && NEW_INTAKE_DOMAIN_ALLOWED_SITES.indexOf(site) !== -1) {
bcaudan marked this conversation as resolved.
Show resolved Hide resolved
urls.push(
`https://rum.browser-intake-${site}/v1/input/`,
`https://logs.browser-intake-${site}/v1/input/`,
`https://trace.browser-intake-${site}/v1/input/`
)
}
}
return urls
}

function mustUseSecureCookie(userConfiguration: UserConfiguration) {
Expand Down
8 changes: 1 addition & 7 deletions packages/core/src/index.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,4 @@
export {
DEFAULT_CONFIGURATION,
Configuration,
UserConfiguration,
isIntakeRequest,
buildCookieOptions,
} from './domain/configuration'
export { DEFAULT_CONFIGURATION, Configuration, UserConfiguration, buildCookieOptions } from './domain/configuration'
export { startAutomaticErrorCollection, ErrorObservable } from './domain/automaticErrorCollection'
export { computeStackTrace } from './domain/tracekit'
export {
Expand Down
10 changes: 10 additions & 0 deletions packages/core/src/tools/specHelper.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,16 @@ export const SPEC_ENDPOINTS: Partial<Configuration> = {
logsEndpoint: 'https://logs-intake.com/v1/input/abcde?foo=bar',
rumEndpoint: 'https://rum-intake.com/v1/input/abcde?foo=bar',
traceEndpoint: 'https://trace-intake.com/v1/input/abcde?foo=bar',

isIntakeUrl: (url: string) => {
const intakeUrls = [
'https://monitoring-intake.com/v1/input/',
'https://logs-intake.com/v1/input/',
'https://rum-intake.com/v1/input/',
'https://trace-intake.com/v1/input/',
]
return intakeUrls.some((intakeUrl) => url.indexOf(intakeUrl) === 0)
},
}

export function isSafari() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@ import {
Configuration,
getPathName,
includes,
isIntakeRequest,
isValidUrl,
msToNs,
ResourceType,
Expand Down Expand Up @@ -204,5 +203,5 @@ export function computeSize(entry: RumPerformanceResourceTiming) {
}

export function isAllowedRequestUrl(configuration: Configuration, url: string) {
return url && !isIntakeRequest(url, configuration)
return url && !configuration.isIntakeUrl(url)
}