Skip to content

Commit 1baddc1

Browse files
authored
fix(apisix): store stream route name correctly (#156)
1 parent d745ad4 commit 1baddc1

File tree

6 files changed

+108
-15
lines changed

6 files changed

+108
-15
lines changed

apps/cli/src/linter/schema.ts

+10-6
Original file line numberDiff line numberDiff line change
@@ -218,12 +218,7 @@ const serviceSchema = z
218218

219219
upstream: upstreamSchema.optional(),
220220
plugins: pluginsSchema.optional(),
221-
path_prefix: z
222-
.string()
223-
.optional()
224-
.refine((val) => val?.startsWith('/'), {
225-
message: 'Path prefix must start with "/"',
226-
}),
221+
path_prefix: z.string().optional(),
227222
strip_path_prefix: z.boolean().optional(),
228223
hosts: z.array(z.string()).optional(),
229224

@@ -237,6 +232,15 @@ const serviceSchema = z
237232
message:
238233
'HTTP routes and Stream routes are mutually exclusive and should not exist in the same service',
239234
},
235+
)
236+
.refine(
237+
(val) => {
238+
if (!val.path_prefix) return true;
239+
return val.path_prefix.startsWith('/');
240+
},
241+
{
242+
message: 'Path prefix must start with "/"',
243+
},
240244
);
241245

242246
const sslSchema = z

libs/backend-apisix/e2e/assets/apisix_conf/http.yaml

+5
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,11 @@ apisix:
55
control:
66
ip: "0.0.0.0"
77
port: 9092
8+
proxy_mode: http&stream
9+
stream_proxy:
10+
tcp:
11+
- addr: 33060
12+
tls: true
813
deployment:
914
admin:
1015
allow_admin:

libs/backend-apisix/e2e/sync-and-dump-1.e2e-spec.ts

+63
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,69 @@ describe('Sync and Dump - 1', () => {
164164
});
165165
});
166166

167+
describe('Sync and dump service with stream route', () => {
168+
const serviceName = 'test';
169+
const service = {
170+
name: serviceName,
171+
upstream: {
172+
scheme: 'tcp',
173+
nodes: [
174+
{
175+
host: '1.1.1.1',
176+
port: 5432,
177+
weight: 100,
178+
},
179+
],
180+
},
181+
} as ADCSDK.Service;
182+
const route1Name = 'postgres';
183+
const route1 = {
184+
name: route1Name,
185+
server_port: 54320,
186+
} as ADCSDK.StreamRoute;
187+
188+
it('Create resources', async () =>
189+
syncEvents(backend, [
190+
createEvent(ADCSDK.ResourceType.SERVICE, serviceName, service),
191+
createEvent(
192+
ADCSDK.ResourceType.STREAM_ROUTE,
193+
route1Name,
194+
route1,
195+
serviceName,
196+
),
197+
]));
198+
199+
it('Dump', async () => {
200+
const result = (await dumpConfiguration(backend)) as ADCSDK.Configuration;
201+
expect(result.services).toHaveLength(1);
202+
expect(result.services[0]).toMatchObject(service);
203+
expect(result.services[0].stream_routes).toHaveLength(1);
204+
expect(result.services[0].stream_routes[0]).toMatchObject(route1);
205+
});
206+
207+
it('Delete stream route', async () =>
208+
syncEvents(backend, [
209+
deleteEvent(ADCSDK.ResourceType.STREAM_ROUTE, route1Name, serviceName),
210+
]));
211+
212+
it('Dump again (non-route)', async () => {
213+
const result = (await dumpConfiguration(backend)) as ADCSDK.Configuration;
214+
expect(result.services).toHaveLength(1);
215+
expect(result.services[0]).toMatchObject(service);
216+
expect(result.services[0].stream_routes).toBeUndefined();
217+
});
218+
219+
it('Delete service', async () =>
220+
syncEvents(backend, [
221+
deleteEvent(ADCSDK.ResourceType.SERVICE, serviceName),
222+
]));
223+
224+
it('Dump again (service should not exist)', async () => {
225+
const result = (await dumpConfiguration(backend)) as ADCSDK.Configuration;
226+
expect(result.services).toHaveLength(0);
227+
});
228+
});
229+
167230
describe('Sync and dump consumers', () => {
168231
const consumer1Name = 'consumer1';
169232
const consumer1 = {

libs/backend-apisix/src/transformer.ts

+25-6
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,20 @@
11
import * as ADCSDK from '@api7/adc-sdk';
2-
import { isEmpty, unset } from 'lodash';
2+
import { filter, isEmpty, unset } from 'lodash';
33

44
import * as typing from './typing';
55

66
export class ToADC {
7+
private static transformLabels(labels?: ADCSDK.Labels): ADCSDK.Labels {
8+
if (!labels) return undefined;
9+
const filteredLabels = filter(
10+
labels,
11+
(val, key) => key !== '__ADC_NAME',
12+
) as unknown as ADCSDK.Labels;
13+
return Object.values(filteredLabels).length > 0
14+
? filteredLabels
15+
: undefined;
16+
}
17+
718
public transformRoute(route: typing.Route): ADCSDK.Route {
819
return ADCSDK.utils.recursiveOmitUndefined({
920
name: route.name ?? route.id,
@@ -120,9 +131,9 @@ export class ToADC {
120131
streamRoute: typing.StreamRoute,
121132
): ADCSDK.StreamRoute {
122133
return ADCSDK.utils.recursiveOmitUndefined({
123-
name: streamRoute.name ?? streamRoute.id,
134+
name: streamRoute.labels?.__ADC_NAME ?? streamRoute.id,
124135
description: streamRoute.desc,
125-
labels: streamRoute.labels,
136+
labels: ToADC.transformLabels(streamRoute.labels),
126137

127138
remote_addr: streamRoute.remote_addr,
128139
server_addr: streamRoute.server_addr,
@@ -343,10 +354,18 @@ export class FromADC {
343354
streamRoute: ADCSDK.StreamRoute,
344355
): typing.StreamRoute {
345356
return ADCSDK.utils.recursiveOmitUndefined({
346-
...streamRoute,
347357
id: undefined,
348-
labels: FromADC.transformLabels(streamRoute.labels),
349-
});
358+
desc: streamRoute.description,
359+
labels: {
360+
...FromADC.transformLabels(streamRoute.labels),
361+
__ADC_NAME: streamRoute.name,
362+
},
363+
plugins: streamRoute.plugins,
364+
remote_addr: streamRoute.remote_addr,
365+
server_addr: streamRoute.server_addr,
366+
server_port: streamRoute.server_port,
367+
sni: streamRoute.sni,
368+
} as typing.StreamRoute);
350369
}
351370

352371
public transformUpstream(upstream: ADCSDK.Upstream): typing.Upstream {

libs/backend-apisix/src/typing.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -108,7 +108,7 @@ export interface StreamRoute {
108108
id: string;
109109
desc?: string;
110110
labels?: Labels;
111-
name: string;
111+
//name: string; // As of 3.9.1, APISIX does not support name on the stream route
112112

113113
remote_addr?: string;
114114
server_addr?: string;

schema.json

+4-2
Original file line numberDiff line numberDiff line change
@@ -230,6 +230,9 @@
230230
"additionalProperties": false
231231
}
232232
},
233+
"required": [
234+
"active"
235+
],
233236
"additionalProperties": false
234237
},
235238
"nodes": {
@@ -538,8 +541,7 @@
538541
}
539542
},
540543
"required": [
541-
"name",
542-
"path_prefix"
544+
"name"
543545
],
544546
"additionalProperties": false
545547
}

0 commit comments

Comments
 (0)