Skip to content

Commit 9357bdd

Browse files
committed
Merge branch 'master' into in-kontekst-varsel
2 parents 04a20de + 2dee97b commit 9357bdd

31 files changed

+217
-219
lines changed

.gitignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -48,8 +48,9 @@ yarn-error.log*
4848

4949
/server/.dist/
5050
/server/tsconfig.tsbuildinfo
51-
/server/__next-test-dummy/.next/
51+
/server/__test-utils/next-dummy/.next/
5252

5353
# VS Code
5454
.vscode
5555
/.env.prod-local.local
56+
/server/.next/

.nais/config.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ spec:
3030
leaderElection: true
3131
redis:
3232
- instance: {{ redis.instance }}
33-
access: readwrite
33+
access: admin
3434
ingresses:
3535
{{#each ingresses as |url|}}
3636
- {{url}}

.nais/redis.yml

+14
Original file line numberDiff line numberDiff line change
@@ -9,3 +9,17 @@ metadata:
99
spec:
1010
plan: {{ redis.plan }}
1111
project: {{ redis.project }}
12+
13+
---
14+
apiVersion: aiven.io/v1alpha1
15+
kind: ServiceIntegration
16+
metadata:
17+
labels:
18+
team: personbruker
19+
name: redis-personbruker-{{ redis.instance }}
20+
namespace: personbruker
21+
spec:
22+
project: {{ redis.project }}
23+
integrationType: prometheus
24+
destinationEndpointId: {{ redis.endpointId }}
25+
sourceServiceName: redis-personbruker-{{ redis.instance }}

.nais/vars/vars-dev1.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,4 +21,5 @@ replicas:
2121
redis:
2222
plan: hobbyist
2323
project: nav-dev
24-
instance: pagecache-dev1
24+
instance: pagecache
25+
endpointId: f20f5b48-18f4-4e2a-8e5f-4ab3edb19733

.nais/vars/vars-dev2.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,4 +21,5 @@ replicas:
2121
redis:
2222
plan: hobbyist
2323
project: nav-dev
24-
instance: pagecache-dev2
24+
instance: pagecache
25+
endpointId: f20f5b48-18f4-4e2a-8e5f-4ab3edb19733

.nais/vars/vars-prod.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -19,4 +19,5 @@ replicas:
1919
redis:
2020
plan: startup-4
2121
project: nav-prod
22-
instance: pagecache
22+
instance: pagecache
23+
endpointId: 76685598-1048-4f56-b34a-9769ef747a92

next.config.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,7 @@ const config = {
126126
__dirname,
127127
'server',
128128
'.dist',
129-
'custom-cache-handler.cjs'
129+
'page-cache-handler.cjs'
130130
),
131131
cacheMaxMemorySize: 0,
132132
}),

nodeenv.d.ts

+4
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,10 @@ declare global {
1818
NAVNO_API_URL: string;
1919
ASSET_PREFIX: string;
2020
TELEMETRY_URL: string;
21+
NEXT_PHASE: string;
22+
REDIS_URI_PAGECACHE: string;
23+
REDIS_USERNAME_PAGECACHE: string;
24+
REDIS_PASSWORD_PAGECACHE: string;
2125
}
2226
}
2327

package-lock.json

+6-4
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+4-5
Original file line numberDiff line numberDiff line change
@@ -17,10 +17,11 @@
1717
"start-local-failover": "cp .env.prod-local .env && IS_FAILOVER_INSTANCE=true PORT=3003 npm start",
1818
"start-clean-failover": "rm -fr .next-static && npm run build-local-failover && npm run start-local-failover",
1919
"analyze-bundle": "ANALYZE_BUNDLE=true npm run build-local",
20-
"lint": "npm-run-all lint:*",
21-
"lint:next": "eslint && tsc",
20+
"lint": "npm-run-all --parallel lint:*",
21+
"lint:eslint": "eslint",
22+
"lint:tsc": "tsc",
2223
"lint:server": "npm run lint --workspace server",
23-
"test": "npm-run-all test:*",
24+
"test": "npm-run-all --parallel test:*",
2425
"test:server": "npm run test --workspace server",
2526
"test:client": "TZ=Europe/Oslo jest --testPathIgnorePatterns=server/src",
2627
"prepare": "husky install"
@@ -43,7 +44,6 @@
4344
"@reduxjs/toolkit": "1.9.7",
4445
"csp-header": "5.2.1",
4546
"dayjs": "1.11.10",
46-
"dotenv": "16.3.1",
4747
"fuse.js": "6.6.2",
4848
"html-react-parser": "5.0.7",
4949
"js-cookie": "3.0.5",
@@ -61,7 +61,6 @@
6161
"react-dom": "18.2.0",
6262
"react-movable": "3.0.4",
6363
"react-redux": "8.1.3",
64-
"redis": "4.6.13",
6564
"rss": "1.2.2",
6665
"sharp": "0.33.2",
6766
"swr": "2.2.4",

server/__test-utils/utils.ts

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
import next from 'next';
2+
import path from 'path';
3+
4+
export const __getNextTestApp = () =>
5+
next({
6+
conf: {},
7+
dir: path.join(__dirname, 'next-dummy'),
8+
});

server/build.mjs

+4-4
Original file line numberDiff line numberDiff line change
@@ -11,19 +11,19 @@ const commonOptions = {
1111
// The cache handler needs to have its own bundle, at it is imported both from our custom server and the next.js server
1212
const cacheHandlerOptions = {
1313
...commonOptions,
14-
entryPoints: ['src/cache/custom-cache-handler.ts'],
15-
outfile: '.dist/custom-cache-handler.cjs',
14+
entryPoints: ['src/cache/page-cache-handler.ts'],
15+
outfile: '.dist/page-cache-handler.cjs',
1616
};
1717

1818
const serverOptions = {
1919
...commonOptions,
2020
entryPoints: ['src/server.ts'],
2121
outfile: '.dist/server.cjs',
2222
// Externalize the cache-handler to ensure our server and next.js both import the same file
23-
external: ['./custom-cache-handler.cjs'],
23+
external: ['./page-cache-handler.cjs'],
2424
// Rewrite the import path for the cache handler
2525
alias: {
26-
'cache/custom-cache-handler': './custom-cache-handler.cjs'
26+
'cache/page-cache-handler': './page-cache-handler.cjs'
2727
},
2828
}
2929

server/package.json

+7-3
Original file line numberDiff line numberDiff line change
@@ -8,18 +8,22 @@
88
"start": "node -r dotenv/config .dist/server.cjs dotenv_config_path=../.env",
99
"nodemon-start": "nodemon -r dotenv/config -w .dist .dist/server.cjs dotenv_config_path=../.env",
1010
"dev": "npm run build && concurrently \"npm run build-watch\" \"npm run nodemon-start\"",
11-
"test": "next build __next-test-dummy && NODE_OPTIONS=--experimental-vm-modules jest --config jest.config.mjs",
12-
"lint": "eslint && tsc"
11+
"test": "next build __test-utils/next-dummy && NODE_OPTIONS=--experimental-vm-modules jest --config jest.config.mjs",
12+
"lint": "npm-run-all --parallel lint:*",
13+
"lint:eslint": "eslint",
14+
"lint:tsc": "tsc"
1315
},
1416
"nodemonConfig": {
1517
"delay": 500
1618
},
1719
"dependencies": {
1820
"cookie-parser": "1.4.6",
21+
"dotenv": "16.3.1",
1922
"express-prom-bundle": "7.0.0",
2023
"http-terminator": "3.2.0",
2124
"on-headers": "1.0.2",
22-
"prom-client": "15.1.0"
25+
"prom-client": "15.1.0",
26+
"redis": "4.6.13"
2327
},
2428
"devDependencies": {
2529
"@types/cookie-parser": "1.4.6",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
import NextNodeServer from 'next/dist/server/next-server';
2+
import mockFs from 'mock-fs';
3+
import fs from 'fs';
4+
import { ImageOptimizerCache } from 'next/dist/server/image-optimizer';
5+
import { injectNextImageCacheDir } from './image-cache-handler';
6+
import { __getNextTestApp } from '../../__test-utils/utils';
7+
import { getNextServer } from '../next-utils';
8+
9+
describe('Set next.js image cache dir', () => {
10+
const nextApp = __getNextTestApp();
11+
let nextServer: NextNodeServer;
12+
13+
const imgCacheDir = 'myImgCacheDir';
14+
15+
process.env.IMAGE_CACHE_DIR = `/${imgCacheDir}`;
16+
17+
beforeAll(async () => {
18+
await nextApp.prepare();
19+
nextServer = await getNextServer(nextApp);
20+
await injectNextImageCacheDir(nextServer, process.env.IMAGE_CACHE_DIR);
21+
});
22+
23+
afterEach(() => {
24+
mockFs.restore();
25+
});
26+
27+
test('IncrementalCache should write to the correct image cache dir', async () => {
28+
mockFs();
29+
30+
// Handles next.js responseCache async fs write operation
31+
let resolveOnWriteFile: any = () => {};
32+
const promise = new Promise((resolve) => {
33+
resolveOnWriteFile = resolve;
34+
});
35+
36+
fs.promises.writeFile = jest.fn(resolveOnWriteFile);
37+
38+
const imgCache = new ImageOptimizerCache({
39+
distDir: 'dummyValueWhichGetsReplaceByInjector',
40+
nextConfig: nextServer['nextConfig'],
41+
});
42+
43+
await nextServer['imageResponseCache'].get(
44+
'foo',
45+
async () => ({
46+
value: {
47+
kind: 'IMAGE',
48+
buffer: Buffer.from('bar', 'utf-8'),
49+
etag: 'test',
50+
extension: 'png',
51+
},
52+
revalidate: 1337,
53+
}),
54+
{
55+
incrementalCache: imgCache,
56+
}
57+
);
58+
59+
await Promise.all([promise]).then(() =>
60+
expect(fs.promises.writeFile).toHaveBeenCalledWith(
61+
expect.stringContaining(imgCacheDir),
62+
expect.anything()
63+
)
64+
);
65+
});
66+
});
+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import ResponseCache from 'next/dist/server/response-cache';
2+
import { logger } from 'srcCommon/logger';
3+
import { ImageOptimizerCache } from 'next/dist/server/image-optimizer';
4+
import NextNodeServer from 'next/dist/server/next-server';
5+
import { propagateServerField } from 'next/dist/server/lib/render-server';
6+
import path from 'path';
7+
8+
class ImageCacheWithCustomCacheDir extends ResponseCache {
9+
private readonly cacheDir: string;
10+
11+
constructor(minimalMode: boolean, cacheDir: string) {
12+
logger.info(`Overriding image cache dir to ${cacheDir}`);
13+
super(minimalMode);
14+
this.cacheDir = cacheDir;
15+
}
16+
17+
public async get(...args: Parameters<ResponseCache['get']>) {
18+
const context = args[2] as { incrementalCache: ImageOptimizerCache };
19+
20+
try {
21+
context.incrementalCache['cacheDir'] = this.cacheDir;
22+
} catch (e) {
23+
logger.error(`Failed to set imageResponseCache cacheDir - ${e}`);
24+
}
25+
26+
return super.get(...args);
27+
}
28+
}
29+
30+
export const injectNextImageCacheDir = async (
31+
nextServer: NextNodeServer,
32+
cacheDir: string
33+
) => {
34+
const responseCache = new ImageCacheWithCustomCacheDir(false, cacheDir);
35+
36+
try {
37+
nextServer['imageResponseCache'] = responseCache;
38+
} catch (e) {
39+
logger.error(`Failed to set image cache dir on main server - ${e}`);
40+
}
41+
42+
// Also override the response cache for the request handler worker server
43+
try {
44+
const nextDir = nextServer['serverOptions'].dir || '.';
45+
46+
await propagateServerField(
47+
path.resolve(nextDir),
48+
'imageResponseCache' as any,
49+
responseCache
50+
);
51+
} catch (e) {
52+
logger.error(`Failed to set image cache dir on worker server - ${e}`);
53+
}
54+
};

server/src/cache/custom-cache-handler.ts server/src/cache/page-cache-handler.ts

+10-10
Original file line numberDiff line numberDiff line change
@@ -2,23 +2,23 @@ import FileSystemCache from 'next/dist/server/lib/incremental-cache/file-system-
22
import { LRUCache } from 'lru-cache';
33
import { CacheHandlerValue } from 'next/dist/server/lib/incremental-cache';
44
import { RedisCache, RedisCacheDummy } from 'cache/redis';
5-
import { isLeaderPod } from 'leader';
6-
import { logger } from 'srcCommon/logger';
5+
import { isLeaderPod } from 'leader-pod';
76

87
const CACHE_TTL_24_HOURS_IN_MS = 3600 * 24 * 1000;
98

10-
export const redisCache =
11-
process.env.ENV === 'localhost' && !process.env.REDIS_URI_PAGECACHE
12-
? new RedisCacheDummy()
13-
: new RedisCache({ ttl: CACHE_TTL_24_HOURS_IN_MS });
9+
const TTL_RESOLUTION_MS = 60 * 1000;
10+
11+
export const redisCache = !process.env.REDIS_URI_PAGECACHE
12+
? new RedisCacheDummy()
13+
: new RedisCache({ ttl: CACHE_TTL_24_HOURS_IN_MS });
1414

1515
const localCache = new LRUCache<string, CacheHandlerValue>({
16-
max: 1000,
16+
max: 2000,
1717
ttl: CACHE_TTL_24_HOURS_IN_MS,
18-
ttlResolution: 1000,
18+
ttlResolution: TTL_RESOLUTION_MS,
1919
});
2020

21-
export default class CustomCacheHandler {
21+
export default class PageCacheHandler {
2222
public async get(...args: Parameters<FileSystemCache['get']>) {
2323
const [key] = args;
2424

@@ -38,7 +38,7 @@ export default class CustomCacheHandler {
3838
Date.now()
3939
: CACHE_TTL_24_HOURS_IN_MS;
4040

41-
if (ttlRemaining > 1000) {
41+
if (ttlRemaining > TTL_RESOLUTION_MS) {
4242
localCache.set(key, fromRedisCache, {
4343
ttl: ttlRemaining,
4444
});

0 commit comments

Comments
 (0)