Skip to content

Commit

Permalink
feat: Support upgrading built-in Wasm plugins (#327)
Browse files Browse the repository at this point in the history
  • Loading branch information
CH3CHO authored Sep 2, 2024
1 parent 459474b commit e30551a
Show file tree
Hide file tree
Showing 7 changed files with 189 additions and 91 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -128,14 +128,13 @@ private void reloadServiceInfoFromK8s() {
V1Secret secret = kubernetesClientService.readSecret(secretName);
Map<String, byte[]> data = secret.getData();
if (MapUtils.isEmpty(data)) {
log.error("Secret {} is empty.", secretName);
log.warn("Secret {} is empty.", secretName);
return;
}
byte[] serviceUrlData = data.get(SERVICE_URL_KEY);
byte[] serviceTokenData = data.get(SERVICE_TOKEN_KEY);
if (serviceUrlData == null || serviceUrlData.length == 0 || serviceTokenData == null
|| serviceTokenData.length == 0) {
log.error("Secret {} does not contain service URL or token for ai-proxy.", secretName);
return;
}
serviceInfoHolder.set(new ServiceInfo(new String(serviceUrlData), new String(serviceTokenData)));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,9 @@ public ResponseEntity<Response<WasmPlugin>> update(@PathVariable("name") @NotBla
throw new ValidationException("Plugin name in the URL doesn't match the one in the body.");
}
plugin.validate();
WasmPlugin updatedPlugin = wasmPluginService.updateCustom(plugin);
WasmPlugin updatedPlugin = Boolean.TRUE.equals(plugin.getBuiltIn())
? wasmPluginService.updateBuiltIn(plugin.getName(), plugin.getImageVersion())
: wasmPluginService.updateCustom(plugin);
return ControllerUtil.buildResponseEntity(updatedPlugin);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,8 @@ public interface WasmPluginService {

String queryReadme(String name, String language);

WasmPlugin updateBuiltIn(String name, String imageVersion);

WasmPlugin addCustom(WasmPlugin plugin);

WasmPlugin updateCustom(WasmPlugin plugin);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
import javax.annotation.PostConstruct;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.google.common.base.Preconditions;
import io.swagger.v3.core.util.Json;
import io.swagger.v3.core.util.Yaml;
import io.swagger.v3.oas.models.media.Schema;
Expand Down Expand Up @@ -174,7 +175,7 @@ private void fillPluginConfigExample(Plugin plugin, String content) {
|| plugin.getSpec().getConfigSchema().getOpenApiV3Schema() == null) {
return;
}
Schema schema = plugin.getSpec().getConfigSchema().getOpenApiV3Schema();
Schema<?> schema = plugin.getSpec().getConfigSchema().getOpenApiV3Schema();
schema.addExtension(EXAMPLE_RAW_PROPERTY_NAME, example);
}

Expand Down Expand Up @@ -255,22 +256,29 @@ private String loadPluginReadme(String pluginId, String fileName) {
@Override
public PaginatedResult<WasmPlugin> list(WasmPluginPageQuery query) {
String lang = query != null ? query.getLang() : null;
List<Object> plugins = new ArrayList<>(builtInPlugins);
List<WasmPlugin> plugins = new ArrayList<>();
for (PluginCacheItem item : builtInPlugins) {
plugins.add(item.buildWasmPlugin(lang));
}
try {
List<V1alpha1WasmPlugin> customPluginCrs = kubernetesClientService.listWasmPlugin(null, null, false);
plugins.addAll(customPluginCrs);
List<V1alpha1WasmPlugin> crs = kubernetesClientService.listWasmPlugin();
for (V1alpha1WasmPlugin cr : crs) {
WasmPlugin plugin = kubernetesModelConverter.wasmPluginFromCr(cr);
if (plugin.getBuiltIn()) {
WasmPlugin builtInPlugin = plugins.stream()
.filter(p -> p.getName().equals(plugin.getName())).findFirst().orElse(null);
if (builtInPlugin != null){
builtInPlugin.setImageRepository(plugin.getImageRepository());
builtInPlugin.setImageVersion(plugin.getImageVersion());
continue;
}
}
plugins.add(plugin);
}
} catch (ApiException e) {
throw new BusinessException("Error occurs when listing custom Wasm plugins", e);
}
return PaginatedResult.createFromFullList(plugins, query, o -> {
if (o instanceof PluginCacheItem) {
return ((PluginCacheItem)o).buildWasmPlugin(lang);
}
if (o instanceof V1alpha1WasmPlugin) {
return kubernetesModelConverter.wasmPluginFromCr((V1alpha1WasmPlugin)o);
}
throw new IllegalStateException("Unexpected element type: " + o.getClass().getName());
});
return PaginatedResult.createFromFullList(plugins, query);
}

@Override
Expand Down Expand Up @@ -325,7 +333,7 @@ public WasmPluginConfig queryConfig(String name, String language) {
if (CollectionUtils.isNotEmpty(crs)) {
// TODO: Config of a custom plugin is not supported yet. Return an empty schema instead.
WasmPluginConfig config = new WasmPluginConfig();
Schema schema = new Schema();
Schema<?> schema = new Schema<>();
schema.setType("object");
config.setSchema(schema);
return config;
Expand Down Expand Up @@ -366,6 +374,63 @@ public String queryReadme(String name, String language) {
return null;
}

@Override
public WasmPlugin updateBuiltIn(String name, String imageVersion) {
Preconditions.checkArgument(StringUtils.isNotEmpty(name), "name cannot be blank.");
Preconditions.checkArgument(StringUtils.isNotEmpty(imageVersion), "imageVersion cannot be blank.");

PluginCacheItem builtInPlugin =
builtInPlugins.stream().filter(p -> p.getName().equals(name)).findFirst().orElse(null);
if (builtInPlugin == null) {
throw new ResourceConflictException("No built-in plugin is found with the given name: " + name);
}

List<V1alpha1WasmPlugin> existedCrs;
try {
final String pluginVersion = builtInPlugin.getPlugin().getInfo().getVersion();
existedCrs = kubernetesClientService.listWasmPlugin(name, pluginVersion, true);
} catch (ApiException e) {
throw new BusinessException("Error occurs when checking existed Wasm plugins with name " + name, e);
}

V1alpha1WasmPlugin updatedCr;
if (CollectionUtils.isEmpty(existedCrs)) {
WasmPlugin plugin = builtInPlugin.buildWasmPlugin();
plugin.setImageVersion(imageVersion);
V1alpha1WasmPlugin cr = kubernetesModelConverter.wasmPluginToCr(plugin);
// Make sure it is disabled by default.
cr.getSpec().setDefaultConfigDisable(true);
try {
updatedCr = kubernetesClientService.createWasmPlugin(cr);
} catch (ApiException e) {
if (e.getCode() == HttpStatus.CONFLICT) {
throw new ResourceConflictException();
}
throw new BusinessException(
"Error occurs when adding a new Wasm plugin with name: " + cr.getMetadata().getName(), e);
}
} else {
V1alpha1WasmPlugin existedCr = existedCrs.get(0);

WasmPlugin plugin = kubernetesModelConverter.wasmPluginFromCr(existedCr);
plugin.setImageVersion(imageVersion);
updatedCr = kubernetesModelConverter.wasmPluginToCr(plugin);
kubernetesModelConverter.mergeWasmPluginSpec(existedCr, updatedCr);

try {
updatedCr = kubernetesClientService.replaceWasmPlugin(updatedCr);
} catch (ApiException e) {
if (e.getCode() == HttpStatus.CONFLICT) {
throw new ResourceConflictException();
}
throw new BusinessException(
"Error occurs when updating the Wasm plugin wth name " + existedCr.getMetadata().getName(), e);
}
}

return kubernetesModelConverter.wasmPluginFromCr(updatedCr);
}

@Override
public WasmPlugin addCustom(WasmPlugin plugin) {
if (Boolean.TRUE.equals(plugin.getBuiltIn())) {
Expand Down Expand Up @@ -549,6 +614,10 @@ public void setReadme(String language, String content) {
}
}

public WasmPlugin buildWasmPlugin() {
return buildWasmPlugin(null);
}

public WasmPlugin buildWasmPlugin(String language) {
WasmPlugin wasmPlugin = new WasmPlugin();
wasmPlugin.setName(name);
Expand Down Expand Up @@ -596,9 +665,11 @@ public WasmPluginConfig buildWasmPluginConfig(String language) {
|| plugin.getSpec().getConfigSchema().getOpenApiV3Schema() == null) {
return new WasmPluginConfig();
}
Schema schema = null;
Schema<?> schema;
try {
schema = Json.mapper().readValue(Json.mapper().writeValueAsString(plugin.getSpec().getConfigSchema().getOpenApiV3Schema()), Schema.class);
schema = Json.mapper().readValue(
Json.mapper().writeValueAsString(plugin.getSpec().getConfigSchema().getOpenApiV3Schema()),
Schema.class);
} catch (JsonProcessingException e) {
throw new RuntimeException(e);
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
// eslint-disable-next-line max-len
export const DEFAULT_PLUGIN_IMG = ``;

export const BUILTIN_PLUGIN_LIST = [
export const BUILTIN_ROUTE_PLUGIN_LIST = [
{
key: 'rewrite',
title: '重写',
Expand Down
96 changes: 51 additions & 45 deletions frontend/src/pages/plugin/components/PluginList/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import { Avatar, Button, Card, Col, Dropdown, Popconfirm, Row, Typography } from
import { useSearchParams } from 'ice';
import { forwardRef, useEffect, useImperativeHandle, useMemo, useState } from 'react';
import { useTranslation } from 'react-i18next';
import { BUILTIN_PLUGIN_LIST, DEFAULT_PLUGIN_IMG } from './constant';
import { BUILTIN_ROUTE_PLUGIN_LIST, DEFAULT_PLUGIN_IMG } from './constant';
import { getI18nValue } from '../../utils';

const { Paragraph } = Typography;
Expand All @@ -29,12 +29,6 @@ const PluginList = forwardRef((props: Props, ref) => {
const { data, onOpen, onEdit, onDelete } = props;
const [searchParams] = useSearchParams();

const type = searchParams.get('type');

const showBuiltInPlugins = useMemo(() => {
return type === 'route';
}, [type]);

const handleClickPlugin = (item) => {
onOpen(item);
};
Expand All @@ -46,8 +40,8 @@ const PluginList = forwardRef((props: Props, ref) => {
}, {
manual: true,
onSuccess: (result = []) => {
if (showBuiltInPlugins) {
setPluginList(BUILTIN_PLUGIN_LIST.concat(result) as any);
if (searchParams.get('type') === 'route') {
setPluginList(BUILTIN_ROUTE_PLUGIN_LIST.concat(result) as any);
return;
}
setPluginList(result);
Expand All @@ -68,6 +62,51 @@ const PluginList = forwardRef((props: Props, ref) => {

i18n.on('languageChanged', () => loadWasmPlugins());

const createPluginDropdown = (plugin) => {
if (BUILTIN_ROUTE_PLUGIN_LIST.some(p => p.key === plugin.key)) {
return null;
}
const items = [
{
key: 'edit',
label: (
<span
onClick={() => {
onEdit?.(plugin);
}}
>
{t('misc.edit')}
</span>
),
},
];
if (!plugin.builtIn) {
items.push({
key: 'delete',
label: (
<Popconfirm
title={t('plugins.deleteConfirmation')}
onConfirm={() => {
onDelete?.(plugin.name);
}}
>
<span>{t('misc.delete')}</span>
</Popconfirm>
),
danger: true,
});
}
return (
<Dropdown
menu={{
items,
}}
>
<EllipsisOutlined />
</Dropdown>
)
};

return (
<Row gutter={[16, 16]}>
{pluginList.map((item) => {
Expand Down Expand Up @@ -101,42 +140,9 @@ const PluginList = forwardRef((props: Props, ref) => {
<div style={{ flex: 1, overflow: 'hidden', textOverflow: 'ellipsis', whiteSpace: 'nowrap' }}>
{getI18nValue(item, 'title')}
</div>
{item.builtIn === false ? (
<Dropdown
menu={{
items: [
{
key: 'edit',
label: (
<span
onClick={() => {
onEdit?.(item);
}}
>
{t('misc.edit')}
</span>
),
},
{
key: 'delete',
label: (
<Popconfirm
title={t('plugins.deleteConfirmation')}
onConfirm={() => {
onDelete?.(item.name);
}}
>
<span>{t('misc.delete')}</span>
</Popconfirm>
),
danger: true,
},
],
}}
>
<EllipsisOutlined />
</Dropdown>
) : undefined}
{
createPluginDropdown(item)
}
</div>
}
description={
Expand Down
Loading

0 comments on commit e30551a

Please sign in to comment.