@nextsparkjs/theme-default 0.1.0-beta.21 → 0.1.0-beta.24
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/package.json +3 -3
- package/tests/cypress/support/e2e.ts +89 -0
- package/tests/cypress.config.ts +31 -16
- package/tests/jest/__mocks__/jose.js +22 -0
- package/tests/jest/__mocks__/next-server.js +56 -0
- package/tests/jest/jest.config.cjs +127 -0
- package/tests/jest/services/tasks.service.test.ts +707 -0
- package/tests/jest/setup.ts +170 -0
- package/tests/jest/jest.config.ts +0 -81
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 NextSpark
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nextsparkjs/theme-default",
|
|
3
|
-
"version": "0.1.0-beta.
|
|
3
|
+
"version": "0.1.0-beta.24",
|
|
4
4
|
"private": false,
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./config/theme.config.ts",
|
|
@@ -9,7 +9,7 @@
|
|
|
9
9
|
],
|
|
10
10
|
"dependencies": {},
|
|
11
11
|
"peerDependencies": {
|
|
12
|
-
"@nextsparkjs/core": "
|
|
12
|
+
"@nextsparkjs/core": "0.1.0-beta.24",
|
|
13
13
|
"@tanstack/react-query": "^5.0.0",
|
|
14
14
|
"lucide-react": "^0.539.0",
|
|
15
15
|
"next": "^15.0.0",
|
|
@@ -23,4 +23,4 @@
|
|
|
23
23
|
"type": "theme",
|
|
24
24
|
"name": "default"
|
|
25
25
|
}
|
|
26
|
-
}
|
|
26
|
+
}
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cypress E2E Support File for Theme
|
|
3
|
+
*
|
|
4
|
+
* This file is used when running tests in npm mode (outside monorepo).
|
|
5
|
+
* In monorepo mode, the core support file is used instead via cypress.config.ts.
|
|
6
|
+
*
|
|
7
|
+
* For documentation video commands, install cypress-slow-down:
|
|
8
|
+
* pnpm add -D cypress-slow-down
|
|
9
|
+
* Then uncomment the doc-commands import below.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import '@testing-library/cypress/add-commands'
|
|
13
|
+
|
|
14
|
+
// Import @cypress/grep for test filtering by tags
|
|
15
|
+
import registerCypressGrep from '@cypress/grep'
|
|
16
|
+
registerCypressGrep()
|
|
17
|
+
|
|
18
|
+
// Doc commands are optional (require cypress-slow-down)
|
|
19
|
+
// Uncomment if you have cypress-slow-down installed:
|
|
20
|
+
// import './doc-commands'
|
|
21
|
+
|
|
22
|
+
// Global error handling
|
|
23
|
+
Cypress.on('uncaught:exception', (err) => {
|
|
24
|
+
// Ignore React hydration errors
|
|
25
|
+
if (err.message.includes('Hydration')) {
|
|
26
|
+
return false
|
|
27
|
+
}
|
|
28
|
+
// Ignore ResizeObserver errors
|
|
29
|
+
if (err.message.includes('ResizeObserver')) {
|
|
30
|
+
return false
|
|
31
|
+
}
|
|
32
|
+
return true
|
|
33
|
+
})
|
|
34
|
+
|
|
35
|
+
// Global before hook
|
|
36
|
+
beforeEach(() => {
|
|
37
|
+
cy.clearCookies()
|
|
38
|
+
cy.clearLocalStorage()
|
|
39
|
+
})
|
|
40
|
+
|
|
41
|
+
// Type declarations for @cypress/grep and custom commands
|
|
42
|
+
declare global {
|
|
43
|
+
namespace Cypress {
|
|
44
|
+
interface Chainable {
|
|
45
|
+
/**
|
|
46
|
+
* Custom command to make API requests with better error handling
|
|
47
|
+
*/
|
|
48
|
+
apiRequest(options: Partial<Cypress.RequestOptions>): Chainable<Cypress.Response<any>>
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* Login command for authenticated tests
|
|
52
|
+
*/
|
|
53
|
+
login(email: string, password: string): Chainable<void>
|
|
54
|
+
}
|
|
55
|
+
interface SuiteConfigOverrides {
|
|
56
|
+
tags?: string | string[]
|
|
57
|
+
}
|
|
58
|
+
interface TestConfigOverrides {
|
|
59
|
+
tags?: string | string[]
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// Custom API request command
|
|
65
|
+
Cypress.Commands.add('apiRequest', (options) => {
|
|
66
|
+
const defaultOptions = {
|
|
67
|
+
failOnStatusCode: false,
|
|
68
|
+
timeout: 15000,
|
|
69
|
+
headers: {
|
|
70
|
+
'Content-Type': 'application/json',
|
|
71
|
+
...options.headers
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
return cy.request({
|
|
75
|
+
...defaultOptions,
|
|
76
|
+
...options
|
|
77
|
+
})
|
|
78
|
+
})
|
|
79
|
+
|
|
80
|
+
// Login command for authenticated tests
|
|
81
|
+
Cypress.Commands.add('login', (email: string, password: string) => {
|
|
82
|
+
cy.session([email, password], () => {
|
|
83
|
+
cy.visit('/login')
|
|
84
|
+
cy.get('[data-testid="email-input"], input[name="email"]').type(email)
|
|
85
|
+
cy.get('[data-testid="password-input"], input[name="password"]').type(password)
|
|
86
|
+
cy.get('[data-testid="submit-button"], button[type="submit"]').click()
|
|
87
|
+
cy.url().should('include', '/dashboard')
|
|
88
|
+
})
|
|
89
|
+
})
|
package/tests/cypress.config.ts
CHANGED
|
@@ -1,41 +1,55 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Cypress Configuration for Default Theme
|
|
3
3
|
*
|
|
4
|
-
* This config
|
|
4
|
+
* This config works in both monorepo and npm mode.
|
|
5
5
|
* Run with: NEXT_PUBLIC_ACTIVE_THEME=default pnpm cy:open
|
|
6
6
|
*/
|
|
7
7
|
|
|
8
8
|
import { defineConfig } from 'cypress'
|
|
9
9
|
import path from 'path'
|
|
10
10
|
import fs from 'fs'
|
|
11
|
+
import { fileURLToPath } from 'url'
|
|
12
|
+
|
|
13
|
+
// ESM-compatible __dirname
|
|
14
|
+
const __filename = fileURLToPath(import.meta.url)
|
|
15
|
+
const __dirname = path.dirname(__filename)
|
|
11
16
|
|
|
12
17
|
// Paths relative to this config file
|
|
13
18
|
const themeRoot = path.resolve(__dirname, '..')
|
|
14
19
|
const projectRoot = path.resolve(__dirname, '../../../..')
|
|
15
20
|
const narrationsOutputDir = path.join(__dirname, 'cypress/videos/narrations')
|
|
16
21
|
|
|
22
|
+
// Detect if running in npm mode (no packages/core folder) vs monorepo
|
|
23
|
+
const isNpmMode = !fs.existsSync(path.join(projectRoot, 'packages/core'))
|
|
24
|
+
|
|
17
25
|
// Load environment variables
|
|
18
26
|
import dotenv from 'dotenv'
|
|
19
27
|
dotenv.config({ path: path.join(projectRoot, '.env') })
|
|
20
28
|
|
|
21
|
-
// Server port (from .env or default
|
|
22
|
-
const port = process.env.PORT ||
|
|
29
|
+
// Server port (from .env or default 3000)
|
|
30
|
+
const port = process.env.PORT || 3000
|
|
23
31
|
|
|
24
32
|
export default defineConfig({
|
|
25
33
|
e2e: {
|
|
26
34
|
// Base URL for the application
|
|
27
35
|
baseUrl: `http://localhost:${port}`,
|
|
28
36
|
|
|
29
|
-
// Spec patterns: core tests
|
|
30
|
-
specPattern:
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
37
|
+
// Spec patterns: theme tests (core tests only in monorepo)
|
|
38
|
+
specPattern: isNpmMode
|
|
39
|
+
? [
|
|
40
|
+
// npm mode: only theme tests
|
|
41
|
+
path.join(__dirname, 'cypress/e2e/**/*.cy.{js,ts}'),
|
|
42
|
+
]
|
|
43
|
+
: [
|
|
44
|
+
// Monorepo: core tests + theme tests
|
|
45
|
+
path.join(projectRoot, 'packages/core/tests/cypress/e2e/core/**/*.cy.{js,ts}'),
|
|
46
|
+
path.join(__dirname, 'cypress/e2e/**/*.cy.{js,ts}'),
|
|
47
|
+
],
|
|
48
|
+
|
|
49
|
+
// Support file (theme-local in npm mode, core in monorepo)
|
|
50
|
+
supportFile: isNpmMode
|
|
51
|
+
? path.join(__dirname, 'cypress/support/e2e.ts')
|
|
52
|
+
: path.join(projectRoot, 'packages/core/tests/cypress/support/e2e.ts'),
|
|
39
53
|
|
|
40
54
|
// Fixtures folder (theme-specific)
|
|
41
55
|
fixturesFolder: path.join(__dirname, 'cypress/fixtures'),
|
|
@@ -96,15 +110,16 @@ export default defineConfig({
|
|
|
96
110
|
grepOmitFiltered: true,
|
|
97
111
|
},
|
|
98
112
|
|
|
99
|
-
setupNodeEvents(on, config) {
|
|
113
|
+
async setupNodeEvents(on, config) {
|
|
100
114
|
// Allure plugin setup (allure-cypress)
|
|
101
|
-
const { allureCypress } =
|
|
115
|
+
const { allureCypress } = await import('allure-cypress/reporter')
|
|
102
116
|
allureCypress(on, config, {
|
|
103
117
|
resultsDir: path.join(__dirname, 'cypress/allure-results'),
|
|
104
118
|
})
|
|
105
119
|
|
|
106
120
|
// @cypress/grep plugin for test filtering by tags
|
|
107
|
-
|
|
121
|
+
const grepPlugin = await import('@cypress/grep/src/plugin.js')
|
|
122
|
+
;(grepPlugin.default || grepPlugin)(config)
|
|
108
123
|
|
|
109
124
|
// Documentation video tasks
|
|
110
125
|
on('task', {
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Mock for jose package
|
|
3
|
+
* Resolves ES module import issues in Jest tests
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
module.exports = {
|
|
7
|
+
jwtVerify: jest.fn(),
|
|
8
|
+
SignJWT: jest.fn().mockImplementation(() => ({
|
|
9
|
+
setProtectedHeader: jest.fn().mockReturnThis(),
|
|
10
|
+
setIssuedAt: jest.fn().mockReturnThis(),
|
|
11
|
+
setExpirationTime: jest.fn().mockReturnThis(),
|
|
12
|
+
sign: jest.fn().mockResolvedValue('mock-jwt-token')
|
|
13
|
+
})),
|
|
14
|
+
importJWK: jest.fn(),
|
|
15
|
+
generateSecret: jest.fn(),
|
|
16
|
+
createRemoteJWKSet: jest.fn(),
|
|
17
|
+
errors: {
|
|
18
|
+
JWTExpired: class JWTExpired extends Error {},
|
|
19
|
+
JWTInvalid: class JWTInvalid extends Error {},
|
|
20
|
+
JWTClaimValidationFailed: class JWTClaimValidationFailed extends Error {}
|
|
21
|
+
}
|
|
22
|
+
}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Mock for next/server
|
|
3
|
+
* Provides NextRequest and NextResponse mocks for API testing
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
class MockNextRequest {
|
|
7
|
+
constructor(url, options = {}) {
|
|
8
|
+
this.url = url
|
|
9
|
+
this.method = options.method || 'GET'
|
|
10
|
+
this.headers = new Map(Object.entries(options.headers || {}))
|
|
11
|
+
this._body = options.body
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
async json() {
|
|
15
|
+
if (!this._body) return {}
|
|
16
|
+
try {
|
|
17
|
+
return typeof this._body === 'string' ? JSON.parse(this._body) : this._body
|
|
18
|
+
} catch {
|
|
19
|
+
throw new SyntaxError('Invalid JSON')
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async text() {
|
|
24
|
+
return this._body || ''
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
class MockNextResponse {
|
|
29
|
+
constructor(body, options = {}) {
|
|
30
|
+
this.body = body
|
|
31
|
+
this.status = options.status || 200
|
|
32
|
+
this.statusText = options.statusText || 'OK'
|
|
33
|
+
this.headers = new Map(Object.entries(options.headers || {}))
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
async json() {
|
|
37
|
+
if (!this.body) return {}
|
|
38
|
+
try {
|
|
39
|
+
return typeof this.body === 'string' ? JSON.parse(this.body) : this.body
|
|
40
|
+
} catch {
|
|
41
|
+
throw new SyntaxError('Invalid JSON')
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
static json(data, options = {}) {
|
|
46
|
+
return new MockNextResponse(data, {
|
|
47
|
+
...options,
|
|
48
|
+
headers: { 'Content-Type': 'application/json', ...options.headers }
|
|
49
|
+
})
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
module.exports = {
|
|
54
|
+
NextRequest: MockNextRequest,
|
|
55
|
+
NextResponse: MockNextResponse
|
|
56
|
+
}
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Jest Configuration for Default Theme
|
|
3
|
+
*
|
|
4
|
+
* This config works in both monorepo and npm mode.
|
|
5
|
+
* Run with: pnpm test:theme
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const path = require('path')
|
|
9
|
+
const fs = require('fs')
|
|
10
|
+
|
|
11
|
+
// Paths relative to this config file
|
|
12
|
+
const themeTestsRoot = __dirname
|
|
13
|
+
const themeRoot = path.resolve(__dirname, '../..')
|
|
14
|
+
|
|
15
|
+
// In monorepo: themes/default/tests/jest -> apps/dev (via symlink in contents/)
|
|
16
|
+
// In npm mode: contents/themes/default/tests/jest -> project root (5 levels up)
|
|
17
|
+
const monorepoRepoRoot = path.resolve(__dirname, '../../../..')
|
|
18
|
+
const npmModeRoot = path.resolve(__dirname, '../../../../..')
|
|
19
|
+
|
|
20
|
+
// Detect if running in npm mode (no packages/core folder) vs monorepo
|
|
21
|
+
const isNpmMode = !fs.existsSync(path.join(monorepoRepoRoot, 'packages/core'))
|
|
22
|
+
|
|
23
|
+
// In monorepo, use apps/dev as rootDir since contents/ symlinks to themes/
|
|
24
|
+
const monorepoAppRoot = path.join(monorepoRepoRoot, 'apps/dev')
|
|
25
|
+
const projectRoot = isNpmMode ? npmModeRoot : monorepoAppRoot
|
|
26
|
+
|
|
27
|
+
// Module name mapper based on mode
|
|
28
|
+
const moduleNameMapper = isNpmMode
|
|
29
|
+
? {
|
|
30
|
+
// NPM mode: resolve from node_modules
|
|
31
|
+
'^@nextsparkjs/core/(.*)$': '@nextsparkjs/core/$1',
|
|
32
|
+
'^@nextsparkjs/core$': '@nextsparkjs/core',
|
|
33
|
+
'^@/contents/(.*)$': '<rootDir>/contents/$1',
|
|
34
|
+
'^@/entities/(.*)$': '<rootDir>/contents/entities/$1',
|
|
35
|
+
'^@/plugins/(.*)$': '<rootDir>/contents/plugins/$1',
|
|
36
|
+
'^@/themes/(.*)$': '<rootDir>/contents/themes/$1',
|
|
37
|
+
'^@/(.*)$': '<rootDir>/$1',
|
|
38
|
+
// Mocks from theme-local folder
|
|
39
|
+
'next/server': path.join(themeTestsRoot, '__mocks__/next-server.js'),
|
|
40
|
+
'^jose$': path.join(themeTestsRoot, '__mocks__/jose.js'),
|
|
41
|
+
'^jose/(.*)$': path.join(themeTestsRoot, '__mocks__/jose.js'),
|
|
42
|
+
}
|
|
43
|
+
: {
|
|
44
|
+
// Monorepo mode: resolve from packages/core/src (rootDir is apps/dev)
|
|
45
|
+
'^@nextsparkjs/core/(.*)$': '<rootDir>/../../packages/core/src/$1',
|
|
46
|
+
'^@nextsparkjs/core$': '<rootDir>/../../packages/core/src',
|
|
47
|
+
'^@/contents/(.*)$': '<rootDir>/contents/$1',
|
|
48
|
+
'^@/entities/(.*)$': '<rootDir>/contents/entities/$1',
|
|
49
|
+
'^@/plugins/(.*)$': '<rootDir>/contents/plugins/$1',
|
|
50
|
+
'^@/themes/(.*)$': '<rootDir>/contents/themes/$1',
|
|
51
|
+
'^@/(.*)$': '<rootDir>/$1',
|
|
52
|
+
// Mocks from core
|
|
53
|
+
'next/server': '<rootDir>/../../packages/core/tests/jest/__mocks__/next-server.js',
|
|
54
|
+
'^jose$': '<rootDir>/../../packages/core/tests/jest/__mocks__/jose.js',
|
|
55
|
+
'^jose/(.*)$': '<rootDir>/../../packages/core/tests/jest/__mocks__/jose.js',
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// Setup files based on mode
|
|
59
|
+
const setupFilesAfterEnv = isNpmMode
|
|
60
|
+
? [
|
|
61
|
+
// NPM mode: use theme's local setup only (it includes everything needed)
|
|
62
|
+
path.join(themeTestsRoot, 'setup.ts'),
|
|
63
|
+
]
|
|
64
|
+
: [
|
|
65
|
+
// Monorepo mode: use local core setup (rootDir is apps/dev)
|
|
66
|
+
'<rootDir>/../../packages/core/tests/jest/setup.ts',
|
|
67
|
+
]
|
|
68
|
+
|
|
69
|
+
/** @type {import('jest').Config} */
|
|
70
|
+
module.exports = {
|
|
71
|
+
displayName: 'theme-default',
|
|
72
|
+
rootDir: projectRoot,
|
|
73
|
+
|
|
74
|
+
// Use roots to explicitly set test location (handles symlinks better)
|
|
75
|
+
roots: [themeTestsRoot],
|
|
76
|
+
|
|
77
|
+
// Test file patterns
|
|
78
|
+
testMatch: [
|
|
79
|
+
'**/*.{test,spec}.{js,ts,tsx}',
|
|
80
|
+
],
|
|
81
|
+
testPathIgnorePatterns: [
|
|
82
|
+
'<rootDir>/node_modules/',
|
|
83
|
+
'<rootDir>/.next/',
|
|
84
|
+
],
|
|
85
|
+
|
|
86
|
+
// Preset and environment
|
|
87
|
+
preset: 'ts-jest',
|
|
88
|
+
testEnvironment: 'jsdom',
|
|
89
|
+
|
|
90
|
+
// Module resolution
|
|
91
|
+
moduleNameMapper,
|
|
92
|
+
|
|
93
|
+
// Setup files
|
|
94
|
+
setupFilesAfterEnv,
|
|
95
|
+
|
|
96
|
+
// Transform configuration
|
|
97
|
+
transform: {
|
|
98
|
+
'^.+\\.(ts|tsx)$': ['ts-jest', {
|
|
99
|
+
tsconfig: path.join(projectRoot, 'tsconfig.json'),
|
|
100
|
+
}],
|
|
101
|
+
},
|
|
102
|
+
|
|
103
|
+
// Transform ignore patterns - allow TypeScript from core's jest-setup
|
|
104
|
+
transformIgnorePatterns: [
|
|
105
|
+
'node_modules/(?!(uncrypto|better-auth|@noble|.*jose.*|remark.*|unified.*|@nextsparkjs/core/tests|.*\\.mjs$))',
|
|
106
|
+
'node_modules/\\.pnpm/(?!(.*uncrypto.*|.*better-auth.*|.*@noble.*|.*jose.*|.*remark.*|.*unified.*|@nextsparkjs.*core.*tests|.*\\.mjs$))',
|
|
107
|
+
],
|
|
108
|
+
|
|
109
|
+
// File extensions
|
|
110
|
+
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json'],
|
|
111
|
+
|
|
112
|
+
// Test timeout
|
|
113
|
+
testTimeout: 10000,
|
|
114
|
+
|
|
115
|
+
// Verbose output
|
|
116
|
+
verbose: true,
|
|
117
|
+
|
|
118
|
+
// Force exit after tests complete
|
|
119
|
+
forceExit: true,
|
|
120
|
+
|
|
121
|
+
// Disable watchman for symlink support
|
|
122
|
+
watchman: false,
|
|
123
|
+
|
|
124
|
+
// Coverage output directory
|
|
125
|
+
coverageDirectory: path.join(themeTestsRoot, 'coverage'),
|
|
126
|
+
coverageReporters: ['text', 'lcov', 'html'],
|
|
127
|
+
}
|