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

update HeaderMiddleware types, export PatchStrategy with Integration Tests #2316

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
3 changes: 3 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -26,3 +26,6 @@ jobs:
- run: npm run lint
- run: npm audit --audit-level=critical
- run: npm run build-with-tests && npm run test-transpiled
- name: Create k8s Kind Cluster
uses: helm/kind-action@v1
- run: npm run integration-test
4 changes: 2 additions & 2 deletions examples/typescript/patch/patch-example.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { CoreV1Api, KubeConfig, setHeaderOptions, JsonPatch } from '@kubernetes/client-node';
import { CoreV1Api, KubeConfig, setHeaderOptions, PatchStrategy } from '@kubernetes/client-node';

const kc = new KubeConfig();
kc.loadFromDefault();
Expand Down Expand Up @@ -28,7 +28,7 @@ try {
namespace: 'default',
body: patch,
},
setHeaderOptions('Content-Type', JsonPatch),
setHeaderOptions('Content-Type', PatchStrategy.JsonPatch),
);

console.log('Patched.');
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
"watch": "tsc --watch",
"test": "c8 mocha",
"test-transpiled": "mocha --no-config dist",
"integration-test": "tsx src/test/integration/index.ts",
"prepare": "npm run build && husky",
"prepack": "npm run build",
"docs": "typedoc src/gen"
Expand Down
1 change: 1 addition & 0 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ export * from './metrics.js';
export * from './object.js';
export * from './health.js';
export * from './middleware.js';
export * from './patch.js';
export { type ConfigOptions, type User, type Cluster, type Context } from './config_types.js';

// Export FetchError so that instanceof checks in user code will definitely use the same instance
Expand Down
34 changes: 18 additions & 16 deletions src/middleware.ts
Original file line number Diff line number Diff line change
@@ -1,27 +1,29 @@
import type { RequestContext, ResponseContext, ConfigurationOptions, Middleware } from './gen/index.js';
import { PromiseMiddleware } from './gen/middleware.js';
import type {
RequestContext,
ResponseContext,
ConfigurationOptions,
ObservableMiddleware,
} from './gen/index.js';
import { of } from './gen/rxjsStub.js';

// setHeaderMiddleware returns Middleware[] that sets a header value
export function setHeaderMiddleware(key: string, value: string): Middleware[] {
return [
{
pre: async (c: RequestContext) => {
c.setHeaderParam(key, value);
return c;
},
post: async (c: ResponseContext) => {
return c;
},
export function setHeaderMiddleware(key: string, value: string): ObservableMiddleware {
return {
pre: (request: RequestContext) => {
request.setHeaderParam(key, value);
return of(request);
},
post: (response: ResponseContext) => {
return of(response);
},
];
};
}

// Returns ConfigurationOptions that set a header
export function setHeaderOptions(
key: string,
value: string,
opt?: ConfigurationOptions<PromiseMiddleware>,
): ConfigurationOptions<PromiseMiddleware> {
opt?: ConfigurationOptions<ObservableMiddleware>,
): ConfigurationOptions<ObservableMiddleware> {
const newMiddlware = setHeaderMiddleware(key, value);
const existingMiddlware = opt?.middleware || [];
return {
Expand Down
15 changes: 9 additions & 6 deletions src/middleware_test.ts
Original file line number Diff line number Diff line change
@@ -1,15 +1,15 @@
import { RequestContext, ConfigurationOptions, HttpMethod } from './gen/index.js';
import { RequestContext, ConfigurationOptions, HttpMethod, ObservableMiddleware } from './gen/index.js';
import { deepStrictEqual } from 'node:assert';
import { setHeaderMiddleware, setHeaderOptions } from './middleware.js';
import { PromiseMiddleware } from './gen/middleware.js';

describe('Middleware', async () => {
describe('setHeaderMiddleware', async () => {
it('should set a header when it is blank', async () => {
const reqContext = new RequestContext('http://nowhere.com', HttpMethod.GET);
deepStrictEqual(reqContext.getHeaders(), {});
const headerMiddleware = setHeaderMiddleware('test-key', 'test-value');
const postMiddlewareRequest = await headerMiddleware[0].pre(reqContext);
const postMiddlewareRequestObservable = await headerMiddleware.pre(reqContext);
const postMiddlewareRequest = await postMiddlewareRequestObservable.toPromise();
deepStrictEqual(postMiddlewareRequest.getHeaders(), { 'test-key': 'test-value' });
});

Expand All @@ -18,7 +18,7 @@ describe('Middleware', async () => {
reqContext.setHeaderParam('test-key', 'wrong-value');
deepStrictEqual(reqContext.getHeaders(), { 'test-key': 'wrong-value' });
const headerMiddleware = setHeaderMiddleware('test-key', 'test-value');
const postMiddlewareRequest = await headerMiddleware[0].pre(reqContext);
const postMiddlewareRequest = await headerMiddleware.pre(reqContext).toPromise();
deepStrictEqual(postMiddlewareRequest.getHeaders(), { 'test-key': 'test-value' });
});
});
Expand All @@ -27,7 +27,7 @@ describe('Middleware', async () => {
it('should add middleware to set header with no input options arg', async () => {
const reqContext = new RequestContext('http://nowhere.com', HttpMethod.GET);
deepStrictEqual(reqContext.getHeaders(), {});
const testConfigurationOptions: ConfigurationOptions<PromiseMiddleware> = {};
const testConfigurationOptions: ConfigurationOptions<ObservableMiddleware> = {};
const headerConfigurationOptions = setHeaderOptions(
'test-key',
'test-value',
Expand All @@ -39,7 +39,10 @@ describe('Middleware', async () => {
) {
throw new Error('missing middleware in ConfigurationOptions');
}
const postMiddlewareRequest = await headerConfigurationOptions.middleware[0].pre(reqContext);
const postMiddlewareRequest = await headerConfigurationOptions.middleware[0]
.pre(reqContext)
.toPromise();

deepStrictEqual(postMiddlewareRequest.getHeaders(), { 'test-key': 'test-value' });
});
});
Expand Down
5 changes: 5 additions & 0 deletions src/test/integration/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import patchNamespace from './patchNamespace.js';

console.log('Integration testing');

await patchNamespace();
7 changes: 7 additions & 0 deletions src/test/integration/name.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
/*
* This function is used to generate a random name for the resources created in the tests.
* This is to avoid conflicts with existing resources.
*/
export function generateName(name: string) {
return name + '-' + Math.random().toString(36).substring(7);
}
48 changes: 48 additions & 0 deletions src/test/integration/patchNamespace.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import assert from 'node:assert';
import { PatchStrategy, CoreV1Api, KubeConfig, setHeaderOptions, V1Namespace } from '../../index.js';
import { generateName } from './name.js';

export default async function patchNamespace() {
const kc = new KubeConfig();
kc.loadFromDefault();

const coreV1Client = kc.makeApiClient(CoreV1Api);
let newNS = new V1Namespace();
const testNSName = generateName('patch-ns');
newNS.metadata = {
name: testNSName,
labels: {
initialLabel: 'initialValue',
},
};
console.log(`Creating namespace ${testNSName}`);
await coreV1Client.createNamespace({ body: newNS });
newNS = await coreV1Client.readNamespace({ name: testNSName });
assert.strictEqual(newNS.metadata?.name, testNSName);
assert.strictEqual(newNS.metadata?.labels?.initialLabel, 'initialValue');
console.log('Namespace created with initial label.');

const patch = [
{
op: 'replace',
path: '/metadata/labels',
value: {
foo: 'bar',
},
},
];

console.log(`Patching namespace ${testNSName} to replace labels`);
await coreV1Client.patchNamespace(
{
name: testNSName,
body: patch,
},
setHeaderOptions('Content-Type', PatchStrategy.JsonPatch),
);

const patchedNS = await coreV1Client.readNamespace({ name: testNSName });
assert.strictEqual(patchedNS.metadata?.labels?.foo, 'bar');
assert.strictEqual(patchedNS.metadata?.labels?.initialLabel, undefined);
console.log('Namespace patched with new label.');
}