mirror of
https://github.com/kestra-io/kestra.git
synced 2025-12-25 20:00:14 -05:00
Compare commits
19 Commits
spike/gene
...
develop
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cb31e48f4f | ||
|
|
a3f96a2741 | ||
|
|
5ca6fa8d77 | ||
|
|
a3a206f3c4 | ||
|
|
31f1e505e3 | ||
|
|
75e0c1d11f | ||
|
|
4cf883877d | ||
|
|
87b1e8fb01 | ||
|
|
b5c6101090 | ||
|
|
e4323728d6 | ||
|
|
5495971ecf | ||
|
|
dec1ee4272 | ||
|
|
69cc6b2715 | ||
|
|
431b4ccdb9 | ||
|
|
e9207a6f53 | ||
|
|
419c1041d5 | ||
|
|
02cd5efb05 | ||
|
|
2d549940c4 | ||
|
|
97e138fbae |
4
.github/CONTRIBUTING.md
vendored
4
.github/CONTRIBUTING.md
vendored
@@ -63,9 +63,9 @@ You can also build it from a terminal using `./gradlew build`, the Gradle wrappe
|
||||
- Configure the following environment variables:
|
||||
- `MICRONAUT_ENVIRONMENTS`: can be set to any string and will load a custom configuration file in `cli/src/main/resources/application-{env}.yml`.
|
||||
- `KESTRA_PLUGINS_PATH`: is the path where you will save plugins as Jar and will be load on startup.
|
||||
- See the screenshot below for an example: 
|
||||
- See the screenshot below for an example: 
|
||||
- If you encounter **JavaScript memory heap out** error during startup, configure `NODE_OPTIONS` environment variable with some large value.
|
||||
- Example `NODE_OPTIONS: --max-old-space-size=4096` or `NODE_OPTIONS: --max-old-space-size=8192` 
|
||||
- Example `NODE_OPTIONS: --max-old-space-size=4096` or `NODE_OPTIONS: --max-old-space-size=8192` 
|
||||
- The server starts by default on port 8080 and is reachable on `http://localhost:8080`
|
||||
|
||||
If you want to launch all tests, you need Python and some packages installed on your machine, on Ubuntu you can install them with:
|
||||
|
||||
|
Before Width: | Height: | Size: 130 KiB After Width: | Height: | Size: 130 KiB |
|
Before Width: | Height: | Size: 210 KiB After Width: | Height: | Size: 210 KiB |
@@ -222,6 +222,7 @@ subprojects {subProj ->
|
||||
|
||||
def commonTestConfig = { Test t ->
|
||||
t.ignoreFailures = true
|
||||
t.finalizedBy jacocoTestReport
|
||||
|
||||
// set Xmx for test workers
|
||||
t.maxHeapSize = '4g'
|
||||
|
||||
@@ -4,7 +4,6 @@ import io.micronaut.configuration.picocli.PicocliRunner;
|
||||
import io.micronaut.context.ApplicationContext;
|
||||
import io.micronaut.context.env.Environment;
|
||||
import org.apache.commons.io.FileUtils;
|
||||
import org.junit.jupiter.api.BeforeAll;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
import java.io.File;
|
||||
@@ -15,7 +14,6 @@ import java.nio.file.Files;
|
||||
import java.nio.file.Path;
|
||||
import java.util.List;
|
||||
import java.util.Objects;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
import static org.assertj.core.api.Assertions.assertThat;
|
||||
|
||||
@@ -25,7 +23,8 @@ class PluginDocCommandTest {
|
||||
|
||||
@Test
|
||||
void run() throws IOException, URISyntaxException {
|
||||
Path pluginsPath = Files.createTempDirectory(PluginListCommandTest.class.getSimpleName());
|
||||
var testDirectoryName = PluginListCommandTest.class.getSimpleName();
|
||||
Path pluginsPath = Files.createTempDirectory(testDirectoryName + "_pluginsPath_");
|
||||
pluginsPath.toFile().deleteOnExit();
|
||||
|
||||
FileUtils.copyFile(
|
||||
@@ -34,7 +33,7 @@ class PluginDocCommandTest {
|
||||
new File(URI.create("file://" + pluginsPath.toAbsolutePath() + "/" + PLUGIN_TEMPLATE_TEST))
|
||||
);
|
||||
|
||||
Path docPath = Files.createTempDirectory(PluginInstallCommandTest.class.getSimpleName());
|
||||
Path docPath = Files.createTempDirectory(testDirectoryName + "_docPath_");
|
||||
docPath.toFile().deleteOnExit();
|
||||
|
||||
try (ApplicationContext ctx = ApplicationContext.run(Environment.CLI, Environment.TEST)) {
|
||||
@@ -43,9 +42,9 @@ class PluginDocCommandTest {
|
||||
|
||||
List<Path> files = Files.list(docPath).toList();
|
||||
|
||||
assertThat(files.size()).isEqualTo(1);
|
||||
assertThat(files.getFirst().getFileName().toString()).isEqualTo("plugin-template-test");
|
||||
var directory = files.getFirst().toFile();
|
||||
assertThat(files.stream().map(path -> path.getFileName().toString())).contains("plugin-template-test");
|
||||
// don't know why, but sometimes there is an addition "plugin-notifications" directory present
|
||||
var directory = files.stream().filter(path -> "plugin-template-test".equals(path.getFileName().toString())).findFirst().get().toFile();
|
||||
assertThat(directory.isDirectory()).isTrue();
|
||||
assertThat(directory.listFiles().length).isEqualTo(3);
|
||||
|
||||
|
||||
@@ -8,25 +8,31 @@ import io.micronaut.test.extensions.junit5.MicronautJunit5Extension;
|
||||
import org.junit.jupiter.api.extension.ExtensionContext;
|
||||
import org.junit.platform.commons.support.AnnotationSupport;
|
||||
|
||||
import java.util.Set;
|
||||
|
||||
public class KestraTestExtension extends MicronautJunit5Extension {
|
||||
@Override
|
||||
protected MicronautTestValue buildMicronautTestValue(Class<?> testClass) {
|
||||
testProperties.put("kestra.jdbc.executor.thread-count", Runtime.getRuntime().availableProcessors() * 4);
|
||||
return AnnotationSupport
|
||||
.findAnnotation(testClass, KestraTest.class)
|
||||
.map(kestraTestAnnotation -> new MicronautTestValue(
|
||||
kestraTestAnnotation.application(),
|
||||
kestraTestAnnotation.environments(),
|
||||
kestraTestAnnotation.packages(),
|
||||
kestraTestAnnotation.propertySources(),
|
||||
kestraTestAnnotation.rollback(),
|
||||
kestraTestAnnotation.transactional(),
|
||||
kestraTestAnnotation.rebuildContext(),
|
||||
kestraTestAnnotation.contextBuilder(),
|
||||
kestraTestAnnotation.transactionMode(),
|
||||
kestraTestAnnotation.startApplication(),
|
||||
kestraTestAnnotation.resolveParameters()
|
||||
))
|
||||
.map(kestraTestAnnotation -> {
|
||||
var envsSet = new java.util.HashSet<>(Set.of(kestraTestAnnotation.environments()));
|
||||
envsSet.add("test");// add test env if not already present
|
||||
return new MicronautTestValue(
|
||||
kestraTestAnnotation.application(),
|
||||
envsSet.toArray(new String[0]),
|
||||
kestraTestAnnotation.packages(),
|
||||
kestraTestAnnotation.propertySources(),
|
||||
kestraTestAnnotation.rollback(),
|
||||
kestraTestAnnotation.transactional(),
|
||||
kestraTestAnnotation.rebuildContext(),
|
||||
kestraTestAnnotation.contextBuilder(),
|
||||
kestraTestAnnotation.transactionMode(),
|
||||
kestraTestAnnotation.startApplication(),
|
||||
kestraTestAnnotation.resolveParameters()
|
||||
);
|
||||
})
|
||||
.orElse(null);
|
||||
}
|
||||
|
||||
|
||||
3
ui/.gitignore
vendored
3
ui/.gitignore
vendored
@@ -3,5 +3,4 @@ test-results
|
||||
tests/.env
|
||||
tests/data/
|
||||
tests/e2e/.env
|
||||
tests/e2e/data/application-secrets.yml
|
||||
generated/
|
||||
tests/e2e/data/application-secrets.yml
|
||||
@@ -23,13 +23,6 @@ export default defineConfig([
|
||||
],
|
||||
languageOptions: {globals: globals.node},
|
||||
},
|
||||
{
|
||||
files: ["src/generated/**/*.ts"],
|
||||
rules: {
|
||||
"@typescript-eslint/ban-ts-comment": "off",
|
||||
"@typescript-eslint/no-empty-object-type": "off",
|
||||
},
|
||||
},
|
||||
...pluginVue.configs["flat/strongly-recommended"],
|
||||
{
|
||||
files: ["**/*.vue", "**/*.tsx", "**/*.jsx"],
|
||||
|
||||
@@ -1,20 +0,0 @@
|
||||
import {definePluginConfig} from "@hey-api/openapi-ts";
|
||||
import {handler} from "./plugin";
|
||||
import type {KestraSdkPlugin} from "./types";
|
||||
|
||||
const defaultConfig: KestraSdkPlugin["Config"] = {
|
||||
config: {
|
||||
output: "kestra-sdk",
|
||||
methodNameBuilder(operation) {
|
||||
return operation.operationId
|
||||
}
|
||||
},
|
||||
dependencies: ["@hey-api/typescript", "@hey-api/client-axios", "@hey-api/sdk"],
|
||||
handler,
|
||||
name: "ks-sdk",
|
||||
};
|
||||
|
||||
/**
|
||||
* Type helper for `@kestra-io/sdk-plugin` plugin, returns {@link Plugin.Config} object
|
||||
*/
|
||||
export const defineKestraHeyConfig = definePluginConfig(defaultConfig);
|
||||
@@ -1,2 +0,0 @@
|
||||
export {defineKestraHeyConfig} from "./config";
|
||||
export type {KestraSdkPlugin} from "./types";
|
||||
@@ -1,169 +0,0 @@
|
||||
import {$} from "@hey-api/openapi-ts";
|
||||
import type {KestraSdkPlugin} from "./types";
|
||||
|
||||
|
||||
export const handler: KestraSdkPlugin["Handler"] = ({plugin}) => {
|
||||
const useRouteSymbol = plugin.symbol(
|
||||
"useRoute",
|
||||
{
|
||||
external: "vue-router"
|
||||
});
|
||||
|
||||
const addTenantToParametersSymbol = plugin.symbol("addTenantToParameters",{
|
||||
getFilePath: () => "sdk/ks-shared",
|
||||
});
|
||||
|
||||
const functionNode = $.func().generic("TParams")
|
||||
.params(
|
||||
$.param("parameters").type($.type("TParams"))
|
||||
).returns($.type.and($.type("TParams"), $.type.object().prop("tenant", (p) => p.type("string"))))
|
||||
.do(
|
||||
// const tenant = useRouter().params.tenant
|
||||
$.const("tenant").assign(
|
||||
$(useRouteSymbol).call().attr("params").attr("tenant").optional().as($.type("string"))
|
||||
),
|
||||
$.return($.object()
|
||||
.spread($.id("parameters"))
|
||||
.prop("tenant", "tenant")
|
||||
)
|
||||
)
|
||||
|
||||
const exportedFunctionNode = $.const(addTenantToParametersSymbol).export().assign(functionNode);
|
||||
plugin.node(exportedFunctionNode);
|
||||
|
||||
const operationsDict: Record<string, {symbol:ReturnType<typeof plugin.symbol>, methodName: string}[]> = {}
|
||||
|
||||
plugin.forEach(
|
||||
"operation",
|
||||
({operation}) => {
|
||||
// on each operation, create a method that executes the operation from the sdk
|
||||
const methodName = plugin.config.methodNameBuilder?.(operation);
|
||||
if (!methodName) {
|
||||
return;
|
||||
}
|
||||
|
||||
const pathParams = operation.parameters?.path || {};
|
||||
|
||||
const sym = plugin.querySymbol({
|
||||
category: "sdk",
|
||||
resource: "operation",
|
||||
resourceId: operation.id,
|
||||
})
|
||||
|
||||
if(!sym) {
|
||||
return;
|
||||
}
|
||||
|
||||
const originalOperationSymbol = $(sym);
|
||||
|
||||
const funcSymbol = plugin.symbol(methodName, {
|
||||
getFilePath: () => `sdk/ks-${operation.tags?.[0] ?? "default"}`,
|
||||
})
|
||||
|
||||
if (!operationsDict[operation.tags?.[0] ?? "default"]) {
|
||||
operationsDict[operation.tags?.[0] ?? "default"] = [];
|
||||
}
|
||||
operationsDict[operation.tags?.[0] ?? "default"].push({symbol:funcSymbol, methodName});
|
||||
|
||||
if(!pathParams || !("tenant" in pathParams)) {
|
||||
// if there is no path parameter named "tenant",
|
||||
// we export this method as is
|
||||
plugin.node(
|
||||
$.const(funcSymbol)
|
||||
.assign(originalOperationSymbol)
|
||||
.export()
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
const optionsId = "options"
|
||||
|
||||
// find a cleaner way to do that (expose parameters symbol from operation ?)
|
||||
const parametersWithoutTenant = sym.node?.value._params[0]._type._exprInput["~ref"].props.filter((p: any) => p.name !== "tenant") as any
|
||||
|
||||
if(parametersWithoutTenant.length === 0) {
|
||||
|
||||
// if the only path parameter is "tenant", we can simplify the function
|
||||
const functionNode = $.func()
|
||||
.params($.param(optionsId)
|
||||
.required(false)
|
||||
.type(
|
||||
$.type("Parameters")
|
||||
.generic($.type.query(originalOperationSymbol))
|
||||
.idx(1)
|
||||
)
|
||||
)
|
||||
.do(
|
||||
$.return(originalOperationSymbol.call(
|
||||
$(addTenantToParametersSymbol).call($.object()),
|
||||
optionsId,
|
||||
))
|
||||
)
|
||||
|
||||
const exportedFunctionNode = $.const(funcSymbol).export().assign(functionNode);
|
||||
|
||||
plugin.node(exportedFunctionNode);
|
||||
return;
|
||||
}
|
||||
|
||||
const isTenantOnlyRequiredParam = Object.values(pathParams).filter(p => p.name !== "tenant" && p.required).length === 0;
|
||||
|
||||
const parameterObj = $.type.object()
|
||||
|
||||
for (const param in parametersWithoutTenant) {
|
||||
const paramDef = parametersWithoutTenant[param];
|
||||
parameterObj.prop(paramDef.name, (p) => p.required(!paramDef._optional).type(paramDef._type["~ref"]));
|
||||
}
|
||||
|
||||
const paramId = "parameters"
|
||||
const functionNode = $.func()
|
||||
.params(
|
||||
$.param(paramId)
|
||||
.required(!isTenantOnlyRequiredParam)
|
||||
.type(parameterObj)
|
||||
,
|
||||
$.param(optionsId)
|
||||
.required(false)
|
||||
.type(
|
||||
$.type("Parameters")
|
||||
.generic($.type.query(originalOperationSymbol))
|
||||
.idx(1)
|
||||
)
|
||||
)
|
||||
.do(
|
||||
isTenantOnlyRequiredParam ?
|
||||
$.return(originalOperationSymbol.call(
|
||||
$(addTenantToParametersSymbol).call($(paramId)),
|
||||
optionsId,
|
||||
))
|
||||
: $.return(originalOperationSymbol.call(
|
||||
$(addTenantToParametersSymbol).call(paramId),
|
||||
optionsId,
|
||||
))
|
||||
)
|
||||
|
||||
const exportedFunctionNode = $.const(funcSymbol).export().assign(functionNode);
|
||||
|
||||
plugin.node(exportedFunctionNode);
|
||||
},
|
||||
{
|
||||
order: "declarations",
|
||||
},
|
||||
);
|
||||
|
||||
for (const tag in operationsDict) {
|
||||
const operations = operationsDict[tag];
|
||||
const symbol = plugin.symbol(tag, {
|
||||
getFilePath: () => "ks-sdk",
|
||||
});
|
||||
|
||||
plugin.node(
|
||||
$.const(symbol)
|
||||
.export()
|
||||
.assign($.object().props(...operations.map(op => $.prop({
|
||||
kind: "prop",
|
||||
name: op.methodName
|
||||
}).value(op.symbol))))
|
||||
);
|
||||
}
|
||||
};
|
||||
21
ui/heyapi-sdk-plugin/types.d.ts
vendored
21
ui/heyapi-sdk-plugin/types.d.ts
vendored
@@ -1,21 +0,0 @@
|
||||
import type {DefinePlugin} from "@hey-api/openapi-ts";
|
||||
|
||||
export type UserConfig = {
|
||||
/**
|
||||
* Plugin name. Must be unique.
|
||||
*/
|
||||
name: "ks-sdk";
|
||||
/**
|
||||
* Name of the generated file.
|
||||
*
|
||||
* @default 'ks-sdk'
|
||||
*/
|
||||
output?: string;
|
||||
/**
|
||||
* Function to build method names from operations.
|
||||
* Receives the operation object and must return a string or undefined to skip the operation.
|
||||
*/
|
||||
methodNameBuilder?: (operation: any) => string;
|
||||
};
|
||||
|
||||
export type KestraSdkPlugin = DefinePlugin<UserConfig>;
|
||||
@@ -1,35 +0,0 @@
|
||||
import {defineConfig} from "@hey-api/openapi-ts";
|
||||
import {defineKestraHeyConfig} from "./heyapi-sdk-plugin";
|
||||
|
||||
const generateHash = (str: string) => {
|
||||
let hash = 0;
|
||||
for (const char of str) {
|
||||
hash = (hash << 5) - hash + char.charCodeAt(0);
|
||||
hash |= 0; // Constrain to 32bit integer
|
||||
}
|
||||
return hash.toString(16).replace("-", "0");
|
||||
};
|
||||
|
||||
export default defineConfig({
|
||||
input: "../webserver/build/classes/java/main/META-INF/swagger/kestra.yml",
|
||||
output: {
|
||||
path: "./src/generated/kestra-api",
|
||||
lint: "eslint"
|
||||
},
|
||||
|
||||
plugins: [
|
||||
{
|
||||
name: "@hey-api/client-axios",
|
||||
},
|
||||
{
|
||||
name: "@hey-api/sdk",
|
||||
paramsStructure: "flat",
|
||||
methodNameBuilder(operation) {
|
||||
return `__${generateHash(operation.id)}__`
|
||||
}
|
||||
},
|
||||
defineKestraHeyConfig({
|
||||
output: "./src/generated/kestra-heyapi-sdk",
|
||||
})
|
||||
],
|
||||
});
|
||||
3395
ui/package-lock.json
generated
3395
ui/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -20,15 +20,14 @@
|
||||
"storybook": "storybook dev -p 6006",
|
||||
"build-storybook": "storybook build",
|
||||
"prepare": "cd .. && husky ui/.husky && rimraf .git/hooks",
|
||||
"generate:openapi": "openapi-ts",
|
||||
"postinstall": "patch-package && openapi-ts"
|
||||
"postinstall": "patch-package"
|
||||
},
|
||||
"dependencies": {
|
||||
"@js-joda/core": "^5.6.5",
|
||||
"@kestra-io/ui-libs": "^0.0.268",
|
||||
"@vue-flow/background": "^1.3.2",
|
||||
"@vue-flow/controls": "^1.1.2",
|
||||
"@vue-flow/core": "^1.48.0",
|
||||
"@vue-flow/core": "^1.48.1",
|
||||
"@vueuse/core": "^14.1.0",
|
||||
"ansi-to-html": "^0.7.2",
|
||||
"axios": "^1.13.2",
|
||||
@@ -40,7 +39,7 @@
|
||||
"cytoscape": "^3.33.0",
|
||||
"dagre": "^0.8.5",
|
||||
"dotenv": "^17.2.3",
|
||||
"element-plus": "2.12.0",
|
||||
"element-plus": "2.13.0",
|
||||
"humanize-duration": "^3.33.2",
|
||||
"js-yaml": "^4.1.1",
|
||||
"lodash": "^4.17.21",
|
||||
@@ -60,15 +59,15 @@
|
||||
"path-browserify": "^1.0.1",
|
||||
"pdfjs-dist": "^5.4.449",
|
||||
"pinia": "^3.0.4",
|
||||
"posthog-js": "^1.308.0",
|
||||
"posthog-js": "^1.310.1",
|
||||
"rapidoc": "^9.3.8",
|
||||
"semver": "^7.7.3",
|
||||
"shiki": "^3.20.0",
|
||||
"vue": "^3.5.25",
|
||||
"vue": "^3.5.26",
|
||||
"vue-axios": "^3.5.2",
|
||||
"vue-chartjs": "^5.3.3",
|
||||
"vue-gtag": "^3.6.3",
|
||||
"vue-i18n": "^11.2.2",
|
||||
"vue-i18n": "^11.2.7",
|
||||
"vue-material-design-icons": "^5.3.1",
|
||||
"vue-router": "^4.6.4",
|
||||
"vue-sidebar-menu": "^5.9.1",
|
||||
@@ -82,7 +81,6 @@
|
||||
"@codecov/vite-plugin": "^1.9.1",
|
||||
"@esbuild-plugins/node-modules-polyfill": "^0.2.2",
|
||||
"@eslint/js": "^9.39.2",
|
||||
"@hey-api/openapi-ts": "^0.89.2",
|
||||
"@playwright/test": "^1.57.0",
|
||||
"@rushstack/eslint-patch": "^1.14.1",
|
||||
"@shikijs/markdown-it": "^3.20.0",
|
||||
@@ -99,9 +97,9 @@
|
||||
"@types/semver": "^7.7.1",
|
||||
"@types/testing-library__jest-dom": "^6.0.0",
|
||||
"@types/testing-library__user-event": "^4.2.0",
|
||||
"@typescript-eslint/parser": "^8.50.0",
|
||||
"@typescript-eslint/parser": "^8.50.1",
|
||||
"@vitejs/plugin-vue": "^6.0.3",
|
||||
"@vitejs/plugin-vue-jsx": "^5.1.2",
|
||||
"@vitejs/plugin-vue-jsx": "^5.1.3",
|
||||
"@vitest/browser": "^3.2.4",
|
||||
"@vitest/coverage-v8": "^3.2.4",
|
||||
"@vue/eslint-config-prettier": "^10.2.0",
|
||||
@@ -122,29 +120,29 @@
|
||||
"playwright": "^1.55.0",
|
||||
"prettier": "^3.7.4",
|
||||
"rimraf": "^6.1.2",
|
||||
"rolldown-vite": "^7.2.11",
|
||||
"rolldown-vite": "^7.3.0",
|
||||
"rollup-plugin-copy": "^3.5.0",
|
||||
"sass": "^1.97.0",
|
||||
"storybook": "^9.1.16",
|
||||
"sass": "^1.97.1",
|
||||
"storybook": "^9.1.17",
|
||||
"storybook-vue3-router": "^6.0.2",
|
||||
"ts-node": "^10.9.2",
|
||||
"typescript": "^5.9.3",
|
||||
"typescript-eslint": "^8.50.0",
|
||||
"typescript-eslint": "^8.50.1",
|
||||
"uuid": "^13.0.0",
|
||||
"vite": "npm:rolldown-vite@latest",
|
||||
"vitest": "^3.2.4",
|
||||
"vue-tsc": "^3.1.8"
|
||||
"vue-tsc": "^3.2.1"
|
||||
},
|
||||
"optionalDependencies": {
|
||||
"@esbuild/darwin-arm64": "^0.27.2",
|
||||
"@esbuild/darwin-x64": "^0.27.2",
|
||||
"@esbuild/linux-x64": "^0.27.2",
|
||||
"@rollup/rollup-darwin-arm64": "^4.53.5",
|
||||
"@rollup/rollup-darwin-x64": "^4.53.5",
|
||||
"@rollup/rollup-linux-x64-gnu": "^4.53.5",
|
||||
"@swc/core-darwin-arm64": "^1.15.5",
|
||||
"@swc/core-darwin-x64": "^1.15.5",
|
||||
"@swc/core-linux-x64-gnu": "^1.15.5"
|
||||
"@rollup/rollup-darwin-arm64": "^4.54.0",
|
||||
"@rollup/rollup-darwin-x64": "^4.54.0",
|
||||
"@rollup/rollup-linux-x64-gnu": "^4.54.0",
|
||||
"@swc/core-darwin-arm64": "^1.15.7",
|
||||
"@swc/core-darwin-x64": "^1.15.7",
|
||||
"@swc/core-linux-x64-gnu": "^1.15.7"
|
||||
},
|
||||
"overrides": {
|
||||
"bootstrap": {
|
||||
|
||||
@@ -54,12 +54,11 @@
|
||||
import {useI18n} from "vue-i18n";
|
||||
import TopNavBar from "../layout/TopNavBar.vue";
|
||||
import useRouteContext from "../../composables/useRouteContext";
|
||||
import {useAxios, useSDK} from "../../utils/axios";
|
||||
import {useAxios} from "../../utils/axios";
|
||||
import IconEdit from "vue-material-design-icons/Pencil.vue";
|
||||
import {apiUrlWithoutTenants} from "override/utils/route";
|
||||
import {apiUrl, apiUrlWithoutTenants} from "override/utils/route";
|
||||
import DataTable from "../layout/DataTable.vue";
|
||||
import NoData from "../layout/NoData.vue";
|
||||
import {PagedResultsConcurrencyLimit} from "../../generated/kestra-api";
|
||||
|
||||
const {t} = useI18n();
|
||||
|
||||
@@ -79,13 +78,15 @@
|
||||
const KEYS: (keyof ConcurrencyLimit)[] = ["tenantId", "namespace", "flowId", "running"];
|
||||
|
||||
const axios = useAxios();
|
||||
const sdk = useSDK();
|
||||
const data = ref<PagedResultsConcurrencyLimit>();
|
||||
const data = ref<{
|
||||
total: number;
|
||||
results: ConcurrencyLimit[]
|
||||
}>();
|
||||
|
||||
async function loadData(){
|
||||
const response = await sdk.Executions.searchConcurrencyLimits();
|
||||
if(response?.status !== 200 && response?.error){
|
||||
throw new Error(`Failed to load concurrency limits: ${response.error ?? "unknown error"}`);
|
||||
const response = await axios.get(`${apiUrl()}/concurrency-limit/search`);
|
||||
if(response?.status !== 200){
|
||||
throw new Error(`Failed to load concurrency limits: ${response?.statusText}`);
|
||||
}
|
||||
data.value = response.data;
|
||||
}
|
||||
|
||||
@@ -43,7 +43,7 @@
|
||||
REF_PATH_INJECTION_KEY,
|
||||
ROOT_SCHEMA_INJECTION_KEY,
|
||||
SCHEMA_DEFINITIONS_INJECTION_KEY,
|
||||
UPDATE_TASK_FUNCTION_INJECTION_KEY
|
||||
UPDATE_YAML_FUNCTION_INJECTION_KEY
|
||||
} from "../../no-code/injectionKeys";
|
||||
import {NoCodeProps} from "../../flows/noCodeTypes";
|
||||
import {deepEqual} from "../../../utils/utils";
|
||||
@@ -68,7 +68,7 @@
|
||||
dashboardStore.sourceCode = YAML_UTILS.stringify(app);
|
||||
}
|
||||
|
||||
provide(UPDATE_TASK_FUNCTION_INJECTION_KEY, (yaml) => {
|
||||
provide(UPDATE_YAML_FUNCTION_INJECTION_KEY, (yaml) => {
|
||||
editorUpdate(yaml)
|
||||
})
|
||||
|
||||
|
||||
@@ -1,16 +1,13 @@
|
||||
<template>
|
||||
<el-tooltip placement="bottom" :content="t('playground.tooltip_persistence')">
|
||||
<el-switch v-model="playgroundStore.enabled" :activeText="t('playground.toggle')" class="toggle" :class="{'is-active': playgroundStore.enabled}" />
|
||||
<el-tooltip placement="bottom" :content="$t('playground.tooltip_persistence')">
|
||||
<el-switch v-model="playgroundStore.enabled" :activeText="$t('playground.toggle')" class="toggle" :class="{'is-active': playgroundStore.enabled}" />
|
||||
</el-tooltip>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import {useI18n} from "vue-i18n";
|
||||
import {usePlaygroundStore} from "../../stores/playground";
|
||||
|
||||
const {t} = useI18n();
|
||||
const playgroundStore = usePlaygroundStore();
|
||||
|
||||
</script>
|
||||
|
||||
<style scoped lang="scss">
|
||||
|
||||
@@ -6,7 +6,7 @@
|
||||
<Keyboard />
|
||||
</el-icon>
|
||||
<span class="fs-6">
|
||||
{{ t("editor_shortcuts.label") }}
|
||||
{{ $t("editor_shortcuts.label") }}
|
||||
</span>
|
||||
</div>
|
||||
</template>
|
||||
@@ -27,7 +27,7 @@
|
||||
</template>
|
||||
</div>
|
||||
<div class="text-break">
|
||||
{{ t(command.description) }}
|
||||
{{ $t(command.description) }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -35,11 +35,9 @@
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import {useI18n} from "vue-i18n";
|
||||
import Keyboard from "vue-material-design-icons/Keyboard.vue";
|
||||
import {useKeyShortcuts} from "../../utils/useKeyShortcuts";
|
||||
|
||||
const {t} = useI18n();
|
||||
const {isKeyShortcutsDialogShown} = useKeyShortcuts();
|
||||
|
||||
const commands = [
|
||||
|
||||
@@ -97,7 +97,6 @@
|
||||
<script setup lang="ts">
|
||||
import {
|
||||
computed,
|
||||
getCurrentInstance,
|
||||
h,
|
||||
inject,
|
||||
onBeforeUnmount,
|
||||
@@ -126,7 +125,7 @@
|
||||
import uniqBy from "lodash/uniqBy";
|
||||
import {useI18n} from "vue-i18n";
|
||||
import {ElDatePicker} from "element-plus";
|
||||
import {Moment} from "moment";
|
||||
import moment, {Moment} from "moment";
|
||||
import PlaceholderContentWidget from "../../composables/monaco/PlaceholderContentWidget";
|
||||
import Utils from "../../utils/utils";
|
||||
import {hashCode} from "../../utils/global";
|
||||
@@ -137,7 +136,6 @@
|
||||
import EditorType = editor.EditorType;
|
||||
import {useRoute} from "vue-router";
|
||||
|
||||
const currentInstance = getCurrentInstance()!;
|
||||
const {t} = useI18n();
|
||||
|
||||
const textAreaValue = computed({
|
||||
@@ -371,8 +369,7 @@
|
||||
}
|
||||
}, {deep: true});
|
||||
|
||||
const nowMoment: Moment = currentInstance.appContext.config.globalProperties.$moment().startOf("day");
|
||||
|
||||
const nowMoment: Moment = moment().startOf("day");
|
||||
function addedSuggestRows(mutations: MutationRecord[]) {
|
||||
return mutations.flatMap(({addedNodes}) => {
|
||||
const nodes = [...addedNodes];
|
||||
@@ -461,7 +458,7 @@
|
||||
endColumn: wordAtPosition?.endColumn ?? position?.column
|
||||
},
|
||||
// We don't use the selectedDate directly because if user modifies the input value directly it doesn't work otherwise
|
||||
text: `${currentInstance.appContext.config.globalProperties.$moment(
|
||||
text: `${moment(
|
||||
datePicker.value!.$el.nextElementSibling.querySelector("input").value
|
||||
).toISOString(true)} `,
|
||||
forceMoveMarkers: true
|
||||
|
||||
@@ -1,11 +1,11 @@
|
||||
<template>
|
||||
<nav class="d-flex align-items-center w-100 gap-3 top-bar">
|
||||
<SidebarToggleButton
|
||||
v-if="layoutStore.sideMenuCollapsed"
|
||||
@toggle="layoutStore.setSideMenuCollapsed(false)"
|
||||
/>
|
||||
<div class="d-flex flex-column flex-grow-1 flex-shrink-1 overflow-hidden top-title">
|
||||
<div class="d-flex align-items-end gap-2">
|
||||
<SidebarToggleButton
|
||||
v-if="layoutStore.sideMenuCollapsed"
|
||||
@toggle="layoutStore.setSideMenuCollapsed(false)"
|
||||
/>
|
||||
<div class="d-flex flex-column gap-2">
|
||||
<el-breadcrumb v-if="breadcrumb">
|
||||
<el-breadcrumb-item v-for="(item, x) in breadcrumb" :key="x" :class="{'pe-none': item.disabled}">
|
||||
|
||||
@@ -43,7 +43,9 @@
|
||||
BLOCK_SCHEMA_PATH_INJECTION_KEY,
|
||||
CLOSE_TASK_FUNCTION_INJECTION_KEY,
|
||||
CREATE_TASK_FUNCTION_INJECTION_KEY,
|
||||
CREATING_FLOW_INJECTION_KEY,
|
||||
CREATING_TASK_INJECTION_KEY,
|
||||
DEFAULT_NAMESPACE_INJECTION_KEY,
|
||||
EDIT_TASK_FUNCTION_INJECTION_KEY,
|
||||
EDITING_TASK_INJECTION_KEY,
|
||||
FIELDNAME_INJECTION_KEY,
|
||||
@@ -55,7 +57,7 @@
|
||||
REF_PATH_INJECTION_KEY,
|
||||
ROOT_SCHEMA_INJECTION_KEY,
|
||||
SCHEMA_DEFINITIONS_INJECTION_KEY,
|
||||
UPDATE_TASK_FUNCTION_INJECTION_KEY,
|
||||
UPDATE_YAML_FUNCTION_INJECTION_KEY,
|
||||
} from "./injectionKeys";
|
||||
import {useFlowFields, SECTIONS_IDS} from "./utils/useFlowFields";
|
||||
import debounce from "lodash/debounce";
|
||||
@@ -65,6 +67,7 @@
|
||||
import {useKeyboardSave} from "./utils/useKeyboardSave";
|
||||
import {deepEqual} from "../../utils/utils";
|
||||
import {useScrollMemory} from "../../composables/useScrollMemory";
|
||||
import {defaultNamespace} from "../../composables/useNamespaces";
|
||||
|
||||
|
||||
const props = defineProps<NoCodeProps>();
|
||||
@@ -166,6 +169,8 @@
|
||||
provide(REF_PATH_INJECTION_KEY, props.refPath);
|
||||
provide(PANEL_INJECTION_KEY, panel)
|
||||
provide(POSITION_INJECTION_KEY, props.position ?? "after");
|
||||
provide(CREATING_FLOW_INJECTION_KEY, flowStore.isCreating ?? false);
|
||||
provide(DEFAULT_NAMESPACE_INJECTION_KEY, computed(() => flowStore.flow?.namespace ?? defaultNamespace() ?? "company.team"));
|
||||
provide(CREATING_TASK_INJECTION_KEY, props.creatingTask);
|
||||
provide(EDITING_TASK_INJECTION_KEY, props.editingTask);
|
||||
provide(FIELDNAME_INJECTION_KEY, props.fieldName);
|
||||
@@ -184,7 +189,7 @@
|
||||
emit("closeTask")
|
||||
})
|
||||
|
||||
provide(UPDATE_TASK_FUNCTION_INJECTION_KEY, (yaml) => {
|
||||
provide(UPDATE_YAML_FUNCTION_INJECTION_KEY, (yaml) => {
|
||||
editorUpdate(yaml)
|
||||
})
|
||||
|
||||
|
||||
@@ -44,7 +44,6 @@
|
||||
<script setup lang="ts">
|
||||
import {computed, inject, ref} from "vue";
|
||||
import {BLOCK_SCHEMA_PATH_INJECTION_KEY} from "../../injectionKeys";
|
||||
import {useFlowStore} from "../../../../stores/flow";
|
||||
import Creation from "./taskList/buttons/Creation.vue";
|
||||
import Element from "./taskList/Element.vue";
|
||||
import * as YAML_UTILS from "@kestra-io/ui-libs/flow-yaml-utils";
|
||||
@@ -53,7 +52,7 @@
|
||||
|
||||
import {
|
||||
CREATING_TASK_INJECTION_KEY, FULL_SCHEMA_INJECTION_KEY, FULL_SOURCE_INJECTION_KEY,
|
||||
PARENT_PATH_INJECTION_KEY, REF_PATH_INJECTION_KEY,
|
||||
PARENT_PATH_INJECTION_KEY, REF_PATH_INJECTION_KEY, UPDATE_YAML_FUNCTION_INJECTION_KEY,
|
||||
} from "../../injectionKeys";
|
||||
import {SECTIONS_MAP} from "../../../../utils/constants";
|
||||
import {getValueAtJsonPath} from "../../../../utils/utils";
|
||||
@@ -83,8 +82,6 @@
|
||||
inheritAttrs: false
|
||||
});
|
||||
|
||||
const flowStore = useFlowStore();
|
||||
|
||||
interface Task {
|
||||
id:string,
|
||||
type:string
|
||||
@@ -150,6 +147,8 @@
|
||||
|
||||
const movedIndex = ref(-1);
|
||||
|
||||
const updateYaml = inject(UPDATE_YAML_FUNCTION_INJECTION_KEY, () => {});
|
||||
|
||||
const moveElement = (
|
||||
items: Record<string, any>[] | undefined,
|
||||
elementID: string,
|
||||
@@ -171,7 +170,7 @@
|
||||
movedIndex.value = -1;
|
||||
}, 200);
|
||||
|
||||
flowStore.flowYaml =
|
||||
updateYaml(
|
||||
YAML_UTILS.swapBlocks({
|
||||
source:flow.value,
|
||||
section: SECTIONS_MAP[section.value.toLowerCase() as keyof typeof SECTIONS_MAP],
|
||||
@@ -179,6 +178,7 @@
|
||||
key2:items[newIndex][keyName],
|
||||
keyName,
|
||||
})
|
||||
);
|
||||
};
|
||||
|
||||
const fullSchema = inject(FULL_SCHEMA_INJECTION_KEY, ref<Record<string, any>>({}));
|
||||
|
||||
@@ -8,18 +8,17 @@
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import {computed, onMounted} from "vue";
|
||||
import {useFlowStore} from "../../../../stores/flow";
|
||||
import {onMounted, inject, computed, provide} from "vue";
|
||||
import NamespaceSelect from "../../../namespaces/components/NamespaceSelect.vue";
|
||||
import {CREATING_FLOW_INJECTION_KEY, DEFAULT_NAMESPACE_INJECTION_KEY} from "../../injectionKeys";
|
||||
|
||||
const modelValue = defineModel<string>();
|
||||
|
||||
const flowStore = useFlowStore();
|
||||
|
||||
const isCreating = computed(() => flowStore.isCreating);
|
||||
const isCreating = inject(CREATING_FLOW_INJECTION_KEY, false);
|
||||
const defaultNamespace = inject(DEFAULT_NAMESPACE_INJECTION_KEY, computed(() => ""));
|
||||
provide(DEFAULT_NAMESPACE_INJECTION_KEY, computed(() => modelValue.value || defaultNamespace.value));
|
||||
|
||||
onMounted(() => {
|
||||
const flowNamespace = flowStore.flow?.namespace;
|
||||
const flowNamespace = defaultNamespace.value;
|
||||
if (!modelValue.value && flowNamespace) {
|
||||
modelValue.value = flowNamespace;
|
||||
}
|
||||
|
||||
@@ -24,6 +24,7 @@ export const POSITION_INJECTION_KEY = Symbol("position-injection-key") as Inject
|
||||
* NOTE: different from the `isCreating` flag coming from the store. `isCreating` refers to the Complete flow being in creation
|
||||
*/
|
||||
export const CREATING_TASK_INJECTION_KEY = Symbol("creating-injection-key") as InjectionKey<boolean>
|
||||
export const CREATING_FLOW_INJECTION_KEY = Symbol("creating-flow-injection-key") as InjectionKey<boolean>
|
||||
/**
|
||||
* When creating anew task, allows to specify a field where the new task should be injected.
|
||||
* @example
|
||||
@@ -51,9 +52,9 @@ export const EDIT_TASK_FUNCTION_INJECTION_KEY = Symbol("edit-function-injection-
|
||||
*/
|
||||
export const CLOSE_TASK_FUNCTION_INJECTION_KEY = Symbol("close-function-injection-key") as InjectionKey<() => void>
|
||||
/**
|
||||
* We call this function when a task is changed, as soon as the first click or type is done
|
||||
* Call this function to update the full Yaml content
|
||||
*/
|
||||
export const UPDATE_TASK_FUNCTION_INJECTION_KEY = Symbol("update-function-injection-key") as InjectionKey<(yaml: string) => void>
|
||||
export const UPDATE_YAML_FUNCTION_INJECTION_KEY = Symbol("update-function-injection-key") as InjectionKey<(yaml: string) => void>
|
||||
/**
|
||||
* Set this to override the contents of the no-code editor with a component of your choice
|
||||
* This is used to display the metadata edition inputs
|
||||
@@ -92,4 +93,6 @@ export const SCHEMA_DEFINITIONS_INJECTION_KEY = Symbol("schema-definitions-injec
|
||||
|
||||
export const DATA_TYPES_MAP_INJECTION_KEY = Symbol("data-types-injection-key") as InjectionKey<ComputedRef<Record<string, string[] | undefined>>>
|
||||
|
||||
export const ON_TASK_EDITOR_CLICK_INJECTION_KEY = Symbol("on-task-editor-click-injection-key") as InjectionKey<(elt?: Partial<NoCodeElement>) => void>;
|
||||
export const ON_TASK_EDITOR_CLICK_INJECTION_KEY = Symbol("on-task-editor-click-injection-key") as InjectionKey<(elt?: Partial<NoCodeElement>) => void>;
|
||||
|
||||
export const DEFAULT_NAMESPACE_INJECTION_KEY = Symbol("default-namespace-injection-key") as InjectionKey<ComputedRef<string>>;
|
||||
@@ -16,7 +16,7 @@
|
||||
import {PLUGIN_DEFAULTS_SECTION, SECTIONS_MAP} from "../../../utils/constants";
|
||||
import {
|
||||
CLOSE_TASK_FUNCTION_INJECTION_KEY,
|
||||
UPDATE_TASK_FUNCTION_INJECTION_KEY,
|
||||
UPDATE_YAML_FUNCTION_INJECTION_KEY,
|
||||
FULL_SOURCE_INJECTION_KEY, CREATING_TASK_INJECTION_KEY,
|
||||
PARENT_PATH_INJECTION_KEY, POSITION_INJECTION_KEY,
|
||||
REF_PATH_INJECTION_KEY, EDIT_TASK_FUNCTION_INJECTION_KEY,
|
||||
@@ -37,7 +37,7 @@
|
||||
|
||||
const fieldName = inject(FIELDNAME_INJECTION_KEY, undefined);
|
||||
const blockSchemaPath = inject(BLOCK_SCHEMA_PATH_INJECTION_KEY, ref(""));
|
||||
const updateTask = inject(UPDATE_TASK_FUNCTION_INJECTION_KEY, () => {})
|
||||
const updateTask = inject(UPDATE_YAML_FUNCTION_INJECTION_KEY, () => {})
|
||||
|
||||
const closeTaskAddition = inject(
|
||||
CLOSE_TASK_FUNCTION_INJECTION_KEY,
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
<template #additional-right>
|
||||
<ul>
|
||||
<li>
|
||||
<el-button v-if="canCreate" tag="router-link" :to="{name: 'flows/create', query: {namespace: $route.query.namespace}}" :icon="Plus" type="primary">
|
||||
<el-button v-if="canCreate" tag="router-link" :to="{name: 'flows/create', query: {namespace: $route.query.namespace}}" :icon="Plus" type="secondary">
|
||||
{{ $t('create_flow') }}
|
||||
</el-button>
|
||||
</li>
|
||||
@@ -30,7 +30,7 @@
|
||||
<el-button
|
||||
v-if="isOSS"
|
||||
@click="startTour"
|
||||
:icon="Plus"
|
||||
:icon="Compass"
|
||||
size="large"
|
||||
type="primary"
|
||||
class="px-3 p-4 section-1-link product-link"
|
||||
@@ -39,7 +39,7 @@
|
||||
</el-button>
|
||||
<el-button
|
||||
v-else
|
||||
:icon="Plus"
|
||||
:icon="Compass"
|
||||
tag="router-link"
|
||||
:to="{name: 'flows/create'}"
|
||||
size="large"
|
||||
@@ -74,6 +74,7 @@
|
||||
import {useCoreStore} from "../../stores/core";
|
||||
import {useI18n} from "vue-i18n";
|
||||
import Plus from "vue-material-design-icons/Plus.vue";
|
||||
import Compass from "vue-material-design-icons/Compass.vue";
|
||||
import Play from "vue-material-design-icons/Play.vue";
|
||||
import OnboardingBottom from "override/components/OnboardingBottom.vue";
|
||||
import kestraWelcome from "../../assets/onboarding/kestra_welcome.svg";
|
||||
|
||||
@@ -156,7 +156,7 @@
|
||||
<el-form-item :label="$t('secret.key')" prop="key">
|
||||
<el-input v-model="secret.key" :disabled="secret.update" required />
|
||||
</el-form-item>
|
||||
<el-form-item v-if="!secret.update" :label="$t('secret.name')" prop="value">
|
||||
<el-form-item v-if="!secret.update" :label="$t('secret.name')" prop="value" required>
|
||||
<MultilineSecret v-model="secret.value" :placeholder="secretModalTitle" />
|
||||
</el-form-item>
|
||||
<el-form-item v-if="secret.update" :label="$t('secret.name')" prop="value">
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
import {computed, watch} from "vue";
|
||||
import {useI18n} from "vue-i18n";
|
||||
import {configureMonacoYaml} from "monaco-yaml";
|
||||
import * as monaco from "monaco-editor/esm/vs/editor/editor.api";
|
||||
@@ -21,6 +22,7 @@ import {
|
||||
registerPebbleAutocompletion
|
||||
} from "./pebbleLanguageConfigurator";
|
||||
import {usePluginsStore} from "../../../stores/plugins";
|
||||
import {useBlueprintsStore} from "../../../stores/blueprints";
|
||||
import {languages} from "monaco-editor/esm/vs/editor/editor.api";
|
||||
import CompletionItem = languages.CompletionItem;
|
||||
|
||||
@@ -34,11 +36,14 @@ export class YamlLanguageConfigurator extends AbstractLanguageConfigurator {
|
||||
}
|
||||
|
||||
async configureLanguage(pluginsStore: ReturnType<typeof usePluginsStore>) {
|
||||
const validateYAML = computed(() => useBlueprintsStore().validateYAML);
|
||||
watch(validateYAML, (shouldValidate) => configureMonacoYaml(monaco, {validate: shouldValidate}));
|
||||
|
||||
configureMonacoYaml(monaco, {
|
||||
enableSchemaRequest: true,
|
||||
hover: localStorage.getItem("hoverTextEditor") === "true",
|
||||
completion: true,
|
||||
validate: true,
|
||||
validate: validateYAML.value ?? true,
|
||||
format: true,
|
||||
schemas: yamlSchemas()
|
||||
});
|
||||
|
||||
@@ -1,5 +1,11 @@
|
||||
<template>
|
||||
<SideBar v-if="menu" :menu :showLink="showLink" @menu-collapse="onCollapse">
|
||||
<SideBar
|
||||
v-if="menu"
|
||||
:menu
|
||||
:showLink
|
||||
@menu-collapse="onCollapse"
|
||||
:class="{overlay: verticalLayout}"
|
||||
>
|
||||
<template #footer>
|
||||
<Auth />
|
||||
</template>
|
||||
@@ -11,6 +17,9 @@
|
||||
import SideBar from "../../components/layout/SideBar.vue";
|
||||
import Auth from "../../override/components/auth/Auth.vue";
|
||||
|
||||
import {useBreakpoints, breakpointsElement} from "@vueuse/core";
|
||||
const verticalLayout = useBreakpoints(breakpointsElement).smallerOrEqual("sm");
|
||||
|
||||
withDefaults(defineProps<{
|
||||
showLink?: boolean
|
||||
}>(), {
|
||||
|
||||
@@ -56,7 +56,10 @@
|
||||
<div v-if="!system && blueprint.tags?.length > 0" class="tags-section">
|
||||
<span v-for="tag in processedTags(blueprint.tags)" :key="tag.original" class="tag-item">{{ tag.display }}</span>
|
||||
</div>
|
||||
<div class="text-section">
|
||||
<div v-if="blueprint.template" class="tags-section">
|
||||
<span class="tag-item">{{ $t('template') }}</span>
|
||||
</div>
|
||||
<div class="text-section">
|
||||
<h3 class="title">
|
||||
{{ blueprint.title ?? blueprint.id }}
|
||||
</h3>
|
||||
@@ -151,6 +154,7 @@
|
||||
id: string;
|
||||
tags: string[];
|
||||
title?: string;
|
||||
template?: Record<string, any>;
|
||||
}[] | undefined>(undefined);
|
||||
const error = ref(false);
|
||||
const icon = {ContentCopy};
|
||||
|
||||
@@ -1,15 +1,14 @@
|
||||
import axios from "axios";
|
||||
import {defineStore} from "pinia";
|
||||
import {AI} from "../generated/kestra-api/ks-sdk.gen";
|
||||
import {apiUrl} from "override/utils/route";
|
||||
|
||||
export const useAiStore = defineStore("ai", {
|
||||
actions: {
|
||||
async generateFlow({userPrompt, flowYaml, conversationId}: {userPrompt: string, flowYaml: string, conversationId: string}) {
|
||||
const response = await AI.generateFlow({
|
||||
flowGenerationPrompt:{
|
||||
userPrompt,
|
||||
flowYaml,
|
||||
conversationId
|
||||
}
|
||||
const response = await axios.post(`${apiUrl()}/ai/generate/flow`, {
|
||||
userPrompt,
|
||||
flowYaml,
|
||||
conversationId
|
||||
});
|
||||
|
||||
return response.data;
|
||||
|
||||
@@ -25,6 +25,8 @@ interface Blueprint {
|
||||
[key: string]: any;
|
||||
}
|
||||
|
||||
export type TemplateArgument = Record<string, Input>;
|
||||
|
||||
export interface BlueprintTemplate {
|
||||
source: string;
|
||||
templateArguments: Record<string, Input>;
|
||||
@@ -55,6 +57,8 @@ export const useBlueprintsStore = defineStore("blueprints", () => {
|
||||
const source = ref<string | undefined>(undefined);
|
||||
const graph = ref<any | undefined>(undefined);
|
||||
|
||||
const validateYAML = ref<boolean>(true); // Used to enable/disable YAML validation in Monaco editor, for the purpose of Templated Blueprints
|
||||
|
||||
const getBlueprints = async (options: Options) => {
|
||||
const PARAMS = {params: options.params, ...VALIDATE};
|
||||
|
||||
@@ -166,6 +170,8 @@ export const useBlueprintsStore = defineStore("blueprints", () => {
|
||||
source,
|
||||
graph,
|
||||
|
||||
validateYAML,
|
||||
|
||||
getBlueprints,
|
||||
getBlueprint,
|
||||
getBlueprintSource,
|
||||
|
||||
@@ -1,7 +1,8 @@
|
||||
import {defineStore} from "pinia";
|
||||
import {apiUrl} from "override/utils/route";
|
||||
import {ref} from "vue";
|
||||
import {useAxios} from "../utils/axios";
|
||||
import {Message} from "../components/ErrorToast.vue";
|
||||
import * as sdk from "../generated/kestra-api/ks-sdk.gen";
|
||||
|
||||
interface GuidedProperties {
|
||||
tourStarted: boolean;
|
||||
@@ -20,13 +21,13 @@ export const useCoreStore = defineStore("core", () => {
|
||||
template: undefined,
|
||||
})
|
||||
const monacoYamlConfigured = ref(false)
|
||||
const tutorialFlows = ref<any[]>([]);
|
||||
const tutorialFlows = ref<any[]>([])
|
||||
|
||||
const axios = useAxios();
|
||||
|
||||
async function readTutorialFlows() {
|
||||
const response = await sdk.Flows.listFlowsByNamespace({
|
||||
namespace: "tutorials",
|
||||
})
|
||||
tutorialFlows.value = response.data ?? [];
|
||||
const response = await axios.get(`${apiUrl()}/flows/tutorial`);
|
||||
tutorialFlows.value = response.data;
|
||||
return response.data;
|
||||
}
|
||||
|
||||
|
||||
@@ -17,15 +17,13 @@ import {apiUrl} from "override/utils/route";
|
||||
|
||||
import Utils from "../utils/utils";
|
||||
|
||||
import type {Chart, Request, Parameters} from "../components/dashboard/composables/useDashboards";
|
||||
import type {Dashboard, Chart, Request, Parameters} from "../components/dashboard/composables/useDashboards";
|
||||
import {useAxios} from "../utils/axios";
|
||||
import {removeRefPrefix, usePluginsStore} from "./plugins";
|
||||
import * as YAML_UTILS from "@kestra-io/ui-libs/flow-yaml-utils";
|
||||
import _throttle from "lodash/throttle";
|
||||
import {useCoreStore} from "./core";
|
||||
import {useI18n} from "vue-i18n";
|
||||
import {Dashboards} from "../generated/kestra-api/ks-sdk.gen";
|
||||
import {Dashboard} from "../generated/kestra-api";
|
||||
|
||||
|
||||
|
||||
@@ -48,25 +46,17 @@ export const useDashboardStore = defineStore("dashboard", () => {
|
||||
|
||||
async function list(options: Record<string, any>) {
|
||||
const {sort, ...params} = options;
|
||||
const response = await Dashboards.searchDashboards({
|
||||
size: 100,
|
||||
sort,
|
||||
page: 1,
|
||||
...params
|
||||
})
|
||||
const response = await axios.get(`${apiUrl()}/dashboards?size=100${sort ? `&sort=${sort}` : ""}`, {params});
|
||||
|
||||
return response.data;
|
||||
}
|
||||
|
||||
async function load(id: string) {
|
||||
const response = await Dashboards.getDashboard({id}, {validateStatus});
|
||||
let dashboardLoaded: Dashboard & {id: string};
|
||||
async function load(id: Dashboard["id"]) {
|
||||
const response = await axios.get(`${apiUrl()}/dashboards/${id}`, {validateStatus});
|
||||
let dashboardLoaded: Dashboard;
|
||||
|
||||
if (response.status === 200 && response.data) {
|
||||
dashboardLoaded = {...response.data, id};
|
||||
} else {
|
||||
dashboardLoaded = {title: "Default", id, charts: [], sourceCode: ""};
|
||||
}
|
||||
if (response.status === 200) dashboardLoaded = response.data;
|
||||
else dashboardLoaded = {title: "Default", id, charts: [], sourceCode: ""};
|
||||
|
||||
dashboard.value = dashboardLoaded;
|
||||
sourceCode.value = dashboardLoaded.sourceCode ?? ""
|
||||
@@ -79,12 +69,12 @@ export const useDashboardStore = defineStore("dashboard", () => {
|
||||
return response.data;
|
||||
}
|
||||
|
||||
async function update({id, source}: {id: string; source: Dashboard["sourceCode"];}) {
|
||||
async function update({id, source}: {id: Dashboard["id"]; source: Dashboard["sourceCode"];}) {
|
||||
const response = await axios.put(`${apiUrl()}/dashboards/${id}`, source, header);
|
||||
return response.data;
|
||||
}
|
||||
|
||||
async function deleteDashboard(id: string) {
|
||||
async function deleteDashboard(id: Dashboard["id"]) {
|
||||
const response = await axios.delete(`${apiUrl()}/dashboards/${id}`);
|
||||
return response.data;
|
||||
}
|
||||
@@ -94,7 +84,7 @@ export const useDashboardStore = defineStore("dashboard", () => {
|
||||
return response.data;
|
||||
}
|
||||
|
||||
async function generate(id: string, chartId: Chart["id"], parameters: Parameters) {
|
||||
async function generate(id: Dashboard["id"], chartId: Chart["id"], parameters: Parameters) {
|
||||
const response = await axios.post(`${apiUrl()}/dashboards/${id}/charts/${chartId}`, parameters, {validateStatus});
|
||||
return response.data;
|
||||
}
|
||||
|
||||
@@ -10,15 +10,13 @@ import {useUnsavedChangesStore} from "./unsavedChanges";
|
||||
import {defineStore} from "pinia";
|
||||
import {FlowGraph} from "@kestra-io/ui-libs/vue-flow-utils";
|
||||
import {makeToast} from "../utils/toast";
|
||||
import {InputType} from "../utils/inputs";
|
||||
import {globalI18n} from "../translations/i18n";
|
||||
import {transformResponse} from "../components/dependencies/composables/useDependencies";
|
||||
import {useAuthStore} from "override/stores/auth";
|
||||
import {useRoute} from "vue-router";
|
||||
import {useAxios} from "../utils/axios";
|
||||
import {defaultNamespace} from "../composables/useNamespaces";
|
||||
import {Flow, FlowWithSource} from "../generated/kestra-api";
|
||||
import * as sdk from "../generated/kestra-api/ks-sdk.gen";
|
||||
import {InputType} from "../utils/inputs";
|
||||
|
||||
const textYamlHeader = {
|
||||
headers: {
|
||||
@@ -56,9 +54,26 @@ interface FlowValidations {
|
||||
deprecationPaths?: string[];
|
||||
}
|
||||
|
||||
export interface Flow {
|
||||
id: string;
|
||||
namespace: string;
|
||||
source: string;
|
||||
revision?: number;
|
||||
deleted?: boolean;
|
||||
disabled?: boolean;
|
||||
labels?: Record<string, string | boolean>;
|
||||
triggers?: Trigger[];
|
||||
inputs?: Input[];
|
||||
errors?: { message: string; code?: string, id?: string }[];
|
||||
concurrency?: {
|
||||
limit: number;
|
||||
behavior: string;
|
||||
};
|
||||
}
|
||||
|
||||
export const useFlowStore = defineStore("flow", () => {
|
||||
const flows = ref<Flow[]>()
|
||||
const flow = ref<FlowWithSource>()
|
||||
const flow = ref<Flow>()
|
||||
const task = ref<Task>()
|
||||
const search = ref<any[]>()
|
||||
const total = ref<number>(0)
|
||||
@@ -239,10 +254,7 @@ export const useFlowStore = defineStore("flow", () => {
|
||||
const isCreatingBackup = isCreating.value;
|
||||
if (isCreating.value && !overrideFlow) {
|
||||
await createFlow({flow: flowSource ?? ""})
|
||||
.then((response) => {
|
||||
if(!response){
|
||||
return;
|
||||
}
|
||||
.then((response: Flow) => {
|
||||
toast.saved(response.id);
|
||||
isCreating.value = false;
|
||||
});
|
||||
@@ -279,7 +291,7 @@ export const useFlowStore = defineStore("flow", () => {
|
||||
|
||||
async function initYamlSource() {
|
||||
if (!flow.value) return;
|
||||
const {source = ""} = flow.value;
|
||||
const {source} = flow.value;
|
||||
flowYaml.value = source;
|
||||
flowYamlOrigin.value = source;
|
||||
if (flowHaveTasks.value) {
|
||||
@@ -287,22 +299,21 @@ export const useFlowStore = defineStore("flow", () => {
|
||||
}
|
||||
|
||||
// validate flow on first load
|
||||
return validateFlow({
|
||||
flow: isCreating.value ? source : yamlWithNextRevision.value
|
||||
})
|
||||
return validateFlow({flow: isCreating.value ? source : yamlWithNextRevision.value})
|
||||
}
|
||||
|
||||
function findFlows(options: Parameters<typeof sdk.flowsSearchFlows>[0] & { onlyTotal?: boolean }) {
|
||||
return sdk.flowsSearchFlows(options).then(response => {
|
||||
if(!response.data){
|
||||
return undefined
|
||||
}
|
||||
function findFlows(options: { [key: string]: any }) {
|
||||
const sortString = options.sort ? `?sort=${options.sort}` : ""
|
||||
delete options.sort
|
||||
return axios.get(`${apiUrl()}/flows/search${sortString}`, {
|
||||
params: options
|
||||
}).then(response => {
|
||||
if (options.onlyTotal) {
|
||||
return response.data.total;
|
||||
}
|
||||
|
||||
else {
|
||||
flows.value = response.data?.results
|
||||
flows.value = response.data.results
|
||||
total.value = response.data.total
|
||||
overallTotal.value = response.data.results.filter((f: any) => f.namespace !== "tutorial").length
|
||||
|
||||
@@ -329,22 +340,20 @@ export const useFlowStore = defineStore("flow", () => {
|
||||
})
|
||||
}
|
||||
|
||||
function loadFlow(options: { namespace: string, id: string, revision?: number, allowDeleted?: boolean, source?: boolean, store?: boolean, deleted?: boolean, httpClient?: any }) {
|
||||
const httpClient = options.httpClient
|
||||
return sdk.flowsGetFlow({
|
||||
id: options.id,
|
||||
namespace: options.namespace,
|
||||
function loadFlow(options: { namespace: string, id: string, revision?: string, allowDeleted?: boolean, source?: boolean, store?: boolean, deleted?: boolean, httpClient?: any }) {
|
||||
const httpClient = options.httpClient ?? axios
|
||||
return httpClient.get(`${apiUrl()}/flows/${options.namespace}/${options.id}`,
|
||||
{
|
||||
params: {
|
||||
revision: options.revision,
|
||||
allowDeleted: options.allowDeleted ?? false,
|
||||
source: options.source === undefined ? true : false
|
||||
allowDeleted: options.allowDeleted,
|
||||
source: options.source === undefined ? true : undefined
|
||||
},
|
||||
validateStatus: (status: number) => {
|
||||
return options.deleted ? status === 200 || status === 404 : status === 200;
|
||||
}
|
||||
, {
|
||||
client: httpClient
|
||||
})
|
||||
.then((response) => {
|
||||
if(!response.data){
|
||||
return Promise.reject("Flow not found");
|
||||
}
|
||||
.then((response: any) => {
|
||||
if (response.data.exception) {
|
||||
coreStore.message = {
|
||||
title: "Invalid source code",
|
||||
@@ -364,8 +373,8 @@ export const useFlowStore = defineStore("flow", () => {
|
||||
}
|
||||
|
||||
flow.value = response.data;
|
||||
flowYaml.value = response.data.source ?? "";
|
||||
flowYamlOrigin.value = response.data.source ?? "";
|
||||
flowYaml.value = response.data.source;
|
||||
flowYamlOrigin.value = response.data.source;
|
||||
overallTotal.value = 1;
|
||||
|
||||
return response.data;
|
||||
@@ -421,11 +430,11 @@ export const useFlowStore = defineStore("flow", () => {
|
||||
}
|
||||
|
||||
function createFlow(options: { flow: string }) {
|
||||
return sdk.flowsCreateFlow({body: options.flow}, {
|
||||
return axios.post(`${apiUrl()}/flows`, options.flow, {
|
||||
...textYamlHeader,
|
||||
...VALIDATE
|
||||
}).then(response => {
|
||||
if (!response?.status || response.status >= 300) {
|
||||
if (response.status >= 300) {
|
||||
return Promise.reject(response)
|
||||
}
|
||||
|
||||
@@ -644,12 +653,7 @@ function deleteFlowAndDependencies() {
|
||||
return axios.delete(`${apiUrl()}/flows/delete/by-query`, {params: options})
|
||||
}
|
||||
|
||||
function validateFlow(options: { flow?: string }) {
|
||||
if(!options.flow) {
|
||||
return Promise.resolve({
|
||||
constraints: t("flow must not be empty")
|
||||
});
|
||||
}
|
||||
function validateFlow(options: { flow: string }) {
|
||||
const flowValidationIssues: FlowValidations = {};
|
||||
if(isCreating.value) {
|
||||
const {namespace} = YAML_UTILS.getMetadata(options.flow);
|
||||
|
||||
@@ -1,6 +1,10 @@
|
||||
@import "@kestra-io/ui-libs/src/scss/variables.scss";
|
||||
|
||||
#app {
|
||||
.v-sidebar-menu.vsm_expanded.overlay {
|
||||
position: absolute;
|
||||
}
|
||||
|
||||
.vsm--item {
|
||||
padding: 0 30px;
|
||||
transition: padding 0.2s ease;
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Nach Trigger-ID filtern",
|
||||
"label": "Trigger-ID"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Nach Trigger-Zustand filtern",
|
||||
"disabled": "Deaktiviert",
|
||||
"enabled": "Aktiviert",
|
||||
"label": "Trigger-Zustand"
|
||||
},
|
||||
"update": "Aktualisieren",
|
||||
"value": "Wert",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Weiter",
|
||||
"no_flows": "Keine Flows unter der tutorial Namespace verfügbar.",
|
||||
"previous": "Zurück",
|
||||
"skip": "Tutorial überspringen",
|
||||
"skip": "Produkt-Tour überspringen",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Wir freuen uns, dass Sie hier sind.<br />Lassen Sie uns Ihren ersten Flow erstellen.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Willkommen bei Kestra!",
|
||||
"welcome aboard content": "Nutzen Sie unsere geführte Tour, um Ihren ersten Flow zu erstellen, und schauen Sie sich Blueprints an, um weitere Beispiele zu finden.",
|
||||
"welcome button create": "Meinen ersten Flow erstellen",
|
||||
"welcome button create": "Produkt-Tour starten",
|
||||
"welcome display require": "Führen Sie Ihren <strong>ersten Flow</strong> aus, um loszulegen",
|
||||
"welcome_page": {
|
||||
"guide": "Benötigen Sie Unterstützung, um Ihren ersten flow auszuführen?",
|
||||
|
||||
@@ -540,7 +540,7 @@
|
||||
"welcome aboard": "\uD83D\uDE80 Welcome to Kestra!",
|
||||
"welcome aboard content": "Use our Guided Tour to create your first flow and check Blueprints to find more examples.",
|
||||
"welcome display require": "Run your <strong>first flow</strong> to get started",
|
||||
"welcome button create": "Create my first flow",
|
||||
"welcome button create": "Start Product Tour",
|
||||
"live help": "Live help",
|
||||
"show task documentation": "Show task documentation",
|
||||
"hide task documentation": "Hide task documentation",
|
||||
@@ -947,7 +947,7 @@
|
||||
"next": "Next",
|
||||
"previous": "Previous",
|
||||
"finish": "Finish",
|
||||
"skip": "Skip Tutorial",
|
||||
"skip": "Skip Product Tour",
|
||||
"no_flows": "No flows available under tutorial namespace.",
|
||||
"steps": {
|
||||
"0": {
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Filtrar por trigger ID",
|
||||
"label": "ID de Trigger"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Filtrar por estado del trigger",
|
||||
"disabled": "Desactivado",
|
||||
"enabled": "Habilitado",
|
||||
"label": "Estado del Trigger"
|
||||
},
|
||||
"update": "Actualizar",
|
||||
"value": "Valor",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Siguiente",
|
||||
"no_flows": "No hay flows disponibles bajo el namespace del tutorial.",
|
||||
"previous": "Anterior",
|
||||
"skip": "Saltar Tutorial",
|
||||
"skip": "Omitir recorrido del producto",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Estamos encantados de tenerte aquí.<br />Vamos a crear tu primer flow.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 ¡Bienvenido a Kestra!",
|
||||
"welcome aboard content": "Usa nuestro Tour Guiado para crear tu primer flow y revisa los Blueprints para encontrar más ejemplos.",
|
||||
"welcome button create": "Crear mi primer flow",
|
||||
"welcome button create": "Iniciar Tour del Producto",
|
||||
"welcome display require": "Ejecuta tu <strong>primer flow</strong> para comenzar",
|
||||
"welcome_page": {
|
||||
"guide": "¿Necesitas orientación para ejecutar tu primer flow?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Filtrer par trigger ID",
|
||||
"label": "ID du trigger"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Filtrer par état du trigger",
|
||||
"disabled": "Désactivé",
|
||||
"enabled": "Activé",
|
||||
"label": "État du Trigger"
|
||||
},
|
||||
"update": "Mettre à jour",
|
||||
"value": "Valeur",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Suivant",
|
||||
"no_flows": "Aucun flux disponible dans l'espace de noms du tutoriel.",
|
||||
"previous": "Précédent",
|
||||
"skip": "Passer le tutoriel",
|
||||
"skip": "Passer la visite guidée du produit",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Nous sommes ravis de vous avoir ici.<br />Créons votre premier flux.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Bienvenue à bord !",
|
||||
"welcome aboard content": "Tout est prêt pour Kestra, commencez la création de votre flow et admirez la magie !",
|
||||
"welcome button create": "Créer mon premier flow",
|
||||
"welcome button create": "Démarrer la visite guidée du produit",
|
||||
"welcome display require": "Prêt à commencer à utiliser Kestra ? Créons ensemble <strong>votre premier flow</strong> !",
|
||||
"welcome_page": {
|
||||
"guide": "Besoin d'aide pour exécuter votre premier flow ?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "ट्रिगर ID द्वारा फ़िल्टर करें",
|
||||
"label": "ट्रिगर ID"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "ट्रिगर स्थिति द्वारा फ़िल्टर करें",
|
||||
"disabled": "अक्षम",
|
||||
"enabled": "सक्रिय",
|
||||
"label": "ट्रिगर स्थिति"
|
||||
},
|
||||
"update": "अपडेट",
|
||||
"value": "मान",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "अगला",
|
||||
"no_flows": "ट्यूटोरियल namespace के अंतर्गत कोई flows उपलब्ध नहीं हैं।",
|
||||
"previous": "पिछला",
|
||||
"skip": "ट्यूटोरियल छोड़ें",
|
||||
"skip": "उत्पाद टूर छोड़ें",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "हमें खुशी है कि आप यहाँ हैं।<br />आइए अपना पहला flow बनाएं।",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 केस्ट्रा में आपका स्वागत है!",
|
||||
"welcome aboard content": "अपना पहला flow बनाने के लिए हमारे Guided Tour का उपयोग करें और अधिक उदाहरण खोजने के लिए Blueprints देखें।",
|
||||
"welcome button create": "मेरा पहला flow बनाएं",
|
||||
"welcome button create": "उत्पाद यात्रा शुरू करें",
|
||||
"welcome display require": "शुरू करने के लिए अपना <strong>पहला flow</strong> चलाएँ",
|
||||
"welcome_page": {
|
||||
"guide": "क्या आपको अपना पहला flow निष्पादित करने के लिए मार्गदर्शन चाहिए?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Filtra per trigger ID",
|
||||
"label": "ID del trigger"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Filtra per stato del trigger",
|
||||
"disabled": "Disabilitato",
|
||||
"enabled": "Abilitato",
|
||||
"label": "Stato del Trigger"
|
||||
},
|
||||
"update": "Aggiorna",
|
||||
"value": "Valore",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Successivo",
|
||||
"no_flows": "Nessun flow disponibile sotto il namespace del tutorial.",
|
||||
"previous": "Precedente",
|
||||
"skip": "Salta Tutorial",
|
||||
"skip": "Salta il Tour del Prodotto",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Siamo entusiasti di averti qui.<br />Creiamo il tuo primo flow.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Benvenuto su Kestra!",
|
||||
"welcome aboard content": "Usa il nostro Tour Guidato per creare il tuo primo flow e controlla i Blueprints per trovare altri esempi.",
|
||||
"welcome button create": "Crea il mio primo flow",
|
||||
"welcome button create": "Inizia il Tour del Prodotto",
|
||||
"welcome display require": "Esegui il tuo <strong>primo flow</strong> per iniziare",
|
||||
"welcome_page": {
|
||||
"guide": "Hai bisogno di assistenza per eseguire il tuo primo flow?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "トリガーIDでフィルター",
|
||||
"label": "トリガーID"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "トリガー状態でフィルター",
|
||||
"disabled": "無効",
|
||||
"enabled": "有効",
|
||||
"label": "トリガー状態"
|
||||
},
|
||||
"update": "更新",
|
||||
"value": "値",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "次へ",
|
||||
"no_flows": "チュートリアルnamespaceに利用可能なflowはありません。",
|
||||
"previous": "前へ",
|
||||
"skip": "チュートリアルをスキップ",
|
||||
"skip": "プロダクトツアーをスキップ",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "ここに来てくれてとても嬉しいです。<br />最初のflowを作成しましょう。",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Kestraへようこそ!",
|
||||
"welcome aboard content": "ガイド付きツアーを利用して最初のflowを作成し、Blueprintsでさらに多くの例を見つけてください。",
|
||||
"welcome button create": "最初のflowを作成",
|
||||
"welcome button create": "プロダクトツアーを開始",
|
||||
"welcome display require": "<strong>最初のflow</strong>を実行して始めましょう",
|
||||
"welcome_page": {
|
||||
"guide": "最初のflowを実行するためのガイダンスが必要ですか?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "트리거 ID로 필터링",
|
||||
"label": "트리거 ID"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "트리거 상태별 필터링",
|
||||
"disabled": "비활성화됨",
|
||||
"enabled": "사용 가능",
|
||||
"label": "트리거 상태"
|
||||
},
|
||||
"update": "업데이트",
|
||||
"value": "값",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "다음",
|
||||
"no_flows": "튜토리얼 namespace에 사용할 수 있는 flow가 없습니다.",
|
||||
"previous": "이전",
|
||||
"skip": "튜토리얼 건너뛰기",
|
||||
"skip": "제품 투어 건너뛰기",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "여기 오신 것을 환영합니다.<br />첫 번째 flow를 만들어 봅시다.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Kestra에 오신 것을 환영합니다!",
|
||||
"welcome aboard content": "가이드 투어를 사용하여 첫 번째 flow를 만들고 Blueprints에서 더 많은 예제를 확인하세요.",
|
||||
"welcome button create": "첫 번째 flow 만들기",
|
||||
"welcome button create": "제품 투어 시작",
|
||||
"welcome display require": "<strong>첫 번째 flow</strong>를 실행하여 시작하세요",
|
||||
"welcome_page": {
|
||||
"guide": "첫 번째 flow를 실행하는 데 도움이 필요하신가요?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Filtruj według trigger ID",
|
||||
"label": "Identyfikator Trigger"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Filtruj według stanu triggera",
|
||||
"disabled": "Wyłączone",
|
||||
"enabled": "Włączone",
|
||||
"label": "Stan Trigger"
|
||||
},
|
||||
"update": "Aktualizuj",
|
||||
"value": "Wartość",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Następny",
|
||||
"no_flows": "Brak flowów w namespace o nazwie tutorial.",
|
||||
"previous": "Poprzedni",
|
||||
"skip": "Pomiń samouczek",
|
||||
"skip": "Pomiń Przewodnik po Produkcie",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Jesteśmy zachwyceni, że jesteś tutaj.<br />Stwórzmy twój pierwszy flow.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Witamy w Kestra!",
|
||||
"welcome aboard content": "Skorzystaj z naszego Przewodnika, aby stworzyć swój pierwszy flow i sprawdź Blueprints, aby znaleźć więcej przykładów.",
|
||||
"welcome button create": "Stwórz mój pierwszy flow",
|
||||
"welcome button create": "Rozpocznij Przewodnik po Produkcie",
|
||||
"welcome display require": "Uruchom swój <strong>pierwszy flow</strong>, aby rozpocząć",
|
||||
"welcome_page": {
|
||||
"guide": "Potrzebujesz wskazówek, jak uruchomić swój pierwszy flow?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Filtrar por trigger ID",
|
||||
"label": "ID do Trigger"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Filtrar por estado do trigger",
|
||||
"disabled": "Desativado",
|
||||
"enabled": "Habilitado",
|
||||
"label": "Estado do Trigger"
|
||||
},
|
||||
"update": "Atualizar",
|
||||
"value": "Valor",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Próximo",
|
||||
"no_flows": "Não há flows disponíveis no namespace do tutorial.",
|
||||
"previous": "Anterior",
|
||||
"skip": "Pular Tutorial",
|
||||
"skip": "Pular Tour do Produto",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Estamos entusiasmados em tê-lo aqui.<br />Vamos criar seu primeiro flow.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Bem-vindo ao Kestra!",
|
||||
"welcome aboard content": "Use nosso Tour Guiado para criar seu primeiro flow e confira os Blueprints para encontrar mais exemplos.",
|
||||
"welcome button create": "Criar meu primeiro flow",
|
||||
"welcome button create": "Iniciar Tour do Produto",
|
||||
"welcome display require": "Execute seu <strong>primeiro flow</strong> para começar",
|
||||
"welcome_page": {
|
||||
"guide": "Precisa de orientação para executar seu primeiro flow?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Filtrar por trigger ID",
|
||||
"label": "ID do Trigger"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Filtrar por estado do trigger",
|
||||
"disabled": "Desativado",
|
||||
"enabled": "Habilitado",
|
||||
"label": "Estado do Trigger"
|
||||
},
|
||||
"update": "Atualizar",
|
||||
"value": "Valor",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Próximo",
|
||||
"no_flows": "Não há flows disponíveis no namespace do tutorial.",
|
||||
"previous": "Anterior",
|
||||
"skip": "Pular Tutorial",
|
||||
"skip": "Pular Tour do Produto",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Estamos entusiasmados em tê-lo aqui.<br />Vamos criar seu primeiro flow.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Bem-vindo ao Kestra!",
|
||||
"welcome aboard content": "Use nosso Tour Guiado para criar seu primeiro flow e confira os Blueprints para encontrar mais exemplos.",
|
||||
"welcome button create": "Criar meu primeiro flow",
|
||||
"welcome button create": "Iniciar Tour do Produto",
|
||||
"welcome display require": "Execute seu <strong>primeiro flow</strong> para começar",
|
||||
"welcome_page": {
|
||||
"guide": "Precisa de orientação para executar seu primeiro flow?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "Фильтр по trigger ID",
|
||||
"label": "ID триггера"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "Фильтр по состоянию trigger",
|
||||
"disabled": "Отключено",
|
||||
"enabled": "Включено",
|
||||
"label": "Состояние Trigger"
|
||||
},
|
||||
"update": "Обновить",
|
||||
"value": "Значение",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "Далее",
|
||||
"no_flows": "Нет доступных flows в namespace учебника.",
|
||||
"previous": "Назад",
|
||||
"skip": "Пропустить учебник",
|
||||
"skip": "Пропустить ознакомительный тур с продуктом",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "Мы рады видеть вас здесь.<br />Давайте создадим ваш первый flow.",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 Добро пожаловать в Kestra!",
|
||||
"welcome aboard content": "Используйте наше Руководство, чтобы создать ваш первый flow и ознакомьтесь с Blueprints для поиска дополнительных примеров.",
|
||||
"welcome button create": "Создать мой первый flow",
|
||||
"welcome button create": "Начать ознакомительный тур с продуктом",
|
||||
"welcome display require": "Запустите ваш <strong>первый flow</strong>, чтобы начать",
|
||||
"welcome_page": {
|
||||
"guide": "Нужна помощь в выполнении вашего первого flow?",
|
||||
|
||||
@@ -883,6 +883,12 @@
|
||||
"description": "按 trigger ID 筛选",
|
||||
"label": "触发器 ID"
|
||||
},
|
||||
"triggerState": {
|
||||
"description": "按触发器状态筛选",
|
||||
"disabled": "禁用",
|
||||
"enabled": "启用",
|
||||
"label": "触发状态"
|
||||
},
|
||||
"update": "更新",
|
||||
"value": "值",
|
||||
"workerId": {
|
||||
@@ -1308,7 +1314,7 @@
|
||||
"next": "下一步",
|
||||
"no_flows": "教程命名空间下没有可用的流程。",
|
||||
"previous": "上一步",
|
||||
"skip": "跳过教程",
|
||||
"skip": "跳过产品指南",
|
||||
"steps": {
|
||||
"0": {
|
||||
"content": "我们很高兴你在这里。<br />让我们创建你的第一个流程。",
|
||||
@@ -1857,7 +1863,7 @@
|
||||
},
|
||||
"welcome aboard": "🚀 欢迎使用 Kestra!",
|
||||
"welcome aboard content": "使用我们的引导游览来创建你的第一个流程,并查看蓝图以找到更多示例。",
|
||||
"welcome button create": "创建我的第一个流程",
|
||||
"welcome button create": "开始产品导览",
|
||||
"welcome display require": "运行你的 <strong>第一个流程</strong> 以开始",
|
||||
"welcome_page": {
|
||||
"guide": "需要指导来执行您的第一个flow吗?",
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import axios, {AxiosRequestConfig, AxiosResponse, AxiosError, AxiosProgressEvent} from "axios"
|
||||
import axios, {AxiosInstance, AxiosRequestConfig, AxiosResponse, AxiosError, AxiosProgressEvent} from "axios"
|
||||
import NProgress from "nprogress"
|
||||
import {Router, useRouter} from "vue-router"
|
||||
import {storageKeys} from "./constants"
|
||||
@@ -8,9 +8,6 @@ import * as BasicAuth from "../utils/basicAuth"
|
||||
import {useAuthStore} from "override/stores/auth"
|
||||
import {useMiscStore} from "override/stores/misc";
|
||||
import {useUnsavedChangesStore} from "../stores/unsavedChanges"
|
||||
import {client} from "../generated/kestra-api/client.gen"
|
||||
import {Client} from "../generated/kestra-api/client"
|
||||
export * as sdk from "../generated/kestra-api/sdk.gen"
|
||||
|
||||
let pendingRoute = false
|
||||
let requestsTotal = 0
|
||||
@@ -75,7 +72,7 @@ interface QueueItem {
|
||||
resolve: (value: AxiosResponse | Promise<AxiosResponse>) => void
|
||||
}
|
||||
|
||||
const createAxios = (
|
||||
export const createAxios = (
|
||||
router: Router | undefined,
|
||||
oss: boolean
|
||||
) => {
|
||||
@@ -286,38 +283,28 @@ const createAxios = (
|
||||
}
|
||||
})
|
||||
|
||||
client.setConfig({
|
||||
axios: instance
|
||||
})
|
||||
|
||||
return client;
|
||||
return instance;
|
||||
};
|
||||
|
||||
export default (
|
||||
callback: (clientInstance: Client["instance"]) => void,
|
||||
callback: (instance: AxiosInstance) => void,
|
||||
_store: any,
|
||||
...args: Parameters<typeof createAxios>
|
||||
) => {
|
||||
callback(createAxios(...args).instance);
|
||||
callback(createAxios(...args));
|
||||
}
|
||||
|
||||
let clientInstance: Client | null = null;
|
||||
let axiosInstance: AxiosInstance | null = null;
|
||||
|
||||
export function useClient(){
|
||||
export const useAxios = () => {
|
||||
const router = useRouter();
|
||||
|
||||
const miscStore = useMiscStore();
|
||||
const {edition} = miscStore.configs || {};
|
||||
|
||||
if (!clientInstance) {
|
||||
clientInstance = createAxios(router, edition === "OSS");
|
||||
if (!axiosInstance) {
|
||||
axiosInstance = createAxios(router, edition === "OSS");
|
||||
}
|
||||
|
||||
return clientInstance;
|
||||
};
|
||||
|
||||
export function useAxios(){
|
||||
const clientInstance = useClient();
|
||||
|
||||
return clientInstance.instance;
|
||||
return axiosInstance;
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user