diff --git a/.github/workflows/check-links.yml b/.github/workflows/check-links.yml index 2465569c..307b88e1 100644 --- a/.github/workflows/check-links.yml +++ b/.github/workflows/check-links.yml @@ -9,7 +9,7 @@ jobs: linkChecker: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3.6.0 + - uses: actions/checkout@v4.2.1 - name: Link Checker uses: lycheeverse/lychee-action@v1.8.0 diff --git a/.github/workflows/ci-main.yml b/.github/workflows/ci-main.yml index bdc43daf..18328461 100644 --- a/.github/workflows/ci-main.yml +++ b/.github/workflows/ci-main.yml @@ -11,7 +11,7 @@ jobs: permissions: read-all steps: - name: Checkout - uses: actions/checkout@v3.6.0 + uses: actions/checkout@v4.2.1 - uses: actions/setup-node@v4.0.3 with: node-version: '18' @@ -36,7 +36,7 @@ jobs: browser: [headlessChrome, headlessFirefox] steps: - name: Checkout - uses: actions/checkout@v3.6.0 + uses: actions/checkout@v4.2.1 - uses: actions/setup-node@v4.0.3 with: node-version: '18' diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index f7253f49..08d8e038 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -9,7 +9,7 @@ jobs: permissions: read-all steps: - name: Checkout - uses: actions/checkout@v3.6.0 + uses: actions/checkout@v4.2.1 - uses: actions/setup-node@v4.0.3 with: node-version: '18' @@ -47,7 +47,7 @@ jobs: accessKey: ${{ secrets.SAUCE_ACCESS_KEY }} tunnelName: ${{ secrets.SAUCE_TUNNEL_ID }} - name: Checkout - uses: actions/checkout@v3.6.0 + uses: actions/checkout@v4.2.1 - uses: actions/setup-node@v4.0.3 with: node-version: '18' @@ -69,7 +69,7 @@ jobs: browser: [headlessChrome, headlessFirefox] steps: - name: Checkout - uses: actions/checkout@v3.6.0 + uses: actions/checkout@v4.2.1 - uses: actions/setup-node@v4.0.3 with: node-version: '18' diff --git a/.gitignore b/.gitignore index 5c60c4c3..a1006cfb 100644 --- a/.gitignore +++ b/.gitignore @@ -17,4 +17,3 @@ tests_output/ # contains secrets .env -.npmrc diff --git a/.npmrc b/.npmrc new file mode 100644 index 00000000..7898332e --- /dev/null +++ b/.npmrc @@ -0,0 +1 @@ +git-tag-version = false \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index c13c56a0..419644ce 100644 --- a/package-lock.json +++ b/package-lock.json @@ -6951,9 +6951,9 @@ } }, "node_modules/chromedriver": { - "version": "128.0.3", - "resolved": "https://registry.npmjs.org/chromedriver/-/chromedriver-128.0.3.tgz", - "integrity": "sha512-Xn/bknOpGlY9tKinwS/hVWeNblSeZvbbJbF8XZ73X1jeWfAFPRXx3fMLdNNz8DqruDbx3cKEJ5wR3mnst6G3iw==", + "version": "129.0.4", + "resolved": "https://registry.npmjs.org/chromedriver/-/chromedriver-129.0.4.tgz", + "integrity": "sha512-j5I55cQwodFJUaYa1tWUmj2ss9KcPRBWmUa5Qonq3X8kqv2ASPyTboFYb4YB/YLztkYTUUw2E43txXw0wYzT/A==", "dev": true, "hasInstallScript": true, "dependencies": { @@ -17602,7 +17602,7 @@ "body-parser": "^1.20.2", "chai": "^4.3.7", "chrome-launcher": "^1.0.0", - "chromedriver": "^128.0.3", + "chromedriver": "^129.0.4", "codecov": "^3.8.2", "compression": "^1.7.4", "cors": "^2.8.5", diff --git a/packages/web/karma.conf.js b/packages/web/karma.conf.js index 1318cade..dab2572a 100644 --- a/packages/web/karma.conf.js +++ b/packages/web/karma.conf.js @@ -106,6 +106,13 @@ module.exports = async function (config) { // start these browsers browsers: [], + customLaunchers: { + ChromeHeadlessCI: { + base: 'ChromeHeadless', + flags: ['--no-sandbox'] + } + }, + // Continuous Integration mode // if true, Karma captures browsers, runs the tests and exits singleRun: false, diff --git a/packages/web/package.json b/packages/web/package.json index 63d315b8..ee8cb8cf 100644 --- a/packages/web/package.json +++ b/packages/web/package.json @@ -10,7 +10,7 @@ "compile:browser": "rollup -c", "compile:tsc": "tsc --build tsconfig.cjs.json tsconfig.esm.json", "test:unit:watch": "karma start", - "test:unit:ci": "nyc karma start --single-run --no-auto-watch --browsers ChromeHeadless", + "test:unit:ci": "nyc karma start --single-run --no-auto-watch --browsers ChromeHeadlessCI", "test:unit:ci-node": "env TS_NODE_TRANSPILE_ONLY=true TS_NODE_PROJECT=tsconfig.cjs.json nyc mocha", "test:integration:local": "npm-run-all -s compile test:integration:local:headlessChrome:_execute", "test:integration:local:firefox": "npm-run-all -s compile test:integration:local:firefox:_execute", @@ -80,7 +80,7 @@ "body-parser": "^1.20.2", "chai": "^4.3.7", "chrome-launcher": "^1.0.0", - "chromedriver": "^128.0.3", + "chromedriver": "^129.0.4", "codecov": "^3.8.2", "compression": "^1.7.4", "cors": "^2.8.5", diff --git a/packages/web/src/SplunkErrorInstrumentation.ts b/packages/web/src/SplunkErrorInstrumentation.ts index fc51a84a..d71d0616 100644 --- a/packages/web/src/SplunkErrorInstrumentation.ts +++ b/packages/web/src/SplunkErrorInstrumentation.ts @@ -40,7 +40,7 @@ function stringifyValue(value: unknown) { function parseErrorStack(stack: string): string { //get list of files in stack , find corresponding sourcemap id and add it to the source map id object const sourceMapIds = {}; - const urlPattern = /(https?:\/\/[^\s]+\/[^\s:]+|\/[^\s:]+)/g; + const urlPattern = /([\w]+:\/\/[^\s/]+\/[^\s?:#]+)/g; const urls = stack.match(urlPattern); if (urls) { urls.forEach(url => { diff --git a/packages/web/test/index.ts b/packages/web/test/index.ts index 9da7723e..d90bafe4 100644 --- a/packages/web/test/index.ts +++ b/packages/web/test/index.ts @@ -31,3 +31,4 @@ import './SplunkSpanAttributesProcessor.test'; import './SplunkOtelWeb.test'; import './synthetics.test'; import './socketio.test'; +import './stacktrace.test'; diff --git a/packages/web/test/stacktrace.test.ts b/packages/web/test/stacktrace.test.ts new file mode 100644 index 00000000..7dc6f9f1 --- /dev/null +++ b/packages/web/test/stacktrace.test.ts @@ -0,0 +1,223 @@ +/* +Copyright 2024 Splunk Inc. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + +http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +import * as assert from 'assert'; +import { beforeEach } from 'mocha'; +import { generateFilePaths, generateRandomStackTrace } from './utils'; + +const chromeStackTraceEval = `Error: Something went wrong + at eval (eval at (http://example.com/scripts/main.js:10:20), :1:1) + at Object.functionName (http://example.com/scripts/utils.js:15:25) + at http://example.com/scripts/app.js:20:30 + at new ConstructorName (http://example.com/scripts/controller.js:25:35) + at http://example.com/scripts/main.js:30:40`; +const chromeStackTraceEvalExpected = [ + 'http://example.com/scripts/main.js', + 'http://example.com/scripts/utils.js', + 'http://example.com/scripts/app.js', + 'http://example.com/scripts/controller.js', +]; + +const chromeStackTraceAnonymous = `TypeError: undefined is not a function + at http://example.com/js/anonymous.js:10:5 + at :15:10 + at Object.functionName (http://example.com/js/utils.js:20:15) + at new ConstructorName (http://example.com/js/app.js:25:20) + at :30:25`; +const chromeStackTraceAnonymousExpected = [ + 'http://example.com/js/anonymous.js', + 'http://example.com/js/utils.js', + 'http://example.com/js/app.js', +]; + +const geckoStackTraceEval = `Error: Something went wrong + @http://example.com/scripts/main.js:10:20 + @eval (eval at :1:1) + functionName@http://example.com/scripts/utils.js:15:25 + @http://example.com/scripts/app.js:20:30 + ConstructorName@http://example.com/scripts/controller.js:25:35 + @http://example.com/scripts/main.js:30:40`; +const geckoStackTraceEvalExpected = [ + 'http://example.com/scripts/main.js', + 'http://example.com/scripts/utils.js', + 'http://example.com/scripts/app.js', + 'http://example.com/scripts/controller.js', +]; + +const geckoStackTraceAnonymous = `TypeError: undefined is not a function + @http://example.com/js/anonymous.js:10:5 + @:15:10 + functionName@http://example.com/js/utils.js:20:15 + ConstructorName@http://example.com/js/app.js:25:20 + @:30:25`; +const geckoStackTraceAnonymousExpected = [ + 'http://example.com/js/anonymous.js', + 'http://example.com/js/utils.js', + 'http://example.com/js/app.js' +]; + +// Test 1: simple test w/ dupes +const stack1 = `Error + at http://localhost:8080/js/script1.js:10:15 + at http://localhost:8080/js/script2.js:20:25 + at http://localhost:8080/js/script1.js:30:35`; +const expected1 = ['http://localhost:8080/js/script1.js', 'http://localhost:8080/js/script2.js']; + +// Test 2: http and https +const stack2 = `Error + at https://example.com/js/app.js:50:10 + at http://localhost/js/util.js:100:50`; +const expected2 = ['https://example.com/js/app.js', 'http://localhost/js/util.js']; + +// Test 3: No full path URLs +const stack3 = `Error + at someFunction (file.js:10:15) + at anotherFunction (file.js:20:25)`; +const expected3 = []; + +// Test 4: Only one URL, with port +const stack4 = `Error + at http://localhost:3000/js/main.js:10:15`; +const expected4 = ['http://localhost:3000/js/main.js']; + +// Test 5: Duplicate URLs +const stack5 = `Error + at http://localhost:3000/js/main.js:10:15 + at http://localhost:3000/js/main.js:20:25 + at http://localhost:3000/js/utils.js:30:35`; +const expected5 = ['http://localhost:3000/js/main.js', 'http://localhost:3000/js/utils.js']; + +// Test 6: Urls with query strings and fragments +const stack6 = `Error + at http://example.com:8080/path/js/main.js?name=testname:10:15 + at http://example.com:8080/path/js/main2.js#fragmentHere:20:15 + at http://example.com:8080/path/js/main3.js?name=testname#fragmentHere:30:15`; +const expected6 = ['http://example.com:8080/path/js/main.js', 'http://example.com:8080/path/js/main2.js', 'http://example.com:8080/path/js/main3.js']; + +// Test 7: Urls with different protocols and blobs +const stack7 = `Error + at file://testing.com:8000/js/testFile.js:1:2 + at blob:https://example.com:1000/src/hello.js:2:3`; +const expected7 = ['file://testing.com:8000/js/testFile.js', 'https://example.com:1000/src/hello.js']; + +const regexFilter = /([\w]+:\/\/[^\s/]+\/[^\s?:#]+)/g; +describe('regexFilter', () => { + let urls = new Set(); + let match; + + beforeEach(() => { + urls = new Set(); + match = null; + }); + it('should test chrome eval stack traces', () => { + while ((match = regexFilter.exec(chromeStackTraceEval)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, chromeStackTraceEvalExpected); + }); + + it ('should test chrome anonymous stack traces', () => { + while ((match = regexFilter.exec(chromeStackTraceAnonymous)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, chromeStackTraceAnonymousExpected); + }); + + it ('should test gecko eval stack traces', () => { + while ((match = regexFilter.exec(geckoStackTraceEval)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, geckoStackTraceEvalExpected); + }); + + it ('should test gecko anonymous stack traces', () => { + while ((match = regexFilter.exec(geckoStackTraceAnonymous)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, geckoStackTraceAnonymousExpected); + }); + + it ('should test simple stack trace with dupes', () => { + while ((match = regexFilter.exec(stack1)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected1); + }); + + it ('should test http vs https stack traces', () => { + while ((match = regexFilter.exec(stack2)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected2); + }); + + it ('should test no full url path stack traces', () => { + while ((match = regexFilter.exec(stack3)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected3); + }); + + it ('should test url ports in stack traces', () => { + while ((match = regexFilter.exec(stack4)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected4); + }); + + it ('should test duplicate urls in stack traces', () => { + while ((match = regexFilter.exec(stack5)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected5); + }); + + it ('should test query strings/fragments in stack traces', () => { + while ((match = regexFilter.exec(stack6)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected6); + }); + + it ('should test blobs and diff protocols in stack traces', () => { + while ((match = regexFilter.exec(stack7)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr, expected7); + }); + + it ('should test long stack traces', () => { + const randomPaths = generateFilePaths(20, 20); + const randomStack = generateRandomStackTrace(randomPaths, 10000); + + while ((match = regexFilter.exec(randomStack)) !== null) { + urls.add(match[0]); + } + const urlArr = [...urls]; + assert.deepEqual(urlArr.sort(), randomPaths.sort()); + }); +}); \ No newline at end of file diff --git a/packages/web/test/utils.ts b/packages/web/test/utils.ts index 578bf0d8..540a3c19 100644 --- a/packages/web/test/utils.ts +++ b/packages/web/test/utils.ts @@ -99,3 +99,22 @@ export function initWithSyncPipeline(additionalOptions = {}): { export function deinit(force?: boolean): void { SplunkRum.deinit(force); } + +export function generateFilePaths(domainCount: number, pathCount: number): string[] { + const paths: string[] = []; + for (let i = 0; i < domainCount; i++) { + const domain = `http://domain${i}.com`; + for (let j = 0; j < pathCount; j++) { + paths.push(`${domain}/path${j}.js`); + } + } + return paths; +} + +export function generateRandomStackTrace(paths: string[], stackCount: number): string { + let stack = 'Error\n'; + for (let i = 0; i < stackCount; i++) { + stack += `at ${paths[Math.floor(Math.random() * paths.length)]}:${Math.floor(Math.random() * 1000)}:${Math.floor(Math.random() * 1000)}\n`; + } + return stack; +}