Compare commits

..

21 Commits

Author SHA1 Message Date
github-actions[bot]
23329f4d48 chore(version): update to version '1.1.1' 2025-11-06 17:18:11 +00:00
Loïc Mathieu
ed60cb6670 fix(core): relax assertion on ConcurrencyLimitServiceTest.findById() 2025-11-06 18:16:32 +01:00
brian-mulier-p
f6306883b4 fix(kv): all types properly handled and avoid trimming string KV values (#12765)
closes https://github.com/kestra-io/kestra-ee/issues/5718
2025-11-06 15:47:44 +01:00
Loïc Mathieu
89433dc04c fix(system): killing a paused flow should kill the Pause task attempt
Fixes #12421
2025-11-06 15:33:56 +01:00
Loïc Mathieu
4837408c59 chore(test): try to un-flaky ConcurrencyLimitServiceTest.findById().
By making sure the unqueueExecution() test wait for the unqueued execution to ends to avoid any potential races.
2025-11-06 15:33:56 +01:00
Miloš Paunović
5a8c36caa5 fix(variables): properly send kv value when the type is json (#12759)
Closes https://github.com/kestra-io/kestra/issues/12739.
2025-11-06 15:33:56 +01:00
Piyush Bhaskar
a2335abc0c fix(core): make the interval in triggers work (#12764) 2025-11-06 19:39:10 +05:30
Piyush Bhaskar
310a7bbbe9 Revert "fix(core): apply timeRange filter in triggers (#12721)" 2025-11-06 18:56:37 +05:30
Jay Balwani
162feaf38c Fix(UI)/kv type boolean (#12643)
Co-authored-by: Miloš Paunović <paun992@hotmail.com>
Co-authored-by: Piyush Bhaskar <impiyush0012@gmail.com>
2025-11-06 16:37:01 +05:30
Piyush Bhaskar
94050be49c fix(core): apply timeRange filter in triggers (#12721) 2025-11-06 16:29:47 +05:30
brian-mulier-p
848a5ac9d7 fix(cli): metadata commands weren't working with external storages (#12743)
closes #12713
2025-11-06 11:47:59 +01:00
Barthélémy Ledoux
9ac7a9ce9a fix: responsive dashboard grid (#12608) 2025-11-06 11:03:26 +01:00
Piyush Bhaskar
c42838f3e1 feat(ui): persist scroll across No‑code, editor tabs, and docs via Pinia view-state and scroll-memory (#12358) 2025-11-06 11:53:07 +05:30
Irfan
c499d62b63 fix(core): going back from plugin doc will take to plugins home (#12621)
Co-authored-by: iitzIrFan <irfanlhawk@gmail.com>
Co-authored-by: Piyush Bhaskar <impiyush0012@gmail.com>
2025-11-06 11:50:34 +05:30
Piyush Bhaskar
8fbc62e12c fix(core): proper deletion of single and multi ns files (#12618) 2025-11-06 11:49:51 +05:30
Vipin Chandra Sao
ae143f29f4 fix(ui): prevent "Invalid date" display in Gantt view for executions … (#12605)
* fix(ui): prevent "Invalid date" display in Gantt view for executions that never started

- Added defensive checks wherever histories arrays might be empty
- Now renders blank or safe values instead of "Invalid date"
- Improved comments for maintainability and future debugging
- Addresses issue #12583

* revert the changes

* fix: remove the card when invalid date

---------

Co-authored-by: Piyush Bhaskar <impiyush0012@gmail.com>
Co-authored-by: Piyush Bhaskar <102078527+Piyush-r-bhaskar@users.noreply.github.com>
2025-11-06 11:49:02 +05:30
Piyush Bhaskar
e4a11fc9ce fix(core): remove double info icon (#12623) 2025-11-06 11:48:36 +05:30
Piyush Bhaskar
ebacfc70b9 fix(core): use proper option after P30D in misc (#12624) 2025-11-06 11:47:57 +05:30
Loïc Mathieu
5bf67180a3 fix(system): trigger an execution once per condition on flow triggers
Fixes #12560
2025-11-05 15:31:41 +01:00
Roman Acevedo
1e670b5e7e test(kv): plain text header is sent now 2025-11-04 15:17:02 +01:00
brian.mulier
0dacad5ee1 chore(version): upgrade to v1.1.0 2025-11-04 13:58:32 +01:00
117 changed files with 2713 additions and 3001 deletions

View File

@@ -126,7 +126,7 @@ By default, Kestra will be installed under: `$HOME/.kestra/current`. Set the `KE
```bash
# build and install Kestra
make install
# install plugins (plugins installation is based on the API).
# install plugins (plugins installation is based on the `.plugins` or `.plugins.override` files located at the root of the project.
make install-plugins
# start Kestra in standalone mode with Postgres as backend
make start-standalone-postgres

View File

@@ -0,0 +1,74 @@
name: Run Gradle Release for Kestra Plugins
on:
workflow_dispatch:
inputs:
releaseVersion:
description: 'The release version (e.g., 0.21.0)'
required: true
type: string
nextVersion:
description: 'The next version (e.g., 0.22.0-SNAPSHOT)'
required: true
type: string
dryRun:
description: 'Use DRY_RUN mode'
required: false
default: 'false'
jobs:
release:
name: Release plugins
runs-on: ubuntu-latest
steps:
# Checkout
- uses: actions/checkout@v5
with:
fetch-depth: 0
# Setup build
- uses: kestra-io/actions/composite/setup-build@main
id: build
with:
java-enabled: true
node-enabled: true
python-enabled: true
# Get Plugins List
- name: Get Plugins List
uses: kestra-io/actions/composite/kestra-oss/kestra-oss-plugins-list@main
id: plugins-list
with:
plugin-version: 'LATEST'
- name: 'Configure Git'
run: |
git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com"
git config --global user.name "github-actions[bot]"
# Execute
- name: Run Gradle Release
if: ${{ github.event.inputs.dryRun == 'false' }}
env:
GITHUB_PAT: ${{ secrets.GH_PERSONAL_TOKEN }}
run: |
chmod +x ./dev-tools/release-plugins.sh;
./dev-tools/release-plugins.sh \
--release-version=${{github.event.inputs.releaseVersion}} \
--next-version=${{github.event.inputs.nextVersion}} \
--yes \
${{ steps.plugins-list.outputs.repositories }}
- name: Run Gradle Release (DRY_RUN)
if: ${{ github.event.inputs.dryRun == 'true' }}
env:
GITHUB_PAT: ${{ secrets.GH_PERSONAL_TOKEN }}
run: |
chmod +x ./dev-tools/release-plugins.sh;
./dev-tools/release-plugins.sh \
--release-version=${{github.event.inputs.releaseVersion}} \
--next-version=${{github.event.inputs.nextVersion}} \
--dry-run \
--yes \
${{ steps.plugins-list.outputs.repositories }}

View File

@@ -0,0 +1,60 @@
name: Set Version and Tag Plugins
on:
workflow_dispatch:
inputs:
releaseVersion:
description: 'The release version (e.g., 0.21.0)'
required: true
type: string
dryRun:
description: 'Use DRY_RUN mode'
required: false
default: 'false'
jobs:
tag:
name: Release plugins
runs-on: ubuntu-latest
steps:
# Checkout
- uses: actions/checkout@v5
with:
fetch-depth: 0
# Get Plugins List
- name: Get Plugins List
uses: kestra-io/actions/composite/kestra-oss/kestra-oss-plugins-list@main
id: plugins-list
with:
plugin-version: 'LATEST'
- name: 'Configure Git'
run: |
git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com"
git config --global user.name "github-actions[bot]"
# Execute
- name: Set Version and Tag Plugins
if: ${{ github.event.inputs.dryRun == 'false' }}
env:
GITHUB_PAT: ${{ secrets.GH_PERSONAL_TOKEN }}
run: |
chmod +x ./dev-tools/setversion-tag-plugins.sh;
./dev-tools/setversion-tag-plugins.sh \
--release-version=${{github.event.inputs.releaseVersion}} \
--yes \
${{ steps.plugins-list.outputs.repositories }}
- name: Set Version and Tag Plugins (DRY_RUN)
if: ${{ github.event.inputs.dryRun == 'true' }}
env:
GITHUB_PAT: ${{ secrets.GH_PERSONAL_TOKEN }}
run: |
chmod +x ./dev-tools/setversion-tag-plugins.sh;
./dev-tools/setversion-tag-plugins.sh \
--release-version=${{github.event.inputs.releaseVersion}} \
--dry-run \
--yes \
${{ steps.plugins-list.outputs.repositories }}

View File

@@ -13,7 +13,7 @@ SHELL := /bin/bash
KESTRA_BASEDIR := $(shell echo $${KESTRA_HOME:-$$HOME/.kestra/current})
KESTRA_WORKER_THREAD := $(shell echo $${KESTRA_WORKER_THREAD:-4})
VERSION := $(shell awk -F= '/^version=/ {gsub(/-SNAPSHOT/, "", $$2); gsub(/[[:space:]]/, "", $$2); print $$2}' gradle.properties)
VERSION := $(shell ./gradlew properties -q | awk '/^version:/ {print $$2}')
GIT_COMMIT := $(shell git rev-parse --short HEAD)
GIT_BRANCH := $(shell git rev-parse --abbrev-ref HEAD)
DATE := $(shell date --rfc-3339=seconds)
@@ -48,43 +48,38 @@ build-exec:
./gradlew -q executableJar --no-daemon --priority=normal
install: build-exec
@echo "Installing Kestra in ${KESTRA_BASEDIR}" ; \
KESTRA_BASEDIR="${KESTRA_BASEDIR}" ; \
mkdir -p "$${KESTRA_BASEDIR}/bin" "$${KESTRA_BASEDIR}/plugins" "$${KESTRA_BASEDIR}/flows" "$${KESTRA_BASEDIR}/logs" ; \
echo "Copying executable..." ; \
EXECUTABLE_FILE=$$(ls build/executable/kestra-* 2>/dev/null | head -n1) ; \
if [ -z "$${EXECUTABLE_FILE}" ]; then \
echo "[ERROR] No Kestra executable found in build/executable"; \
exit 1; \
fi ; \
cp "$${EXECUTABLE_FILE}" "$${KESTRA_BASEDIR}/bin/kestra" ; \
chmod +x "$${KESTRA_BASEDIR}/bin/kestra" ; \
VERSION_INSTALLED=$$("$${KESTRA_BASEDIR}/bin/kestra" --version 2>/dev/null || echo "unknown") ; \
echo "Kestra installed successfully (version=$${VERSION_INSTALLED}) 🚀"
echo "Installing Kestra: ${KESTRA_BASEDIR}"
mkdir -p ${KESTRA_BASEDIR}/bin ${KESTRA_BASEDIR}/plugins ${KESTRA_BASEDIR}/flows ${KESTRA_BASEDIR}/logs
cp build/executable/* ${KESTRA_BASEDIR}/bin/kestra && chmod +x ${KESTRA_BASEDIR}/bin
VERSION_INSTALLED=$$(${KESTRA_BASEDIR}/bin/kestra --version); \
echo "Kestra installed successfully (version=$$VERSION_INSTALLED) 🚀"
# Install plugins for Kestra from the API.
# Install plugins for Kestra from (.plugins file).
install-plugins:
@echo "Installing plugins for Kestra version ${VERSION}" ; \
if [ -z "${VERSION}" ]; then \
echo "[ERROR] Kestra version could not be determined."; \
if [[ ! -f ".plugins" && ! -f ".plugins.override" ]]; then \
echo "[ERROR] file '$$(pwd)/.plugins' and '$$(pwd)/.plugins.override' not found."; \
exit 1; \
fi ; \
PLUGINS_PATH="${KESTRA_BASEDIR}/plugins" ; \
echo "Fetching plugin list from Kestra API for version ${VERSION}..." ; \
RESPONSE=$$(curl -s "https://api.kestra.io/v1/plugins/artifacts/core-compatibility/${VERSION}/latest") ; \
if [ -z "$${RESPONSE}" ]; then \
echo "[ERROR] Failed to fetch plugin list from API."; \
exit 1; \
fi ; \
echo "Parsing plugin list (excluding EE and secret plugins)..." ; \
echo "$${RESPONSE}" | jq -r '.[] | select(.license == "OPEN_SOURCE" and (.groupId != "io.kestra.plugin.ee") and (.groupId != "io.kestra.ee.secret")) | .groupId + ":" + .artifactId + ":" + .version' | while read -r plugin; do \
[[ $$plugin =~ ^#.* ]] && continue ; \
CURRENT_PLUGIN=$${plugin} ; \
echo "Installing $$CURRENT_PLUGIN..." ; \
fi; \
PLUGIN_LIST="./.plugins"; \
if [[ -f ".plugins.override" ]]; then \
PLUGIN_LIST="./.plugins.override"; \
fi; \
while IFS= read -r plugin; do \
[[ $$plugin =~ ^#.* ]] && continue; \
PLUGINS_PATH="${KESTRA_INSTALL_DIR}/plugins"; \
CURRENT_PLUGIN=$${plugin/LATEST/"${VERSION}"}; \
CURRENT_PLUGIN=$$(echo $$CURRENT_PLUGIN | cut -d':' -f2-); \
PLUGIN_FILE="$$PLUGINS_PATH/$$(echo $$CURRENT_PLUGIN | awk -F':' '{print $$2"-"$$3}').jar"; \
echo "Installing Kestra plugin $$CURRENT_PLUGIN > ${KESTRA_INSTALL_DIR}/plugins"; \
if [ -f "$$PLUGIN_FILE" ]; then \
echo "Plugin already installed in > $$PLUGIN_FILE"; \
else \
${KESTRA_BASEDIR}/bin/kestra plugins install $$CURRENT_PLUGIN \
--plugins ${KESTRA_BASEDIR}/plugins \
--repositories=https://central.sonatype.com/repository/maven-snapshots || exit 1 ; \
done
--plugins ${KESTRA_BASEDIR}/plugins \
--repositories=https://central.sonatype.com/repository/maven-snapshots || exit 1; \
fi \
done < $$PLUGIN_LIST
# Build docker image from Kestra source.
build-docker: build-exec

View File

@@ -331,7 +331,7 @@ subprojects {
}
dependencies {
agent "org.aspectj:aspectjweaver:1.9.25"
agent "org.aspectj:aspectjweaver:1.9.24"
}
test {

View File

@@ -1,9 +1,7 @@
package io.kestra.cli.commands.servers;
import com.google.common.collect.ImmutableMap;
import io.kestra.cli.services.TenantIdSelectorService;
import io.kestra.core.models.ServerType;
import io.kestra.core.repositories.LocalFlowRepositoryLoader;
import io.kestra.core.runners.ExecutorInterface;
import io.kestra.core.services.SkipExecutionService;
import io.kestra.core.services.StartExecutorService;
@@ -12,8 +10,6 @@ import io.micronaut.context.ApplicationContext;
import jakarta.inject.Inject;
import picocli.CommandLine;
import java.io.File;
import java.io.IOException;
import java.util.Collections;
import java.util.List;
import java.util.Map;
@@ -23,9 +19,6 @@ import java.util.Map;
description = "Start the Kestra executor"
)
public class ExecutorCommand extends AbstractServerCommand {
@CommandLine.Spec
CommandLine.Model.CommandSpec spec;
@Inject
private ApplicationContext applicationContext;
@@ -35,28 +28,22 @@ public class ExecutorCommand extends AbstractServerCommand {
@Inject
private StartExecutorService startExecutorService;
@CommandLine.Option(names = {"-f", "--flow-path"}, description = "Tenant identifier required to load flows from the specified path")
private File flowPath;
@CommandLine.Option(names = "--tenant", description = "Tenant identifier, Required to load flows from path")
private String tenantId;
@CommandLine.Option(names = {"--skip-executions"}, split=",", description = "List of execution IDs to skip, separated by commas; for troubleshooting only")
@CommandLine.Option(names = {"--skip-executions"}, split=",", description = "The list of execution identifiers to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipExecutions = Collections.emptyList();
@CommandLine.Option(names = {"--skip-flows"}, split=",", description = "List of flow identifiers (tenant|namespace|flowId) to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-flows"}, split=",", description = "The list of flow identifiers (tenant|namespace|flowId) to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipFlows = Collections.emptyList();
@CommandLine.Option(names = {"--skip-namespaces"}, split=",", description = "List of namespace identifiers (tenant|namespace) to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-namespaces"}, split=",", description = "The list of namespace identifiers (tenant|namespace) to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipNamespaces = Collections.emptyList();
@CommandLine.Option(names = {"--skip-tenants"}, split=",", description = "List of tenants to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-tenants"}, split=",", description = "The list of tenants to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipTenants = Collections.emptyList();
@CommandLine.Option(names = {"--start-executors"}, split=",", description = "List of Kafka Stream executors to start, separated by a command. Use it only with the Kafka queue; for debugging only")
@CommandLine.Option(names = {"--start-executors"}, split=",", description = "The list of Kafka Stream executors to start, separated by a command. Use it only with the Kafka queue, for debugging purpose.")
private List<String> startExecutors = Collections.emptyList();
@CommandLine.Option(names = {"--not-start-executors"}, split=",", description = "Lst of Kafka Stream executors to not start, separated by a command. Use it only with the Kafka queue; for debugging only")
@CommandLine.Option(names = {"--not-start-executors"}, split=",", description = "The list of Kafka Stream executors to not start, separated by a command. Use it only with the Kafka queue, for debugging purpose.")
private List<String> notStartExecutors = Collections.emptyList();
@SuppressWarnings("unused")
@@ -77,16 +64,6 @@ public class ExecutorCommand extends AbstractServerCommand {
super.call();
if (flowPath != null) {
try {
LocalFlowRepositoryLoader localFlowRepositoryLoader = applicationContext.getBean(LocalFlowRepositoryLoader.class);
TenantIdSelectorService tenantIdSelectorService = applicationContext.getBean(TenantIdSelectorService.class);
localFlowRepositoryLoader.load(tenantIdSelectorService.getTenantId(this.tenantId), this.flowPath);
} catch (IOException e) {
throw new CommandLine.ParameterException(this.spec.commandLine(), "Invalid flow path", e);
}
}
ExecutorInterface executorService = applicationContext.getBean(ExecutorInterface.class);
executorService.run();

View File

@@ -23,7 +23,7 @@ public class IndexerCommand extends AbstractServerCommand {
@Inject
private SkipExecutionService skipExecutionService;
@CommandLine.Option(names = {"--skip-indexer-records"}, split=",", description = "a list of indexer record keys, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-indexer-records"}, split=",", description = "a list of indexer record keys, separated by a coma; for troubleshooting purpose only")
private List<String> skipIndexerRecords = Collections.emptyList();
@SuppressWarnings("unused")

View File

@@ -42,7 +42,7 @@ public class StandAloneCommand extends AbstractServerCommand {
@Nullable
private FileChangedEventListener fileWatcher;
@CommandLine.Option(names = {"-f", "--flow-path"}, description = "Tenant identifier required to load flows from the specified path")
@CommandLine.Option(names = {"-f", "--flow-path"}, description = "the flow path containing flow to inject at startup (when running with a memory flow repository)")
private File flowPath;
@CommandLine.Option(names = "--tenant", description = "Tenant identifier, Required to load flows from path with the enterprise edition")
@@ -51,19 +51,19 @@ public class StandAloneCommand extends AbstractServerCommand {
@CommandLine.Option(names = {"--worker-thread"}, description = "the number of worker threads, defaults to eight times the number of available processors. Set it to 0 to avoid starting a worker.")
private int workerThread = defaultWorkerThread();
@CommandLine.Option(names = {"--skip-executions"}, split=",", description = "a list of execution identifiers to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-executions"}, split=",", description = "a list of execution identifiers to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipExecutions = Collections.emptyList();
@CommandLine.Option(names = {"--skip-flows"}, split=",", description = "a list of flow identifiers (namespace.flowId) to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-flows"}, split=",", description = "a list of flow identifiers (namespace.flowId) to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipFlows = Collections.emptyList();
@CommandLine.Option(names = {"--skip-namespaces"}, split=",", description = "a list of namespace identifiers (tenant|namespace) to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-namespaces"}, split=",", description = "a list of namespace identifiers (tenant|namespace) to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipNamespaces = Collections.emptyList();
@CommandLine.Option(names = {"--skip-tenants"}, split=",", description = "a list of tenants to skip, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-tenants"}, split=",", description = "a list of tenants to skip, separated by a coma; for troubleshooting purpose only")
private List<String> skipTenants = Collections.emptyList();
@CommandLine.Option(names = {"--skip-indexer-records"}, split=",", description = "a list of indexer record keys, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-indexer-records"}, split=",", description = "a list of indexer record keys, separated by a coma; for troubleshooting purpose only")
private List<String> skipIndexerRecords = Collections.emptyList();
@CommandLine.Option(names = {"--no-tutorials"}, description = "Flag to disable auto-loading of tutorial flows.")

View File

@@ -40,7 +40,7 @@ public class WebServerCommand extends AbstractServerCommand {
@Option(names = {"--no-indexer"}, description = "Flag to disable starting an embedded indexer.")
private boolean indexerDisabled = false;
@CommandLine.Option(names = {"--skip-indexer-records"}, split=",", description = "a list of indexer record keys, separated by a coma; for troubleshooting only")
@CommandLine.Option(names = {"--skip-indexer-records"}, split=",", description = "a list of indexer record keys, separated by a coma; for troubleshooting purpose only")
private List<String> skipIndexerRecords = Collections.emptyList();
@Override

View File

@@ -8,7 +8,6 @@ import io.kestra.core.validations.Regex;
import io.swagger.v3.oas.annotations.media.Schema;
import jakarta.validation.ConstraintViolationException;
import jakarta.validation.constraints.NotNull;
import jakarta.validation.constraints.Size;
import lombok.Builder;
import lombok.Getter;
import lombok.NoArgsConstructor;
@@ -28,7 +27,6 @@ public class SelectInput extends Input<String> implements RenderableInput {
@Schema(
title = "List of values."
)
@Size(min = 2)
List<@Regex String> values;
@Schema(

View File

@@ -10,8 +10,6 @@ public interface FlowTopologyRepositoryInterface {
List<FlowTopology> findByNamespace(String tenantId, String namespace);
List<FlowTopology> findByNamespacePrefix(String tenantId, String namespacePrefix);
List<FlowTopology> findAll(String tenantId);
FlowTopology save(FlowTopology flowTopology);

View File

@@ -548,8 +548,6 @@ public class FlowService {
var flowTopologies = flowTopologyRepository.get().findByFlow(tenantId, namespace, id, destinationOnly);
var visitedNodes = new ArrayList<String>();
visitedNodes.add(id);
return flowTopologies.stream()
// ignore already visited topologies
.filter(x -> !visitedTopologies.contains(x.uid()))
@@ -557,13 +555,8 @@ public class FlowService {
visitedTopologies.add(topology.uid());
Stream<FlowTopology> subTopologies = Stream
.of(topology.getDestination(), topology.getSource())
// ignore already visited nodes
.filter(x -> !visitedNodes.contains(x.getId()))
// recursively visit children and parents nodes
.flatMap(relationNode -> {
visitedNodes.add(relationNode.getId());
return recursiveFlowTopology(visitedTopologies, relationNode.getTenantId(), relationNode.getNamespace(), relationNode.getId(), destinationOnly);
});
.flatMap(relationNode -> recursiveFlowTopology(visitedTopologies, relationNode.getTenantId(), relationNode.getNamespace(), relationNode.getId(), destinationOnly));
return Stream.concat(Stream.of(topology), subTopologies);
});
}

View File

@@ -360,41 +360,4 @@ public interface StorageInterface extends AutoCloseable, Plugin {
return path;
}
/**
* Ensures the object name length does not exceed the allowed maximum.
* If it does, the object name is truncated and a short random prefix is added
* to avoid potential name collisions.
*
* @param uri the URI of the object
* @param maxObjectNameLength the maximum allowed length for the object name
* @return a normalized URI respecting the length limit
* @throws IOException if the URI cannot be rebuilt
*/
default URI limit(URI uri, int maxObjectNameLength) throws IOException {
if (uri == null) {
return null;
}
String path = uri.getPath();
String objectName = path.contains("/") ? path.substring(path.lastIndexOf("/") + 1) : path;
if (objectName.length() > maxObjectNameLength) {
objectName = objectName.substring(objectName.length() - maxObjectNameLength + 6);
String prefix = org.apache.commons.lang3.RandomStringUtils.secure()
.nextAlphanumeric(5)
.toLowerCase();
String newPath = (path.contains("/") ? path.substring(0, path.lastIndexOf("/") + 1) : "")
+ prefix + "-" + objectName;
try {
return new URI(uri.getScheme(), uri.getHost(), newPath, uri.getFragment());
} catch (java.net.URISyntaxException e) {
throw new IOException(e);
}
}
return uri;
}
}

View File

@@ -70,7 +70,7 @@ public class FlowTopologyService {
}
public FlowTopologyGraph namespaceGraph(String tenantId, String namespace) {
List<FlowTopology> flowTopologies = flowTopologyRepository.findByNamespacePrefix(tenantId, namespace);
List<FlowTopology> flowTopologies = flowTopologyRepository.findByNamespace(tenantId, namespace);
FlowTopologyGraph graph = this.graph(flowTopologies.stream(), (flowNode -> flowNode));

View File

@@ -206,22 +206,22 @@ import static io.kestra.core.utils.Rethrow.throwPredicate;
tasks:
- id: send_alert
type: io.kestra.plugin.notifications.sentry.SentryExecution
executionId: "{{ trigger.executionId }}"
transaction: "/execution/id/{{ trigger.executionId }}"
dsn: "{{ secret('SENTRY_DSN') }}"
level: ERROR
executionId: "{{ trigger.executionId }}"
transaction: "/execution/id/{{ trigger.executionId }}"
dsn: "{{ secret('SENTRY_DSN') }}"
level: ERROR
triggers:
- id: failed_prod_workflows
type: io.kestra.plugin.core.trigger.Flow
conditions:
- type: io.kestra.plugin.core.condition.ExecutionStatus
in:
- FAILED
- WARNING
- type: io.kestra.plugin.core.condition.ExecutionNamespace
namespace: company.payroll
prefix: false"""
type: io.kestra.plugin.core.trigger.Flow
conditions:
- type: io.kestra.plugin.core.condition.ExecutionStatus
in:
- FAILED
- WARNING
- type: io.kestra.plugin.core.condition.ExecutionNamespace
namespace: company.payroll
prefix: false"""
)
},

View File

@@ -35,7 +35,7 @@ import jakarta.validation.constraints.Size;
description = """
Webhook trigger allows you to create a unique URL that you can use to trigger a Kestra flow execution based on events in another application such as GitHub or Amazon EventBridge. In order to use that URL, you have to add a secret key to secure your webhook URL.
The URL will then follow the following format: `https://{your_hostname}/api/v1/{tenant}/executions/webhook/{namespace}/{flowId}/{key}`. Replace the templated values according to your workflow setup.
The URL will then follow the following format: `https://{your_hostname}/api/v1/executions/webhook/{namespace}/{flowId}/{key}`. Replace the templated values according to your workflow setup.
The webhook URL accepts `GET`, `POST`, and `PUT` requests.
@@ -85,7 +85,7 @@ import jakarta.validation.constraints.Size;
@Plugin(
examples = {
@Example(
title = "Add a webhook trigger to the current flow with the key `4wjtkzwVGBM9yKnjm3yv8r`; the webhook will be available at the URI `/api/v1/{tenant}/executions/webhook/{namespace}/{flowId}/4wjtkzwVGBM9yKnjm3yv8r`.",
title = "Add a webhook trigger to the current flow with the key `4wjtkzwVGBM9yKnjm3yv8r`; the webhook will be available at the URI `/api/v1/executions/webhook/{namespace}/{flowId}/4wjtkzwVGBM9yKnjm3yv8r`.",
code = """
id: webhook_flow
namespace: company.team

View File

@@ -1,31 +1,6 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="debug=Return">
<g id="Bkg" filter="url(#filter0_i_1386_1709)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1709)"/>
</g>
<g clip-path="url(#paint1_angular_1386_1709_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
<g id="bug-check-outline">
<path id="Union" d="M41.2998 36.0703L38.8701 38.5H46V43.5H44.5V40H38.8701L41.2998 42.4297L40.2402 43.5L36 39.25L40.2402 35L41.2998 36.0703ZM37.333 14.8252L34.5488 17.542C35.8299 18.3961 36.8895 19.5915 37.6582 20.958H42.458V24.375H38.8877C38.9902 24.9386 39.042 25.5023 39.042 26.083V27.792H42.458V31.208H39.042V31.8066C37.7438 32.2678 36.5479 32.9849 35.54 33.9072C35.6253 33.5829 35.625 33.2414 35.625 32.917V26.083C35.6248 22.3079 32.5671 19.2502 28.792 19.25C25.0167 19.25 21.9582 22.3078 21.958 26.083V32.917C21.9581 36.6923 25.0167 39.75 28.792 39.75C30.1585 39.7499 31.5083 39.34 32.6357 38.5713C32.3624 39.5107 32.2081 40.4844 32.208 41.458V42.5684C27.5615 44.2082 22.4029 42.3125 19.9258 38.042H15.125V34.625H18.6953C18.5928 34.0614 18.542 33.4977 18.542 32.917V31.208H15.125V27.792H18.542V26.083C18.542 25.5023 18.5928 24.9386 18.6953 24.375H15.125V20.958H19.9258C20.6945 19.6086 21.7365 18.3961 23.0176 17.542L20.25 14.8252L22.6592 12.417L26.3828 16.124C27.1686 15.9361 27.9378 15.833 28.792 15.833C29.646 15.833 30.432 15.9361 31.2178 16.124L34.9248 12.417L37.333 14.8252ZM32.208 34.625H25.375V31.208H32.208V34.625ZM32.208 27.792H25.375V24.375H32.208V27.792Z" fill="url(#paint2_linear_1386_1709)"/>
</g>
</g>
<defs>
<filter id="filter0_i_1386_1709" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1709"/>
</filter>
<clipPath id="paint1_angular_1386_1709_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1709" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1709" x1="32.4922" y1="35.7292" x2="32.4922" y2="10.4743" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path d="M5 21H19C20.1 21 21 20.1 21 19V5C21 3.9 20.1 3 19 3H5C3.9 3 3 3.9 3 5V19C3 20.1 3.9 21 5 21ZM5 5H19V19H5V5Z" fill="currentColor"/>
<path d="M17.19 12C17.19 10.5913 16.5227 9.25676 15.4106 8.29291L14.2243 8.73776C15.1881 9.47918 15.7071 10.7396 15.7071 12C15.7071 13.2604 15.1881 14.5208 14.2243 15.2623L15.4106 15.7071C16.5227 14.7433 17.19 13.4087 17.19 12Z" fill="currentColor"/>
<path d="M9.77575 8.73776L8.58947 8.29291C7.47734 9.25676 6.81006 10.5913 6.81006 12C6.81006 13.4087 7.47734 14.7433 8.58947 15.7071L9.77575 15.2623C8.8119 14.5208 8.2929 13.2604 8.2929 12C8.2929 10.7396 8.8119 9.47918 9.77575 8.73776Z" fill="currentColor"/>
<path d="M13.1121 9.77575L12.0742 11.2586L11.481 9.77575H10.5172L11.481 11.9259L9.77575 14.2243H10.962L12 12.7414L12.5931 14.2243H13.557L12.5931 12L14.2243 9.77575H13.1121Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 5.2 KiB

After

Width:  |  Height:  |  Size: 957 B

View File

@@ -1,31 +1,4 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="debug=Main">
<g id="Bkg" filter="url(#filter0_i_1386_1701)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1701)"/>
</g>
<g clip-path="url(#paint1_angular_1386_1701_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
<g id="bug-outline">
<path id="Vector" d="M42.1666 20.6667H37.3662C36.5975 19.3 35.5383 18.1042 34.2571 17.25L37.0416 14.5337L34.6329 12.125L30.9258 15.8321C30.14 15.6442 29.3541 15.5417 28.5 15.5417C27.6458 15.5417 26.8771 15.6442 26.0912 15.8321L22.3671 12.125L19.9583 14.5337L22.7258 17.25C21.4446 18.1042 20.4025 19.3171 19.6337 20.6667H14.8333V24.0833H18.4037C18.3012 24.6471 18.25 25.2108 18.25 25.7917V27.5H14.8333V30.9167H18.25V32.625C18.25 33.2058 18.3012 33.7696 18.4037 34.3333H14.8333V37.75H19.6337C22.4696 42.6529 28.7391 44.31 33.625 41.4742C35.1796 40.5858 36.4779 39.2875 37.3662 37.75H42.1666V34.3333H38.5962C38.6987 33.7696 38.75 33.2058 38.75 32.625V30.9167H42.1666V27.5H38.75V25.7917C38.75 25.2108 38.6987 24.6471 38.5962 24.0833H42.1666V20.6667ZM35.3333 32.625C35.3333 34.4373 34.6134 36.1754 33.3319 37.4569C32.0504 38.7384 30.3123 39.4583 28.5 39.4583C26.6877 39.4583 24.9496 38.7384 23.6681 37.4569C22.3866 36.1754 21.6666 34.4373 21.6666 32.625V25.7917C21.6666 23.9794 22.3866 22.2413 23.6681 20.9598C24.9496 19.6783 26.6877 18.9583 28.5 18.9583C30.3123 18.9583 32.0504 19.6783 33.3319 20.9598C34.6134 22.2413 35.3333 23.9794 35.3333 25.7917V32.625ZM31.9166 24.0833V27.5H25.0833V24.0833H31.9166ZM25.0833 30.9167H31.9166V34.3333H25.0833V30.9167Z" fill="url(#paint2_linear_1386_1701)"/>
</g>
</g>
<defs>
<filter id="filter0_i_1386_1701" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1701"/>
</filter>
<clipPath id="paint1_angular_1386_1701_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1701" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1701" x1="30.2083" y1="35.1752" x2="30.2083" y2="10.2041" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
</svg>
<svg fill="currentColor" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<path
d="M20,8H17.19C16.74,7.2 16.12,6.5 15.37,6L17,4.41L15.59,3L13.42,5.17C12.96,5.06 12.5,5 12,5C11.5,5 11.05,5.06 10.59,5.17L8.41,3L7,4.41L8.62,6C7.87,6.5 7.26,7.21 6.81,8H4V10H6.09C6.03,10.33 6,10.66 6,11V12H4V14H6V15C6,15.34 6.03,15.67 6.09,16H4V18H6.81C8.47,20.87 12.14,21.84 15,20.18C15.91,19.66 16.67,18.9 17.19,18H20V16H17.91C17.97,15.67 18,15.34 18,15V14H20V12H18V11C18,10.66 17.97,10.33 17.91,10H20V8M16,15A4,4 0 0,1 12,19A4,4 0 0,1 8,15V11A4,4 0 0,1 12,7A4,4 0 0,1 16,11V15M14,10V12H10V10H14M10,14H14V16H10V14Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 5.3 KiB

After

Width:  |  Height:  |  Size: 645 B

View File

@@ -1,50 +0,0 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="kv=Delete">
<g id="Bkg" filter="url(#filter0_i_1386_1687)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1687)"/>
</g>
<g id="KV">
<path id="Vector" d="M32.5532 24L28 14H30.7801L33.6454 20.6525L36.5106 14H39.1915L34.6241 24H32.5532Z" fill="url(#paint1_linear_1386_1687)"/>
<path id="Vector_2" d="M17 24V14H19.6099V18.2837H19.6383L23.5248 14H26.617L22.1631 18.7518L26.773 24H23.6667L19.6383 19.4894H19.6099V24H17Z" fill="url(#paint2_linear_1386_1687)"/>
</g>
<g clip-path="url(#paint3_angular_1386_1687_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
<g id="Vector_3">
<path d="M47.3949 43.2913L43.7056 46.9807L28.9483 32.2233L32.6376 28.534L47.3949 43.2913Z" fill="url(#paint4_linear_1386_1687)"/>
<path d="M32.6376 46.9807L28.9483 43.2913L43.7056 28.534L47.3949 32.2233L32.6376 46.9807Z" fill="url(#paint5_linear_1386_1687)"/>
</g>
</g>
<defs>
<filter id="filter0_i_1386_1687" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1687"/>
</filter>
<clipPath id="paint3_angular_1386_1687_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1687" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1687" x1="28.7989" y1="16.9796" x2="29.988" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1687" x1="26.1471" y1="16.9796" x2="27.3362" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint4_linear_1386_1687" x1="35.4046" y1="42.369" x2="47.3949" y2="30.3786" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
<linearGradient id="paint5_linear_1386_1687" x1="35.4046" y1="42.369" x2="47.3949" y2="30.3786" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
</svg>

Before

Width:  |  Height:  |  Size: 5.3 KiB

View File

@@ -1,45 +1,4 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="kv=Get">
<g id="Bkg" filter="url(#filter0_i_1386_1587)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1587)"/>
</g>
<g id="KV">
<path id="Vector" d="M32.5532 24L28 14H30.7801L33.6454 20.6525L36.5106 14H39.1915L34.6241 24H32.5532Z" fill="url(#paint1_linear_1386_1587)"/>
<path id="Vector_2" d="M17 24V14H19.6099V18.2837H19.6383L23.5248 14H26.617L22.1631 18.7518L26.773 24H23.6667L19.6383 19.4894H19.6099V24H17Z" fill="url(#paint2_linear_1386_1587)"/>
</g>
<g id="download">
<path id="Vector_3" d="M31 46H45V44H31M45 35H41V29H35V35H31L38 42L45 35Z" fill="url(#paint3_linear_1386_1587)"/>
</g>
<g clip-path="url(#paint4_angular_1386_1587_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1587" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1587"/>
</filter>
<clipPath id="paint4_angular_1386_1587_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1587" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1587" x1="28.7989" y1="16.9796" x2="29.988" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1587" x1="26.1471" y1="16.9796" x2="27.3362" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint3_linear_1386_1587" x1="38.875" y1="41.75" x2="38.875" y2="27.9375" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path d="M21.5 19.5L18.5 22.5L15.5 19.5H17.5V15.5H19.5V19.5H21.5Z" fill="currentColor"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M4.97119 2.03955C3.86662 2.03955 2.97119 2.93498 2.97119 4.03955V19.9785C2.97119 21.0831 3.86662 21.9785 4.97119 21.9785H13.7188C13.2039 21.0924 12.9091 20.0627 12.9091 18.964C12.9091 15.6454 15.5994 12.9551 18.918 12.9551C19.6608 12.9551 20.3721 13.0899 21.0288 13.3363V4.03955C21.0288 2.93498 20.1334 2.03955 19.0288 2.03955H4.97119ZM13.5582 10.54L12.1582 5.42804H13.4542L14.4582 9.60804H14.5502L15.5582 5.42804H16.8502L15.4542 10.54H13.5582ZM8.57785 8.30004L10.0458 10.54H11.4978L9.72544 7.86056L11.4418 5.42804H10.0258L8.57785 7.54804H8.38985V5.42804H7.14985V10.54H8.38985V8.30004H8.57785Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 5.0 KiB

After

Width:  |  Height:  |  Size: 866 B

View File

@@ -1,45 +1,3 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="kv=Get&#226;&#128;&#139;Keys">
<g id="Bkg" filter="url(#filter0_i_1386_1593)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1593)"/>
</g>
<g id="KV">
<path id="Vector" d="M32.5532 24L28 14H30.7801L33.6454 20.6525L36.5106 14H39.1915L34.6241 24H32.5532Z" fill="url(#paint1_linear_1386_1593)"/>
<path id="Vector_2" d="M17 24V14H19.6099V18.2837H19.6383L23.5248 14H26.617L22.1631 18.7518L26.773 24H23.6667L19.6383 19.4894H19.6099V24H17Z" fill="url(#paint2_linear_1386_1593)"/>
</g>
<g id="database-search-outline">
<path id="Vector_3" d="M37 44.95C33.77 44.72 32 43.45 32 43V40.77C33.13 41.32 34.5 41.69 36 41.87C36 41.21 36.04 40.54 36.21 39.89C34.5 39.67 32.97 39.16 32 38.45V35.64C33.43 36.45 35.5 36.97 37.82 37C37.85 36.97 37.87 36.93 37.9 36.9C40.1 34.71 43.5 34.41 46 36.03V33C46 30.79 42.42 29 38 29C33.58 29 30 30.79 30 33V43C30 45.21 33.59 47 38 47C38.34 47 38.68 47 39 46.97C38.62 46.72 38.24 46.44 37.9 46.1C37.55 45.74 37.25 45.36 37 44.95ZM38 31C41.87 31 44 32.5 44 33C44 33.5 41.87 35 38 35C34.13 35 32 33.5 32 33C32 32.5 34.13 31 38 31ZM46.31 43.9C46.75 43.21 47 42.38 47 41.5C47 39 45 37 42.5 37C40 37 38 39 38 41.5C38 44 40 46 42.5 46C43.37 46 44.19 45.75 44.88 45.32L48 48.39L49.39 47L46.31 43.9ZM42.5 44C41.12 44 40 42.88 40 41.5C40 40.12 41.12 39 42.5 39C43.88 39 45 40.12 45 41.5C45 42.88 43.88 44 42.5 44Z" fill="url(#paint3_linear_1386_1593)"/>
</g>
<g clip-path="url(#paint4_angular_1386_1593_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1593" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1593"/>
</filter>
<clipPath id="paint4_angular_1386_1593_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1593" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1593" x1="28.7989" y1="16.9796" x2="29.988" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1593" x1="26.1471" y1="16.9796" x2="27.3362" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint3_linear_1386_1593" x1="40.9069" y1="43.5425" x2="40.9069" y2="27.7881" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path fill-rule="evenodd" clip-rule="evenodd" d="M2.97852 4.03955C2.97852 2.93498 3.87395 2.03955 4.97852 2.03955H19.0288C20.1334 2.03955 21.0288 2.93498 21.0288 4.03955V19.957C21.0288 21.0616 20.1334 21.957 19.0288 21.957H4.97852C3.87395 21.957 2.97852 21.0616 2.97852 19.957V4.03955ZM7.00647 14.001C7.00647 13.4487 7.45419 13.001 8.00647 13.001C8.55875 13.001 9.00647 13.4487 9.00647 14.001C9.00647 14.5533 8.55875 15.001 8.00647 15.001C7.45419 15.001 7.00647 14.5533 7.00647 14.001ZM8.00647 16.9235C7.45419 16.9235 7.00647 17.3712 7.00647 17.9235C7.00647 18.4758 7.45419 18.9235 8.00647 18.9235C8.55875 18.9235 9.00647 18.4758 9.00647 17.9235C9.00647 17.3712 8.55875 16.9235 8.00647 16.9235ZM10.0037 14.001C10.0037 13.4487 10.4514 13.001 11.0037 13.001H15.988C16.5403 13.001 16.988 13.4487 16.988 14.001C16.988 14.5533 16.5403 15.001 15.988 15.001H11.0037C10.4514 15.001 10.0037 14.5533 10.0037 14.001ZM11.0037 16.9235C10.4514 16.9235 10.0037 17.3712 10.0037 17.9235C10.0037 18.4758 10.4514 18.9235 11.0037 18.9235H15.988C16.5403 18.9235 16.988 18.4758 16.988 17.9235C16.988 17.3712 16.5403 16.9235 15.988 16.9235H11.0037ZM13.5618 10.54L12.1618 5.42804H13.4578L14.4618 9.60804H14.5538L15.5618 5.42804H16.8538L15.4578 10.54H13.5618ZM8.5815 8.30004L10.0495 10.54H11.5015L9.7291 7.86056L11.4455 5.42804H10.0295L8.5815 7.54804H8.3935V5.42804H7.1535V10.54H8.3935V8.30004H8.5815Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 5.7 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

View File

@@ -1,45 +0,0 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="kv=Purge&#226;&#128;&#139;K&#226;&#128;&#139;V">
<g id="Bkg" filter="url(#filter0_i_1386_1599)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1599)"/>
</g>
<g id="delete-variant">
<path id="Vector" d="M47.03 29L44 46.31C43.83 47.27 43 48 42 48H34C33 48 32.17 47.27 32 46.31L28.97 29H47.03ZM31.36 31L34 46H42L44.64 31H31.36ZM35 44V40H39V44H35ZM39 39.18L35.82 36L39 32.82L42.18 36L39 39.18Z" fill="url(#paint1_linear_1386_1599)"/>
</g>
<g id="KV">
<path id="Vector_2" d="M32.5532 24L28 14H30.7801L33.6454 20.6525L36.5106 14H39.1915L34.6241 24H32.5532Z" fill="url(#paint2_linear_1386_1599)"/>
<path id="Vector_3" d="M17 24V14H19.6099V18.2837H19.6383L23.5248 14H26.617L22.1631 18.7518L26.773 24H23.6667L19.6383 19.4894H19.6099V24H17Z" fill="url(#paint3_linear_1386_1599)"/>
</g>
<g clip-path="url(#paint4_angular_1386_1599_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1599" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1599"/>
</filter>
<clipPath id="paint4_angular_1386_1599_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1599" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1599" x1="39.1287" y1="43.25" x2="39.1287" y2="27.8125" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1599" x1="28.7989" y1="16.9796" x2="29.988" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint3_linear_1386_1599" x1="26.1471" y1="16.9796" x2="27.3362" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
</defs>
</svg>

Before

Width:  |  Height:  |  Size: 5.2 KiB

View File

@@ -1,43 +1,4 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="kv=Set">
<g id="Bkg" filter="url(#filter0_i_1386_1605)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1605)"/>
</g>
<path id="Vector" d="M45.8388 31.7499C46.1564 31.3317 46.0799 30.724 45.6729 30.4318L43.2165 28.5663C42.8257 28.2526 42.2198 28.342 41.9022 28.7603L40.4024 30.7242L44.339 33.7138M30.5324 43.7314L30.9735 47.237L34.4689 46.721L43.4757 34.8506L39.5391 31.861L30.5324 43.7314Z" fill="url(#paint1_linear_1386_1605)"/>
<g id="KV">
<path id="Vector_2" d="M32.5532 24L28 14H30.7801L33.6454 20.6525L36.5106 14H39.1915L34.6241 24H32.5532Z" fill="url(#paint2_linear_1386_1605)"/>
<path id="Vector_3" d="M17 24V14H19.6099V18.2837H19.6383L23.5248 14H26.617L22.1631 18.7518L26.773 24H23.6667L19.6383 19.4894H19.6099V24H17Z" fill="url(#paint3_linear_1386_1605)"/>
</g>
<g clip-path="url(#paint4_angular_1386_1605_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1605" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1605"/>
</filter>
<clipPath id="paint4_angular_1386_1605_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1605" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1605" x1="39.534" y1="38.0057" x2="45.0867" y2="29.2374" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1605" x1="28.7989" y1="16.9796" x2="29.988" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint3_linear_1386_1605" x1="26.1471" y1="16.9796" x2="27.3362" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
</defs>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path fill-rule="evenodd" clip-rule="evenodd" d="M4.97119 2.03955C3.86662 2.03955 2.97119 2.93498 2.97119 4.03955V19.9785C2.97119 21.0831 3.86662 21.9785 4.97119 21.9785H13.7188C13.2039 21.0924 12.9091 20.0627 12.9091 18.964C12.9091 15.6454 15.5994 12.9551 18.918 12.9551C19.6608 12.9551 20.3721 13.0899 21.0288 13.3363V4.03955C21.0288 2.93498 20.1334 2.03955 19.0288 2.03955H4.97119ZM13.5582 10.54L12.1582 5.42804H13.4542L14.4582 9.60804H14.5502L15.5582 5.42804H16.8502L15.4542 10.54H13.5582ZM8.57785 8.30004L10.0458 10.54H11.4978L9.72544 7.86056L11.4418 5.42804H10.0258L8.57785 7.54804H8.38984V5.42804H7.14985V10.54H8.38984V8.30004H8.57785Z" fill="currentColor"/>
<path d="M19.7985 18.4785H21.7985L18.7985 15.4785L15.7985 18.4785H17.7985V22.4785H19.7985V18.4785Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 5.1 KiB

After

Width:  |  Height:  |  Size: 899 B

View File

@@ -1,45 +1,3 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="kv=Main">
<g id="Bkg" filter="url(#filter0_i_1386_1574)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1574)"/>
</g>
<g id="database">
<path id="Vector" d="M38 29C33.58 29 30 30.79 30 33C30 35.21 33.58 37 38 37C42.42 37 46 35.21 46 33C46 30.79 42.42 29 38 29ZM30 35V38C30 40.21 33.58 42 38 42C42.42 42 46 40.21 46 38V35C46 37.21 42.42 39 38 39C33.58 39 30 37.21 30 35ZM30 40V43C30 45.21 33.58 47 38 47C42.42 47 46 45.21 46 43V40C46 42.21 42.42 44 38 44C33.58 44 30 42.21 30 40Z" fill="url(#paint1_linear_1386_1574)"/>
</g>
<g id="KV">
<path id="Vector_2" d="M32.5532 24L28 14H30.7801L33.6454 20.6525L36.5106 14H39.1915L34.6241 24H32.5532Z" fill="url(#paint2_linear_1386_1574)"/>
<path id="Vector_3" d="M17 24V14H19.6099V18.2837H19.6383L23.5248 14H26.617L22.1631 18.7518L26.773 24H23.6667L19.6383 19.4894H19.6099V24H17Z" fill="url(#paint3_linear_1386_1574)"/>
</g>
<g clip-path="url(#paint4_angular_1386_1574_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1574" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1574"/>
</filter>
<clipPath id="paint4_angular_1386_1574_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1574" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1574" x1="39" y1="42.5" x2="39" y2="27.875" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
<linearGradient id="paint2_linear_1386_1574" x1="28.7989" y1="16.9796" x2="29.988" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
<linearGradient id="paint3_linear_1386_1574" x1="26.1471" y1="16.9796" x2="27.3362" y2="28.4007" gradientUnits="userSpaceOnUse">
<stop stop-color="#EEDBFF"/>
<stop offset="0.774038" stop-color="#FBF8FF"/>
<stop offset="1" stop-color="#F7E7FF"/>
</linearGradient>
</defs>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path fill-rule="evenodd" clip-rule="evenodd" d="M2.97852 4.03955C2.97852 2.93498 3.87395 2.03955 4.97852 2.03955H19.0288C20.1334 2.03955 21.0288 2.93498 21.0288 4.03955V19.957C21.0288 21.0616 20.1334 21.957 19.0288 21.957H4.97852C3.87395 21.957 2.97852 21.0616 2.97852 19.957V4.03955ZM13.5618 14.5L12.1618 9.388H13.4578L14.4618 13.568H14.5538L15.5618 9.388H16.8538L15.4578 14.5H13.5618ZM8.5815 12.26L10.0495 14.5H11.5015L9.7291 11.8205L11.4455 9.388H10.0295L8.5815 11.508H8.3935V9.388H7.1535V14.5H8.3935V12.26H8.5815Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 5.2 KiB

After

Width:  |  Height:  |  Size: 653 B

View File

@@ -1,31 +1,6 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="output=Output&#226;&#128;&#139;Values">
<g id="Bkg" filter="url(#filter0_i_1386_1788)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1788)"/>
</g>
<g id="table-arrow-right">
<path id="Vector" d="M15.8334 14.125H39.75C41.6463 14.125 43.1667 15.6625 43.1667 17.5417V29.6367C40.5188 29.1925 37.905 29.8075 35.7867 31.2083H29.5V38.0417H31.345C31.1571 39.2033 31.1571 40.3479 31.345 41.4583H15.8334C13.9542 41.4583 12.4167 39.9379 12.4167 38.0417V17.5417C12.4167 15.6625 13.9542 14.125 15.8334 14.125ZM15.8334 20.9583V27.7917H26.0834V20.9583H15.8334ZM29.5 20.9583V27.7917H39.75V20.9583H29.5ZM15.8334 31.2083V38.0417H26.0834V31.2083H15.8334ZM42.21 44.875V41.4583H35.3767V38.0417H42.21V34.625L47.335 39.75L42.21 44.875Z" fill="url(#paint1_linear_1386_1788)"/>
</g>
<g clip-path="url(#paint2_angular_1386_1788_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1788" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1788"/>
</filter>
<clipPath id="paint2_angular_1386_1788_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1788" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1788" x1="32.0583" y1="37.1875" x2="32.0583" y2="12.2031" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
</svg>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path d="M5 21H19C20.1 21 21 20.1 21 19V5C21 3.9 20.1 3 19 3H5C3.9 3 3 3.9 3 5V19C3 20.1 3.9 21 5 21ZM5 5H19V19H5V5Z" fill="currentColor"/>
<path d="M17.19 12C17.19 10.5913 16.5227 9.25676 15.4106 8.29291L14.2243 8.73776C15.1881 9.47918 15.7071 10.7396 15.7071 12C15.7071 13.2604 15.1881 14.5208 14.2243 15.2623L15.4106 15.7071C16.5227 14.7433 17.19 13.4087 17.19 12Z" fill="currentColor"/>
<path d="M9.77575 8.73776L8.58947 8.29291C7.47734 9.25676 6.81006 10.5913 6.81006 12C6.81006 13.4087 7.47734 14.7433 8.58947 15.7071L9.77575 15.2623C8.8119 14.5208 8.2929 13.2604 8.2929 12C8.2929 10.7396 8.8119 9.47918 9.77575 8.73776Z" fill="currentColor"/>
<path d="M13.1121 9.77575L12.0742 11.2586L11.481 9.77575H10.5172L11.481 11.9259L9.77575 14.2243H10.962L12 12.7414L12.5931 14.2243H13.557L12.5931 12L14.2243 9.77575H13.1121Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 4.9 KiB

After

Width:  |  Height:  |  Size: 957 B

View File

@@ -1,31 +1,6 @@
<svg width="57" height="57" viewBox="0 0 57 57" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="output=Main">
<g id="Bkg" filter="url(#filter0_i_1386_1783)">
<path d="M0 14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57H14.25C6.37994 57 0 50.6201 0 42.75V14.25Z" fill="url(#paint0_linear_1386_1783)"/>
</g>
<g id="table">
<path id="Vector" d="M16.8577 14.2791H40.7744C41.6805 14.2791 42.5496 14.639 43.1903 15.2798C43.8311 15.9205 44.191 16.7896 44.191 17.6957V38.1957C44.191 39.1019 43.8311 39.9709 43.1903 40.6117C42.5496 41.2524 41.6805 41.6124 40.7744 41.6124H16.8577C15.9516 41.6124 15.0825 41.2524 14.4418 40.6117C13.801 39.9709 13.441 39.1019 13.441 38.1957V17.6957C13.441 16.7896 13.801 15.9205 14.4418 15.2798C15.0825 14.639 15.9516 14.2791 16.8577 14.2791ZM16.8577 21.1124V27.9457H27.1077V21.1124H16.8577ZM30.5244 21.1124V27.9457H40.7744V21.1124H30.5244ZM16.8577 31.3624V38.1957H27.1077V31.3624H16.8577ZM30.5244 31.3624V38.1957H40.7744V31.3624H30.5244Z" fill="url(#paint1_linear_1386_1783)"/>
</g>
<g clip-path="url(#paint2_angular_1386_1783_clip_path)" data-figma-skip-parse="true"><g transform="matrix(0 0.0285 -0.0285 0 28.5 28.5)"><foreignObject x="-1035.09" y="-1035.09" width="2070.18" height="2070.18"><div xmlns="http://www.w3.org/1999/xhtml" style="background:conic-gradient(from 90deg,rgba(181, 231, 255, 1) 0deg,rgba(241, 117, 255, 1) 72.6923deg,rgba(241, 117, 255, 0.1) 360deg);height:100%;width:100%;opacity:1"></div></foreignObject></g></g><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z" data-figma-gradient-fill="{&#34;type&#34;:&#34;GRADIENT_ANGULAR&#34;,&#34;stops&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;stopsVar&#34;:[{&#34;color&#34;:{&#34;r&#34;:0.71089994907379150,&#34;g&#34;:0.90845167636871338,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.0},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:1.0},&#34;position&#34;:0.20192307233810425},{&#34;color&#34;:{&#34;r&#34;:0.94509804248809814,&#34;g&#34;:0.45882353186607361,&#34;b&#34;:1.0,&#34;a&#34;:0.10000000149011612},&#34;position&#34;:1.0}],&#34;transform&#34;:{&#34;m00&#34;:3.4902435166328386e-15,&#34;m01&#34;:-57.0,&#34;m02&#34;:57.0,&#34;m10&#34;:57.0,&#34;m11&#34;:3.4902435166328386e-15,&#34;m12&#34;:0.0},&#34;opacity&#34;:1.0,&#34;blendMode&#34;:&#34;NORMAL&#34;,&#34;visible&#34;:true}"/>
</g>
<defs>
<filter id="filter0_i_1386_1783" x="0" y="0" width="57" height="59" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="1"/>
<feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.75 0"/>
<feBlend mode="normal" in2="shape" result="effect1_innerShadow_1386_1783"/>
</filter>
<clipPath id="paint2_angular_1386_1783_clip_path"><path id="Stroke" d="M42.75 54.625V57H14.25V54.625H42.75ZM54.625 42.75V14.25C54.625 7.69162 49.3084 2.375 42.75 2.375H14.25C7.69162 2.375 2.375 7.69162 2.375 14.25V42.75C2.375 49.3084 7.69162 54.625 14.25 54.625V57C6.37994 57 0 50.6201 0 42.75V14.25C0 6.37994 6.37994 0 14.25 0H42.75C50.6201 0 57 6.37994 57 14.25V42.75C57 50.6201 50.6201 57 42.75 57V54.625C49.3084 54.625 54.625 49.3084 54.625 42.75Z"/></clipPath><linearGradient id="paint0_linear_1386_1783" x1="52.25" y1="48.6875" x2="8.3125" y2="7.125" gradientUnits="userSpaceOnUse">
<stop stop-color="#01000B"/>
<stop offset="1" stop-color="#520188"/>
</linearGradient>
<linearGradient id="paint1_linear_1386_1783" x1="30.7379" y1="34.7791" x2="30.7379" y2="12.5707" gradientUnits="userSpaceOnUse">
<stop stop-color="#91F4FF"/>
<stop offset="0.5" stop-color="#D67EE2"/>
</linearGradient>
</defs>
</svg>
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path d="M5 21H19C20.1 21 21 20.1 21 19V5C21 3.9 20.1 3 19 3H5C3.9 3 3 3.9 3 5V19C3 20.1 3.9 21 5 21ZM5 5H19V19H5V5Z" fill="currentColor"/>
<path d="M17.19 12C17.19 10.5913 16.5227 9.25676 15.4106 8.29291L14.2243 8.73776C15.1881 9.47918 15.7071 10.7396 15.7071 12C15.7071 13.2604 15.1881 14.5208 14.2243 15.2623L15.4106 15.7071C16.5227 14.7433 17.19 13.4087 17.19 12Z" fill="currentColor"/>
<path d="M9.77575 8.73776L8.58947 8.29291C7.47734 9.25676 6.81006 10.5913 6.81006 12C6.81006 13.4087 7.47734 14.7433 8.58947 15.7071L9.77575 15.2623C8.8119 14.5208 8.2929 13.2604 8.2929 12C8.2929 10.7396 8.8119 9.47918 9.77575 8.73776Z" fill="currentColor"/>
<path d="M13.1121 9.77575L12.0742 11.2586L11.481 9.77575H10.5172L11.481 11.9259L9.77575 14.2243H10.962L12 12.7414L12.5931 14.2243H13.557L12.5931 12L14.2243 9.77575H13.1121Z" fill="currentColor"/>
</svg>

Before

Width:  |  Height:  |  Size: 4.9 KiB

After

Width:  |  Height:  |  Size: 957 B

View File

@@ -49,32 +49,6 @@ public abstract class AbstractFlowTopologyRepositoryTest {
assertThat(list.size()).isEqualTo(1);
}
@Test
void findByNamespacePrefix() {
String tenant = TestsUtils.randomTenant(this.getClass().getSimpleName());
flowTopologyRepository.save(
createSimpleFlowTopology(tenant, "flow-a", "flow-b", "io.kestra.tests")
);
flowTopologyRepository.save(
createSimpleFlowTopology(tenant, "flow-x", "flow-y", "io.kestra.tests.sub")
);
flowTopologyRepository.save(
createSimpleFlowTopology(tenant, "flow-p", "flow-q", "io.other.namespace")
);
List<FlowTopology> list = flowTopologyRepository.findByNamespacePrefix(tenant, "io.kestra.tests");
assertThat(list)
.extracting(ft -> ft.getSource().getNamespace())
.contains("io.kestra.tests", "io.kestra.tests.sub")
.doesNotContain("io.other.namespace");
assertThat(list.size()).isEqualTo(2);
}
@Test
void findByNamespace() {
String tenant = TestsUtils.randomTenant(this.getClass().getSimpleName());

View File

@@ -3,16 +3,12 @@ package io.kestra.core.topologies;
import io.kestra.core.exceptions.FlowProcessingException;
import io.kestra.core.junit.annotations.KestraTest;
import io.kestra.core.models.flows.FlowWithSource;
import io.kestra.core.models.topologies.FlowNode;
import io.kestra.core.models.topologies.FlowTopology;
import io.kestra.core.models.topologies.FlowTopologyGraph;
import io.kestra.core.repositories.FlowRepositoryInterface;
import io.kestra.core.repositories.FlowTopologyRepositoryInterface;
import io.kestra.core.services.FlowService;
import io.kestra.core.utils.IdUtils;
import io.kestra.plugin.core.execution.AssertTest;
import jakarta.inject.Inject;
import jakarta.validation.constraints.AssertTrue;
import org.jetbrains.annotations.NotNull;
import org.junit.jupiter.api.Test;
@@ -60,8 +56,15 @@ public class FlowTopologyTest {
// When
computeAndSaveTopologies(List.of(child, parent, unrelatedFlow));
System.out.println();
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
var dependencies = flowService.findDependencies(tenantId, "io.kestra.unittest", parent.getId(), false, true);
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
// Then
assertThat(dependencies.map(FlowTopologyTestData::of))
@@ -120,8 +123,16 @@ public class FlowTopologyTest {
// When
computeAndSaveTopologies(List.of(subChild, child, superParent, parent, unrelatedFlow));
System.out.println();
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
System.out.println();
var dependencies = flowService.findDependencies(tenantId, "io.kestra.unittest", parent.getId(), false, true);
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
// Then
assertThat(dependencies.map(FlowTopologyTestData::of))
@@ -169,8 +180,16 @@ public class FlowTopologyTest {
// When
computeAndSaveTopologies(List.of(triggeredFlowOne, triggeredFlowTwo));
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
var dependencies = flowService.findDependencies(tenantId, "io.kestra.unittest", triggeredFlowTwo.getId(), false, true).toList();
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
// Then
assertThat(dependencies.stream().map(FlowTopologyTestData::of))
.containsExactlyInAnyOrder(
@@ -192,7 +211,7 @@ public class FlowTopologyTest {
- id: a
type: BOOL
defaults: true
- id: b
type: BOOL
defaults: "{{ inputs.a == true }}"
@@ -232,8 +251,15 @@ public class FlowTopologyTest {
// When
computeAndSaveTopologies(List.of(child, parent, unrelatedFlow));
System.out.println();
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
var dependencies = flowService.findDependencies(tenantId, "io.kestra.unittest", parent.getId(), false, true);
flowTopologyRepository.findAll(tenantId).forEach(topology -> {
System.out.println(FlowTopologyTestData.of(topology));
});
// Then
assertThat(dependencies.map(FlowTopologyTestData::of))
@@ -242,63 +268,6 @@ public class FlowTopologyTest {
);
}
@Test
void testNamespaceGraph() throws FlowProcessingException {
var tenantId = randomTenantId();
var subChild = flowService.importFlow(tenantId,
"""
id: sub_child
namespace: io.kestra.unittest.sub
tasks:
- id: log
type: io.kestra.plugin.core.log.Log
message: Sub Child
""");
var child = flowService.importFlow(tenantId,
"""
id: child
namespace: io.kestra.unittest
tasks:
- id: callSub
type: io.kestra.core.tasks.flows.Flow
flowId: sub_child
namespace: io.kestra.unittest.sub
""");
var parent = flowService.importFlow(tenantId,
"""
id: parent
namespace: io.kestra.unittest
tasks:
- id: callChild
type: io.kestra.core.tasks.flows.Flow
flowId: child
namespace: io.kestra.unittest
""");
var unrelated = flowService.importFlow(tenantId,
"""
id: unrelated
namespace: io.kestra.unittest
tasks:
- id: log
type: io.kestra.plugin.core.log.Log
message: Not part of deps
""");
computeAndSaveTopologies(List.of(subChild, child, parent, unrelated));
FlowTopologyGraph graph = flowTopologyService.namespaceGraph(tenantId, "io.kestra.unittest");
assertThat(graph.getNodes())
.extracting(FlowNode::getId)
.contains("parent", "child", "sub_child", "unrelated");
assertThat(graph.getEdges().size()).isEqualTo(2);
}
/**
* this function mimics the production behaviour
*/

View File

@@ -0,0 +1,162 @@
#!/bin/bash
#===============================================================================
# SCRIPT: check-plugin-artifacts.sh
#
# DESCRIPTION:
# This script can be used to check if plugins are available on Maven Central or Sonatype
#
# USAGE: ./ check-plugin-artifacts.sh [options]
# OPTIONS:
# --plugin-file File containing the plugin list (default: .plugins)
# --version Version for plugins
# -y, --yes Automatically confirm prompts (non-interactive).
# -h, --help Show the help message and exit
# EXAMPLES:
# To clone all plugins:
# ./check-plugin-artifacts.sh --plugin-file .plugins --version 0.21.0
#===============================================================================
set -e;
###############################################################
# Global vars
###############################################################
BASEDIR=$(dirname "$(readlink -f $0)")
WORKING_DIR=/tmp/kestra-plugins;
PLUGIN_FILE="$BASEDIR/../.plugins"
# Maven Central URL
MAVEN_CENTRAL="https://repo1.maven.org/maven2"
SONATYPE_SNAPSHOT="https://central.sonatype.com/repository/maven-snapshots"
###############################################################
# Functions
###############################################################
# Function to display the help message
usage() {
echo "Usage: $0 --version <version> [--plugin-file]"
echo
echo "Options:"
echo " --plugin-file File containing the plugin list"
echo " --version Version for plugins"
echo " -y, --yes Automatically confirm prompts (non-interactive)."
echo " -h, --help Show this help message and exit."
exit 1
}
# Function to ask to continue
function askToContinue() {
read -p "Are you sure you want to continue? [y/N] " confirm
[[ "$confirm" =~ ^[Yy]$ ]] || { echo "Operation cancelled."; exit 1; }
}
###############################################################
# Options
###############################################################
PLUGINS_ARGS=()
AUTO_YES=false
# Get the options
while [[ "$#" -gt 0 ]]; do
case "$1" in
--plugin-file)
PLUGIN_FILE="$2"
shift 2
;;
--plugin-file=*)
PLUGIN_FILE="${1#*=}"
shift
;;
--version)
VERSION="$2"
shift 2
;;
--version=*)
VERSION="${1#*=}"
shift
;;
-y|--yes)
AUTO_YES=true
shift
;;
-h|--help)
usage
;;
*)
PLUGINS_ARGS+=("$1")
shift
;;
esac
done
## Check options
if [[ -z "$VERSION" ]]; then
echo -e "Missing required argument: --version\n";
usage
fi
## Get plugin list
if [[ "${#PLUGINS_ARGS[@]}" -eq 0 ]]; then
if [ -f "$PLUGIN_FILE" ]; then
PLUGINS=$(cat "$PLUGIN_FILE" | grep "io\\.kestra\\." | sed -e '/#/s/^.//' | cut -d':' -f2-3 | uniq | sort);
PLUGINS_COUNT=$(echo "$PLUGINS" | wc -l);
PLUGINS_ARRAY=$(echo "$PLUGINS" | xargs || echo '');
PLUGINS_ARRAY=($PLUGINS_ARRAY);
fi
else
PLUGINS_ARRAY=("${PLUGINS_ARGS[@]}")
PLUGINS_COUNT="${#PLUGINS_ARGS[@]}"
fi
echo "Arguments: "
echo "VERSION=$VERSION"
echo "PLUGIN_FILE=$PLUGIN_FILE"
for PLUGIN in "${PLUGINS_ARRAY[@]}"; do
echo "$PLUGIN"
done
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
###############################################################
# Main
###############################################################
mkdir -p "$WORKING_DIR"
COUNTER=1;
AVAILABLE=0;
NOT_AVAILABLE=0;
for PLUGIN in "${PLUGINS_ARRAY[@]}"
do
# Extract groupId and artifactId
GROUP_ID="${PLUGIN%%:*}"
ARTIFACT_ID="${PLUGIN##*:}"
# Convert groupId to Maven repository path
GROUP_PATH="${GROUP_ID//./\/}"
if [[ "$VERSION" == *"-SNAPSHOT" ]]; then
ARTIFACT_URL="${GROUP_PATH}/${ARTIFACT_ID}/${VERSION}/maven-metadata.xml"
ARTIFACT_URL="${SONATYPE_SNAPSHOT}/${ARTIFACT_URL}"
else
ARTIFACT_URL="${GROUP_PATH}/${ARTIFACT_ID}/${VERSION}/${ARTIFACT_ID}-${VERSION}.jar"
ARTIFACT_URL="${MAVEN_CENTRAL}/${ARTIFACT_URL}"
fi
if curl --silent --fail --head "$ARTIFACT_URL" > /dev/null; then
printf "✅ [$COUNTER/$PLUGINS_COUNT] %-45s : %s\n" "[$PLUGIN]" "Artifact is available: $ARTIFACT_URL";
AVAILABLE=$(( AVAILABLE + 1 ))
else
printf "❌ [$COUNTER/$PLUGINS_COUNT] %-45s : %s\n" "[$PLUGIN]" "Artifact is NOT available: $ARTIFACT_URL";
NOT_AVAILABLE=$(( NOT_AVAILABLE + 1 ))
fi
COUNTER=$(( COUNTER + 1 ));
done;
echo -e "\n\n✅ Available: $AVAILABLE, ❌ Unavailable: $NOT_AVAILABLE";
exit 0;

239
dev-tools/release-plugins.sh Executable file
View File

@@ -0,0 +1,239 @@
#!/bin/bash
#===============================================================================
# SCRIPT: release-plugins.sh
#
# DESCRIPTION:
# This script can be used to run a ./gradlew release command on each kestra plugin repository.
# By default, if no `GITHUB_PAT` environment variable exist, the script will attempt to clone GitHub repositories using SSH_KEY.
#
# USAGE: ./release-plugins.sh [options]
# OPTIONS:
# --release-version <version> Specify the release version (required)
# --next-version <version> Specify the next version (required)
# --dry-run Specify to run in DRY_RUN.
# -y, --yes Automatically confirm prompts (non-interactive).
# -h, --help Show the help message and exit
# EXAMPLES:
# To release all plugins:
# ./release-plugins.sh --release-version=0.20.0 --next-version=0.21.0-SNAPSHOT
# To release a specific plugin:
# ./release-plugins.sh --release-version=0.20.0 --next-version=0.21.0-SNAPSHOT plugin-kubernetes
# To release specific plugins from file:
# ./release-plugins.sh --release-version=0.20.0 --plugin-file .plugins
#===============================================================================
set -e;
###############################################################
# Global vars
###############################################################
BASEDIR=$(dirname "$(readlink -f $0)")
WORKING_DIR=/tmp/kestra-release-plugins-$(date +%s);
PLUGIN_FILE="$BASEDIR/../.plugins"
GIT_BRANCH=master
###############################################################
# Functions
###############################################################
# Function to display the help message
usage() {
echo "Usage: $0 --release-version <version> --next-version [plugin-repositories...]"
echo
echo "Options:"
echo " --release-version <version> Specify the release version (required)."
echo " --next-version <version> Specify the next version (required)."
echo " --plugin-file File containing the plugin list (default: .plugins)"
echo " --dry-run Specify to run in DRY_RUN."
echo " -y, --yes Automatically confirm prompts (non-interactive)."
echo " -h, --help Show this help message and exit."
exit 1
}
# Function to ask to continue
function askToContinue() {
read -p "Are you sure you want to continue? [y/N] " confirm
[[ "$confirm" =~ ^[Yy]$ ]] || { echo "Operation cancelled."; exit 1; }
}
###############################################################
# Options
###############################################################
PLUGINS_ARGS=()
AUTO_YES=false
DRY_RUN=false
# Get the options
while [[ "$#" -gt 0 ]]; do
case "$1" in
--release-version)
RELEASE_VERSION="$2"
shift 2
;;
--release-version=*)
RELEASE_VERSION="${1#*=}"
shift
;;
--next-version)
NEXT_VERSION="$2"
shift 2
;;
--next-version=*)
NEXT_VERSION="${1#*=}"
shift
;;
--plugin-file)
PLUGIN_FILE="$2"
shift 2
;;
--plugin-file=*)
PLUGIN_FILE="${1#*=}"
shift
;;
--dry-run)
DRY_RUN=true
shift
;;
-y|--yes)
AUTO_YES=true
shift
;;
-h|--help)
usage
;;
*)
PLUGINS_ARGS+=("$1")
shift
;;
esac
done
## Check options
if [[ -z "$RELEASE_VERSION" ]]; then
echo -e "Missing required argument: --release-version\n";
usage
fi
if [[ -z "$NEXT_VERSION" ]]; then
echo -e "Missing required argument: --next-version\n";
usage
fi
## Get plugin list
if [[ "${#PLUGINS_ARGS[@]}" -eq 0 ]]; then
if [ -f "$PLUGIN_FILE" ]; then
PLUGINS=$(cat "$PLUGIN_FILE" | grep "io\\.kestra\\." | sed -e '/#/s/^.//' | cut -d':' -f1 | uniq | sort);
PLUGINS_COUNT=$(echo "$PLUGINS" | wc -l);
PLUGINS_ARRAY=$(echo "$PLUGINS" | xargs || echo '');
PLUGINS_ARRAY=($PLUGINS_ARRAY);
fi
else
PLUGINS_ARRAY=("${PLUGINS_ARGS[@]}")
PLUGINS_COUNT="${#PLUGINS_ARGS[@]}"
fi
# Extract the major and minor versions
BASE_VERSION=$(echo "$RELEASE_VERSION" | sed -E 's/^([0-9]+\.[0-9]+)\..*/\1/')
PUSH_RELEASE_BRANCH="releases/v${BASE_VERSION}.x"
## Get plugin list
echo "RELEASE_VERSION=$RELEASE_VERSION"
echo "NEXT_VERSION=$NEXT_VERSION"
echo "PUSH_RELEASE_BRANCH=$PUSH_RELEASE_BRANCH"
echo "GIT_BRANCH=$GIT_BRANCH"
echo "DRY_RUN=$DRY_RUN"
echo "Found ($PLUGINS_COUNT) plugin repositories:";
for PLUGIN in "${PLUGINS_ARRAY[@]}"; do
echo "$PLUGIN"
done
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
###############################################################
# Main
###############################################################
mkdir -p $WORKING_DIR
COUNTER=1;
for PLUGIN in "${PLUGINS_ARRAY[@]}"
do
cd $WORKING_DIR;
echo "---------------------------------------------------------------------------------------"
echo "[$COUNTER/$PLUGINS_COUNT] Release Plugin: $PLUGIN"
echo "---------------------------------------------------------------------------------------"
if [[ -z "${GITHUB_PAT}" ]]; then
git clone git@github.com:kestra-io/$PLUGIN
else
echo "Clone git repository using GITHUB PAT"
git clone https://${GITHUB_PAT}@github.com/kestra-io/$PLUGIN.git
fi
cd "$PLUGIN";
if [[ "$PLUGIN" == "plugin-transform" ]] && [[ "$GIT_BRANCH" == "master" ]]; then # quickfix
git checkout main;
else
git checkout "$GIT_BRANCH";
fi
# Check if tag already exists on remote
TAG_EXISTS=$(git ls-remote --tags origin "refs/tags/v${RELEASE_VERSION}" | wc -l)
if [[ "$TAG_EXISTS" -ne 0 ]]; then
echo "Tag ${RELEASE_VERSION} already exists for $PLUGIN. Skipping..."
continue
fi
if [[ "$DRY_RUN" == false ]]; then
CURRENT_BRANCH=$(git branch --show-current);
echo "Run gradle release for plugin: $PLUGIN";
echo "Branch: $CURRENT_BRANCH";
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
# Create and push release branch
git checkout -b "$PUSH_RELEASE_BRANCH";
git push -u origin "$PUSH_RELEASE_BRANCH";
# Run gradle release
git checkout "$CURRENT_BRANCH";
if [[ "$RELEASE_VERSION" == *"-SNAPSHOT" ]]; then
# -SNAPSHOT qualifier maybe used to test release-candidates
./gradlew release -Prelease.useAutomaticVersion=true \
-Prelease.releaseVersion="${RELEASE_VERSION}" \
-Prelease.newVersion="${NEXT_VERSION}" \
-Prelease.pushReleaseVersionBranch="${PUSH_RELEASE_BRANCH}" \
-Prelease.failOnSnapshotDependencies=false
else
./gradlew release -Prelease.useAutomaticVersion=true \
-Prelease.releaseVersion="${RELEASE_VERSION}" \
-Prelease.newVersion="${NEXT_VERSION}" \
-Prelease.pushReleaseVersionBranch="${PUSH_RELEASE_BRANCH}"
fi
git push;
# Update the upper bound version of kestra
PLUGIN_KESTRA_VERSION="[${BASE_VERSION},)"
git checkout "$PUSH_RELEASE_BRANCH" && git pull;
sed -i "s/^kestraVersion=.*/kestraVersion=${PLUGIN_KESTRA_VERSION}/" ./gradle.properties
git add ./gradle.properties
# Check if there are staged changes
if ! git diff --cached --quiet; then
git commit -m"chore(deps): update kestraVersion to ${PLUGIN_KESTRA_VERSION}."
git push
fi
sleep 5; # add a short delay to not spam Maven Central
else
echo "Skip gradle release [DRY_RUN=true]";
fi
COUNTER=$(( COUNTER + 1 ));
done;
exit 0;

View File

@@ -0,0 +1,190 @@
#!/bin/bash
#===============================================================================
# SCRIPT: setversion-tag-plugins.sh
#
# DESCRIPTION:
# This script can be used to update and tag plugins from a release branch .e.g., releases/v0.21.x.
# By default, if no `GITHUB_PAT` environment variable exist, the script will attempt to clone GitHub repositories using SSH_KEY.
#
# USAGE: ./setversion-tag-plugins.sh [options]
# OPTIONS:
# --release-version <version> Specify the release version (required)
# --dry-run Specify to run in DRY_RUN.
# -y, --yes Automatically confirm prompts (non-interactive).
# -h, --help Show the help message and exit
# EXAMPLES:
# To release all plugins:
# ./setversion-tag-plugins.sh --release-version=0.20.0
# To release a specific plugin:
# ./setversion-tag-plugins.sh --release-version=0.20.0 plugin-kubernetes
# To release specific plugins from file:
# ./setversion-tag-plugins.sh --release-version=0.20.0 --plugin-file .plugins
#===============================================================================
set -e;
###############################################################
# Global vars
###############################################################
BASEDIR=$(dirname "$(readlink -f $0)")
WORKING_DIR=/tmp/kestra-release-plugins-$(date +%s);
PLUGIN_FILE="$BASEDIR/../.plugins"
###############################################################
# Functions
###############################################################
# Function to display the help message
usage() {
echo "Usage: $0 --release-version <version> [plugin-repositories...]"
echo
echo "Options:"
echo " --release-version <version> Specify the release version (required)."
echo " --plugin-file File containing the plugin list (default: .plugins)"
echo " --dry-run Specify to run in DRY_RUN."
echo " -y, --yes Automatically confirm prompts (non-interactive)."
echo " -h, --help Show this help message and exit."
exit 1
}
# Function to ask to continue
function askToContinue() {
read -p "Are you sure you want to continue? [y/N] " confirm
[[ "$confirm" =~ ^[Yy]$ ]] || { echo "Operation cancelled."; exit 1; }
}
###############################################################
# Options
###############################################################
PLUGINS_ARGS=()
AUTO_YES=false
DRY_RUN=false
# Get the options
while [[ "$#" -gt 0 ]]; do
case "$1" in
--release-version)
RELEASE_VERSION="$2"
shift 2
;;
--release-version=*)
RELEASE_VERSION="${1#*=}"
shift
;;
--plugin-file)
PLUGIN_FILE="$2"
shift 2
;;
--plugin-file=*)
PLUGIN_FILE="${1#*=}"
shift
;;
--dry-run)
DRY_RUN=true
shift
;;
-y|--yes)
AUTO_YES=true
shift
;;
-h|--help)
usage
;;
*)
PLUGINS_ARGS+=("$1")
shift
;;
esac
done
## Check options
if [[ -z "$RELEASE_VERSION" ]]; then
echo -e "Missing required argument: --release-version\n";
usage
fi
## Get plugin list
if [[ "${#PLUGINS_ARGS[@]}" -eq 0 ]]; then
if [ -f "$PLUGIN_FILE" ]; then
PLUGINS=$(cat "$PLUGIN_FILE" | grep "io\\.kestra\\." | sed -e '/#/s/^.//' | cut -d':' -f1 | uniq | sort);
PLUGINS_COUNT=$(echo "$PLUGINS" | wc -l);
PLUGINS_ARRAY=$(echo "$PLUGINS" | xargs || echo '');
PLUGINS_ARRAY=($PLUGINS_ARRAY);
fi
else
PLUGINS_ARRAY=("${PLUGINS_ARGS[@]}")
PLUGINS_COUNT="${#PLUGINS_ARGS[@]}"
fi
# Extract the major and minor versions
BASE_VERSION=$(echo "$RELEASE_VERSION" | sed -E 's/^([0-9]+\.[0-9]+)\..*/\1/')
RELEASE_BRANCH="releases/v${BASE_VERSION}.x"
## Get plugin list
echo "RELEASE_VERSION=$RELEASE_VERSION"
echo "RELEASE_BRANCH=$RELEASE_BRANCH"
echo "DRY_RUN=$DRY_RUN"
echo "Found ($PLUGINS_COUNT) plugin repositories:";
for PLUGIN in "${PLUGINS_ARRAY[@]}"; do
echo "$PLUGIN"
done
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
###############################################################
# Main
###############################################################
mkdir -p $WORKING_DIR
COUNTER=1;
for PLUGIN in "${PLUGINS_ARRAY[@]}"
do
cd $WORKING_DIR;
echo "---------------------------------------------------------------------------------------"
echo "[$COUNTER/$PLUGINS_COUNT] $PLUGIN"
echo "---------------------------------------------------------------------------------------"
if [[ -z "${GITHUB_PAT}" ]]; then
git clone git@github.com:kestra-io/$PLUGIN
else
echo "Clone git repository using GITHUB PAT"
git clone https://${GITHUB_PAT}@github.com/kestra-io/$PLUGIN.git
fi
cd "$PLUGIN";
git checkout $RELEASE_BRANCH;
if [[ "$DRY_RUN" == false ]]; then
CURRENT_BRANCH=$(git branch --show-current);
echo "Update version and tag plugin: $PLUGIN";
echo "Branch: $CURRENT_BRANCH";
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
# Checkout release branch
git checkout "$RELEASE_BRANCH";
# Update version
sed -i "s/^version=.*/version=$RELEASE_VERSION/" ./gradle.properties
git add ./gradle.properties
git commit -m"chore(version): update to version 'v$RELEASE_VERSION'."
git push
git tag -a "v$RELEASE_VERSION" -m"v$RELEASE_VERSION"
git push origin "v$RELEASE_VERSION"
sleep 5; # add a short delay to not spam Maven Central
else
echo "Skip tagging [DRY_RUN=true]";
fi
COUNTER=$(( COUNTER + 1 ));
done;
exit 0;

View File

@@ -0,0 +1,200 @@
#!/bin/bash
#===============================================================================
# SCRIPT: update-plugin-kestra-version.sh
#
# DESCRIPTION:
# This script can be used to update the gradle 'kestraVersion' property on each kestra plugin repository.
# By default, if no `GITHUB_PAT` environment variable exist, the script will attempt to clone GitHub repositories using SSH_KEY.
#
#USAGE:
# ./dev-tools/update-plugin-kestra-version.sh --branch <branch> --version <version> [plugin-repositories...]
#
#OPTIONS:
# --branch <branch> Specify the branch on which to update the kestraCoreVersion (default: master).
# --version <version> Specify the Kestra core version (required).
# --plugin-file File containing the plugin list (default: .plugins)
# --dry-run Specify to run in DRY_RUN.
# -y, --yes Automatically confirm prompts (non-interactive).
# -h, --help Show this help message and exit.
# EXAMPLES:
# To release all plugins:
# ./update-plugin-kestra-version.sh --branch=releases/v0.23.x --version="[0.23,0.24)"
# To release a specific plugin:
# ./update-plugin-kestra-version.sh --branch=releases/v0.23.x --version="[0.23,0.24)" plugin-kubernetes
# To release specific plugins from file:
# ./update-plugin-kestra-version.sh --branch=releases/v0.23.x --version="[0.23,0.24)" --plugin-file .plugins
#===============================================================================
set -e;
###############################################################
# Global vars
###############################################################
BASEDIR=$(dirname "$(readlink -f $0)")
SCRIPT_NAME=$(basename "$0")
SCRIPT_NAME="${SCRIPT_NAME%.*}"
WORKING_DIR="/tmp/kestra-$SCRIPT_NAME-$(date +%s)"
PLUGIN_FILE="$BASEDIR/../.plugins"
GIT_BRANCH=master
###############################################################
# Functions
###############################################################
# Function to display the help message
usage() {
echo "Usage: $0 --branch <branch> --version <version> [plugin-repositories...]"
echo
echo "Options:"
echo " --branch <branch> Specify the branch on which to update the kestraCoreVersion (default: master)."
echo " --version <version> Specify the Kestra core version (required)."
echo " --plugin-file File containing the plugin list (default: .plugins)"
echo " --dry-run Specify to run in DRY_RUN."
echo " -y, --yes Automatically confirm prompts (non-interactive)."
echo " -h, --help Show this help message and exit."
exit 1
}
# Function to ask to continue
function askToContinue() {
read -p "Are you sure you want to continue? [y/N] " confirm
[[ "$confirm" =~ ^[Yy]$ ]] || { echo "Operation cancelled."; exit 1; }
}
###############################################################
# Options
###############################################################
PLUGINS_ARGS=()
AUTO_YES=false
DRY_RUN=false
# Get the options
while [[ "$#" -gt 0 ]]; do
case "$1" in
--branch)
GIT_BRANCH="$2"
shift 2
;;
--branch=*)
GIT_BRANCH="${1#*=}"
shift
;;
--version)
VERSION="$2"
shift 2
;;
--version=*)
VERSION="${1#*=}"
shift
;;
--plugin-file)
PLUGIN_FILE="$2"
shift 2
;;
--plugin-file=*)
PLUGIN_FILE="${1#*=}"
shift
;;
--dry-run)
DRY_RUN=true
shift
;;
-y|--yes)
AUTO_YES=true
shift
;;
-h|--help)
usage
;;
*)
PLUGINS_ARGS+=("$1")
shift
;;
esac
done
## Check options
if [[ -z "$VERSION" ]]; then
echo -e "Missing required argument: --version\n";
usage
fi
## Get plugin list
if [[ "${#PLUGINS_ARGS[@]}" -eq 0 ]]; then
if [ -f "$PLUGIN_FILE" ]; then
PLUGINS=$(cat "$PLUGIN_FILE" | grep "io\\.kestra\\." | sed -e '/#/s/^.//' | cut -d':' -f1 | uniq | sort);
PLUGINS_COUNT=$(echo "$PLUGINS" | wc -l);
PLUGINS_ARRAY=$(echo "$PLUGINS" | xargs || echo '');
PLUGINS_ARRAY=($PLUGINS_ARRAY);
fi
else
PLUGINS_ARRAY=("${PLUGINS_ARGS[@]}")
PLUGINS_COUNT="${#PLUGINS_ARGS[@]}"
fi
## Get plugin list
echo "VERSION=$RELEASE_VERSION"
echo "GIT_BRANCH=$GIT_BRANCH"
echo "DRY_RUN=$DRY_RUN"
echo "Found ($PLUGINS_COUNT) plugin repositories:";
for PLUGIN in "${PLUGINS_ARRAY[@]}"; do
echo "$PLUGIN"
done
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
###############################################################
# Main
###############################################################
mkdir -p $WORKING_DIR
COUNTER=1;
for PLUGIN in "${PLUGINS_ARRAY[@]}"
do
cd $WORKING_DIR;
echo "---------------------------------------------------------------------------------------"
echo "[$COUNTER/$PLUGINS_COUNT] Update Plugin: $PLUGIN"
echo "---------------------------------------------------------------------------------------"
if [[ -z "${GITHUB_PAT}" ]]; then
git clone git@github.com:kestra-io/$PLUGIN
else
echo "Clone git repository using GITHUB PAT"
git clone https://${GITHUB_PAT}@github.com/kestra-io/$PLUGIN.git
fi
cd "$PLUGIN";
if [[ "$PLUGIN" == "plugin-transform" ]] && [[ "$GIT_BRANCH" == "master" ]]; then # quickfix
git checkout main;
else
git checkout "$GIT_BRANCH" || git checkout -b "$GIT_BRANCH";
fi
CURRENT_BRANCH=$(git branch --show-current);
echo "Update kestraVersion for plugin: $PLUGIN on branch $CURRENT_BRANCH:";
# Update the kestraVersion property
sed -i "s/^kestraVersion=.*/kestraVersion=${VERSION}/" ./gradle.properties
# Display diff
git diff --exit-code --unified=0 ./gradle.properties | grep -E '^\+|^-' | grep -v -E '^\+\+\+|^---' || echo "No changes detected in gradle.properties";
if [[ "$DRY_RUN" == false ]]; then
if [[ "$AUTO_YES" == false ]]; then
askToContinue
fi
git add ./gradle.properties
git commit -m"chore(deps): update kestraVersion to ${VERSION}." || true
git push --set-upstream origin $GIT_BRANCH || true
else
echo "Skip git commit/push [DRY_RUN=true]";
fi
COUNTER=$(( COUNTER + 1 ));
done;
exit 0;

View File

@@ -15,9 +15,9 @@ volumes:
services:
postgres:
image: postgres:18
image: postgres
volumes:
- postgres-data:/var/lib/postgresql/18/docker
- postgres-data:/var/lib/postgresql/data
environment:
POSTGRES_DB: kestra
POSTGRES_USER: kestra

View File

@@ -6,9 +6,9 @@ volumes:
services:
postgres:
image: postgres:18
image: postgres
volumes:
- postgres-data:/var/lib/postgresql/18/docker
- postgres-data:/var/lib/postgresql/data
environment:
POSTGRES_DB: kestra
POSTGRES_USER: kestra

View File

@@ -1,4 +1,4 @@
version=1.2.0-SNAPSHOT
version=1.1.1
org.gradle.jvmargs=-Xmx2g -XX:MaxMetaspaceSize=512m -XX:+HeapDumpOnOutOfMemoryError
org.gradle.parallel=true

View File

@@ -79,28 +79,6 @@ public abstract class AbstractJdbcFlowTopologyRepository extends AbstractJdbcRep
});
}
@Override
public List<FlowTopology> findByNamespacePrefix(String tenantId, String namespacePrefix) {
return jdbcRepository
.getDslContextWrapper()
.transactionResult(configuration -> {
// Match flows that originate from the namespace or its children
Condition sourceCondition = field("source_namespace").eq(namespacePrefix)
.or(field("source_namespace").likeIgnoreCase(namespacePrefix + ".%"));
Condition tenantSource = buildTenantCondition("source", tenantId);
Condition tenantDest = buildTenantCondition("destination", tenantId);
Select<Record1<Object>> from = DSL
.using(configuration)
.select(field("value"))
.from(this.jdbcRepository.getTable())
.where(tenantSource.and(tenantDest).and(sourceCondition));
return this.jdbcRepository.fetch(from);
});
}
@Override
public List<FlowTopology> findAll(String tenantId) {
return jdbcRepository

View File

@@ -19,9 +19,9 @@ dependencies {
def jsonschemaVersion = "4.38.0"
def kafkaVersion = "4.1.0"
def opensearchVersion = "3.2.0"
def opensearchRestVersion = "3.3.2"
def opensearchRestVersion = "3.3.1"
def flyingSaucerVersion = "10.0.3"
def jacksonVersion = "2.20.1"
def jacksonVersion = "2.20.0"
def jacksonAnnotationsVersion = "2.20"
def jugVersion = "5.1.1"
def langchain4jVersion = "1.8.0"
@@ -34,8 +34,8 @@ dependencies {
api platform("io.qameta.allure:allure-bom:2.30.0")
// we define cloud bom here for GCP, Azure and AWS so they are aligned for all plugins that use them (secret, storage, oss and ee plugins)
api platform('com.google.cloud:libraries-bom:26.71.0')
api platform("com.azure:azure-sdk-bom:1.3.2")
api platform('software.amazon.awssdk:bom:2.37.5')
api platform("com.azure:azure-sdk-bom:1.3.0")
api platform('software.amazon.awssdk:bom:2.36.3')
api platform("dev.langchain4j:langchain4j-bom:$langchain4jVersion")
api platform("dev.langchain4j:langchain4j-community-bom:$langchain4jCommunityVersion")
@@ -103,7 +103,7 @@ dependencies {
api group: 'co.elastic.logging', name: 'logback-ecs-encoder', version: '1.7.0'
api group: 'de.focus-shift', name: 'jollyday-core', version: jollydayVersion
api group: 'de.focus-shift', name: 'jollyday-jaxb', version: jollydayVersion
api 'nl.basjes.gitignore:gitignore-reader:1.12.1'
api 'nl.basjes.gitignore:gitignore-reader:1.12.0'
api group: 'dev.failsafe', name: 'failsafe', version: '3.3.2'
api group: 'com.cronutils', name: 'cron-utils', version: '9.2.1'
api group: 'com.github.victools', name: 'jsonschema-generator', version: jsonschemaVersion

View File

@@ -25,7 +25,7 @@ import java.util.concurrent.atomic.AtomicBoolean;
@Slf4j
@Singleton
public class TestRunner implements Runnable, AutoCloseable {
@Setter private int workerThread = Math.max(3, Runtime.getRuntime().availableProcessors()) * 16;
@Setter private int workerThread = Math.max(3, Runtime.getRuntime().availableProcessors());
@Setter private boolean schedulerEnabled = true;
@Setter private boolean workerEnabled = true;

228
ui/package-lock.json generated
View File

@@ -10,7 +10,7 @@
"hasInstallScript": true,
"dependencies": {
"@js-joda/core": "^5.6.5",
"@kestra-io/ui-libs": "^0.0.263",
"@kestra-io/ui-libs": "^0.0.261",
"@vue-flow/background": "^1.3.2",
"@vue-flow/controls": "^1.1.2",
"@vue-flow/core": "^1.47.0",
@@ -105,7 +105,6 @@
"patch-package": "^8.0.1",
"playwright": "^1.55.0",
"prettier": "^3.6.2",
"rimraf": "^6.0.1",
"rolldown-vite": "^7.1.20",
"rollup-plugin-copy": "^3.5.0",
"sass": "^1.92.3",
@@ -2119,29 +2118,6 @@
"url": "https://github.com/sponsors/kazupon"
}
},
"node_modules/@isaacs/balanced-match": {
"version": "4.0.1",
"resolved": "https://registry.npmjs.org/@isaacs/balanced-match/-/balanced-match-4.0.1.tgz",
"integrity": "sha512-yzMTt9lEb8Gv7zRioUilSglI0c0smZ9k5D65677DLWLtWJaXIS3CqcGyUFByYKlnUj6TkjLVs54fBl6+TiGQDQ==",
"dev": true,
"license": "MIT",
"engines": {
"node": "20 || >=22"
}
},
"node_modules/@isaacs/brace-expansion": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/@isaacs/brace-expansion/-/brace-expansion-5.0.0.tgz",
"integrity": "sha512-ZT55BDLV0yv0RBm2czMiZ+SqCGO7AvmOM3G/w2xhVPH+te0aKgFjmBvGlL1dH+ql2tgGO3MVrbb3jCKyvpgnxA==",
"dev": true,
"license": "MIT",
"dependencies": {
"@isaacs/balanced-match": "^4.0.1"
},
"engines": {
"node": "20 || >=22"
}
},
"node_modules/@isaacs/cliui": {
"version": "8.0.2",
"resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz",
@@ -3320,9 +3296,9 @@
"license": "BSD-3-Clause"
},
"node_modules/@kestra-io/ui-libs": {
"version": "0.0.263",
"resolved": "https://registry.npmjs.org/@kestra-io/ui-libs/-/ui-libs-0.0.263.tgz",
"integrity": "sha512-j1rWqcQAK2CudNBkcDPjUXyaGFeBzJ7QEhPKFAbleHSw0N3QFu/iy0rFZxJNIMWRi1mGZBh74D6vL0OqQJkT2Q==",
"version": "0.0.261",
"resolved": "https://registry.npmjs.org/@kestra-io/ui-libs/-/ui-libs-0.0.261.tgz",
"integrity": "sha512-beKqB90UYHdJ8jwlTjrenqecYWxM8DuWRaxWogWLVOqFS191kmKo5Qme6fhLbOVMVVdsxH2nY4VIMIvYFnHIOw==",
"dependencies": {
"@nuxtjs/mdc": "^0.17.3",
"@popperjs/core": "^2.11.8",
@@ -12608,23 +12584,6 @@
"node": ">=8"
}
},
"node_modules/istanbul-lib-processinfo/node_modules/rimraf": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
"integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
"deprecated": "Rimraf versions prior to v4 are no longer supported",
"dev": true,
"license": "ISC",
"dependencies": {
"glob": "^7.1.3"
},
"bin": {
"rimraf": "bin.js"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/istanbul-lib-processinfo/node_modules/uuid": {
"version": "8.3.2",
"resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
@@ -13462,23 +13421,6 @@
"jest-runner": "^29.3.1"
}
},
"node_modules/jest-playwright-preset/node_modules/rimraf": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
"integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
"deprecated": "Rimraf versions prior to v4 are no longer supported",
"dev": true,
"license": "ISC",
"dependencies": {
"glob": "^7.1.3"
},
"bin": {
"rimraf": "bin.js"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/jest-playwright-preset/node_modules/uuid": {
"version": "8.3.2",
"resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
@@ -17122,23 +17064,6 @@
"node": ">=8"
}
},
"node_modules/nyc/node_modules/rimraf": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
"integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
"deprecated": "Rimraf versions prior to v4 are no longer supported",
"dev": true,
"license": "ISC",
"dependencies": {
"glob": "^7.1.3"
},
"bin": {
"rimraf": "bin.js"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/nyc/node_modules/semver": {
"version": "6.3.1",
"resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
@@ -18985,133 +18910,17 @@
"license": "MIT"
},
"node_modules/rimraf": {
"version": "6.0.1",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-6.0.1.tgz",
"integrity": "sha512-9dkvaxAsk/xNXSJzMgFqqMCuFgt2+KsOFek3TMLfo8NCPfWpBmqwyNn5Y+NX56QUYfCtsyhF3ayiboEoUmJk/A==",
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
"integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
"deprecated": "Rimraf versions prior to v4 are no longer supported",
"dev": true,
"license": "ISC",
"dependencies": {
"glob": "^11.0.0",
"package-json-from-dist": "^1.0.0"
"glob": "^7.1.3"
},
"bin": {
"rimraf": "dist/esm/bin.mjs"
},
"engines": {
"node": "20 || >=22"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/rimraf/node_modules/foreground-child": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz",
"integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==",
"dev": true,
"license": "ISC",
"dependencies": {
"cross-spawn": "^7.0.6",
"signal-exit": "^4.0.1"
},
"engines": {
"node": ">=14"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/rimraf/node_modules/glob": {
"version": "11.0.3",
"resolved": "https://registry.npmjs.org/glob/-/glob-11.0.3.tgz",
"integrity": "sha512-2Nim7dha1KVkaiF4q6Dj+ngPPMdfvLJEOpZk/jKiUAkqKebpGAWQXAq9z1xu9HKu5lWfqw/FASuccEjyznjPaA==",
"dev": true,
"license": "ISC",
"dependencies": {
"foreground-child": "^3.3.1",
"jackspeak": "^4.1.1",
"minimatch": "^10.0.3",
"minipass": "^7.1.2",
"package-json-from-dist": "^1.0.0",
"path-scurry": "^2.0.0"
},
"bin": {
"glob": "dist/esm/bin.mjs"
},
"engines": {
"node": "20 || >=22"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/rimraf/node_modules/jackspeak": {
"version": "4.1.1",
"resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-4.1.1.tgz",
"integrity": "sha512-zptv57P3GpL+O0I7VdMJNBZCu+BPHVQUk55Ft8/QCJjTVxrnJHuVuX/0Bl2A6/+2oyR/ZMEuFKwmzqqZ/U5nPQ==",
"dev": true,
"license": "BlueOak-1.0.0",
"dependencies": {
"@isaacs/cliui": "^8.0.2"
},
"engines": {
"node": "20 || >=22"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/rimraf/node_modules/lru-cache": {
"version": "11.2.2",
"resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.2.2.tgz",
"integrity": "sha512-F9ODfyqML2coTIsQpSkRHnLSZMtkU8Q+mSfcaIyKwy58u+8k5nvAYeiNhsyMARvzNcXJ9QfWVrcPsC9e9rAxtg==",
"dev": true,
"license": "ISC",
"engines": {
"node": "20 || >=22"
}
},
"node_modules/rimraf/node_modules/minimatch": {
"version": "10.0.3",
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.0.3.tgz",
"integrity": "sha512-IPZ167aShDZZUMdRk66cyQAW3qr0WzbHkPdMYa8bzZhlHhO3jALbKdxcaak7W9FfT2rZNpQuUu4Od7ILEpXSaw==",
"dev": true,
"license": "ISC",
"dependencies": {
"@isaacs/brace-expansion": "^5.0.0"
},
"engines": {
"node": "20 || >=22"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/rimraf/node_modules/path-scurry": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.0.tgz",
"integrity": "sha512-ypGJsmGtdXUOeM5u93TyeIEfEhM6s+ljAhrk5vAvSx8uyY/02OvrZnA0YNGUrPXfpJMgI1ODd3nwz8Npx4O4cg==",
"dev": true,
"license": "BlueOak-1.0.0",
"dependencies": {
"lru-cache": "^11.0.0",
"minipass": "^7.1.2"
},
"engines": {
"node": "20 || >=22"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/rimraf/node_modules/signal-exit": {
"version": "4.1.0",
"resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz",
"integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==",
"dev": true,
"license": "ISC",
"engines": {
"node": ">=14"
"rimraf": "bin.js"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
@@ -19899,23 +19708,6 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/spawn-wrap/node_modules/rimraf": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
"integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
"deprecated": "Rimraf versions prior to v4 are no longer supported",
"dev": true,
"license": "ISC",
"dependencies": {
"glob": "^7.1.3"
},
"bin": {
"rimraf": "bin.js"
},
"funding": {
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/spawn-wrap/node_modules/semver": {
"version": "6.3.1",
"resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",

View File

@@ -19,12 +19,12 @@
"lint": "eslint --fix",
"storybook": "storybook dev -p 6006",
"build-storybook": "storybook build",
"prepare": "cd .. && husky ui/.husky && rimraf .git/hooks",
"prepare": "cd .. && husky ui/.husky && rm -f .git/hooks/*",
"postinstall": "patch-package"
},
"dependencies": {
"@js-joda/core": "^5.6.5",
"@kestra-io/ui-libs": "^0.0.263",
"@kestra-io/ui-libs": "^0.0.261",
"@vue-flow/background": "^1.3.2",
"@vue-flow/controls": "^1.1.2",
"@vue-flow/core": "^1.47.0",
@@ -119,7 +119,6 @@
"patch-package": "^8.0.1",
"playwright": "^1.55.0",
"prettier": "^3.6.2",
"rimraf": "^6.0.1",
"rolldown-vite": "^7.1.20",
"rollup-plugin-copy": "^3.5.0",
"sass": "^1.92.3",

View File

@@ -40,7 +40,7 @@
<button v-if="activeTab.length" class="closeButton" @click="setActiveTab('')">
<Close />
</button>
<KeepAlive v-if="activeTab">
<KeepAlive>
<ContextDocs v-if="activeTab === 'docs'" />
<ContextNews v-else-if="activeTab === 'news'" />
<template v-else>

View File

@@ -0,0 +1,39 @@
<template>
<el-button data-test-id="execution-status" class="status" :size="props.size ?? ''" :style="style">
{{ props.title || FILTERS.cap(props.status) }}
</el-button>
</template>
<script setup lang="ts">
import {computed} from "vue";
import * as FILTERS from "../utils/filters";
const props = defineProps<{
status: string;
size?: string;
title?: string;
label?: boolean;
}>();
const style = computed(() => {
const statusVarname = props.status?.toLowerCase();
return {
color: `var(--ks-content-${statusVarname}) !important`,
"border-color": `var(--ks-border-${statusVarname}) !important`,
"background-color": `var(--ks-background-${statusVarname}) !important`,
};
});
</script>
<style scoped lang="scss">
button.el-button {
white-space: nowrap;
border-radius: var(--el-border-radius-base);
width: 7rem;
cursor: default;
&.no-label {
padding: 0.5rem;
line-height: 1;
}
}
</style>

View File

@@ -128,9 +128,14 @@
}
},
getTabClasses(tab) {
if(tab.locked) return {"px-0": true};
return {"container": true, "mt-4": true};
}
const isEnterpriseTab = tab.locked;
return {
"container": !isEnterpriseTab,
"mt-4": !isEnterpriseTab,
"px-0": isEnterpriseTab,
};
},
},
computed: {
containerClass() {

View File

@@ -5,13 +5,17 @@
</div>
</template>
<script setup lang="ts">
import type {Component} from "vue";
const props = defineProps<{
icon: Component;
label: string;
}>();
<script setup>
const props = defineProps({
icon: {
type: Object,
required: true
},
label: {
type: String,
required: true
}
});
</script>
<style scoped lang="scss">

View File

@@ -41,13 +41,7 @@
<template #expand>
<el-table-column type="expand">
<template #default="props">
<LogsWrapper
class="m-3"
:filters="props.row"
v-if="hasLogsContent(props.row)"
:withCharts="false"
embed
/>
<LogsWrapper class="m-3" :filters="props.row" v-if="hasLogsContent(props.row)" :withCharts="false" embed />
</template>
</el-table-column>
</template>
@@ -77,9 +71,6 @@
<el-button @click="deleteBackfills()">
{{ $t("delete backfills") }}
</el-button>
<el-button @click="deleteTriggers()" type="danger">
{{ $t("delete triggers") }}
</el-button>
</BulkSelect>
</template>
<el-table-column
@@ -104,32 +95,17 @@
:sortOrders="['flowId', 'namespace', 'nextExecutionDate'].includes(col.prop) ? ['ascending', 'descending'] : undefined"
>
<template #header v-if="col.prop === 'date'">
<el-tooltip
:content="$t('last trigger date tooltip')"
placement="top"
effect="light"
popperClass="wide-tooltip"
>
<el-tooltip :content="$t('last trigger date tooltip')" placement="top" effect="light" popperClass="wide-tooltip">
<span>{{ col.label }}</span>
</el-tooltip>
</template>
<template #header v-else-if="col.prop === 'updatedDate'">
<el-tooltip
:content="$t('context updated date tooltip')"
placement="top"
effect="light"
popperClass="wide-tooltip"
>
<el-tooltip :content="$t('context updated date tooltip')" placement="top" effect="light" popperClass="wide-tooltip">
<span>{{ col.label }}</span>
</el-tooltip>
</template>
<template #header v-else-if="col.prop === 'nextExecutionDate'">
<el-tooltip
:content="$t('next evaluation date tooltip')"
placement="top"
effect="light"
popperClass="wide-tooltip"
>
<el-tooltip :content="$t('next evaluation date tooltip')" placement="top" effect="light" popperClass="wide-tooltip">
<span>{{ col.label }}</span>
</el-tooltip>
</template>
@@ -205,26 +181,13 @@
<LockOff />
</Kicon>
</el-button>
<el-button>
<Kicon
:tooltip="$t('delete trigger')"
placement="left"
@click="confirmDeleteTrigger(scope.row)"
>
<Delete />
</Kicon>
</el-button>
</template>
</el-table-column>
<el-table-column :label="$t('backfill')" columnKey="backfill">
<template #default="scope">
<div class="backfillContainer items-center gap-2">
<span v-if="scope.row.backfill" class="statusIcon">
<el-tooltip
v-if="!scope.row.backfill.paused"
:content="$t('backfill running')"
effect="light"
>
<el-tooltip v-if="!scope.row.backfill.paused" :content="$t('backfill running')" effect="light">
<PlayBox font />
</el-tooltip>
<el-tooltip v-else :content="$t('backfill paused')">
@@ -335,7 +298,6 @@
<script setup lang="ts">
import _merge from "lodash/merge";
import {ref, computed, watch} from "vue";
import moment from "moment";
import {useI18n} from "vue-i18n";
import {useRoute} from "vue-router";
import {ElMessage} from "element-plus";
@@ -349,16 +311,18 @@
import {useTriggerFilter} from "../filter/configurations";
import {useDataTableActions} from "../../composables/useDataTableActions";
import {useSelectTableActions} from "../../composables/useSelectTableActions";
import {type ColumnConfig, useTableColumns} from "../../composables/useTableColumns";
import {useTableColumns, type ColumnConfig} from "../../composables/useTableColumns";
import action from "../../models/action";
import permission from "../../models/permission";
const triggerFilter = useTriggerFilter();
import LockOff from "vue-material-design-icons/LockOff.vue";
import PlayBox from "vue-material-design-icons/PlayBox.vue";
import PauseBox from "vue-material-design-icons/PauseBox.vue";
import AlertCircle from "vue-material-design-icons/AlertCircle.vue";
import CalendarCollapseHorizontalOutline from "vue-material-design-icons/CalendarCollapseHorizontalOutline.vue";
import Delete from "vue-material-design-icons/Delete.vue";
import Id from "../Id.vue";
import Kicon from "../Kicon.vue";
@@ -377,8 +341,6 @@
import MarkdownTooltip from "../layout/MarkdownTooltip.vue";
import useRouteContext from "../../composables/useRouteContext";
const triggerFilter = useTriggerFilter();
const route = useRoute();
const toast = useToast();
@@ -407,55 +369,55 @@
end: null,
inputs: null,
labels: []
});
});
const optionalColumns = computed(() => [
{
label: t("flow"),
prop: "flowId",
default: true,
label: t("flow"),
prop: "flowId",
default: true,
description: t("filter.table_column.triggers.flow")
},
{
label: t("namespace"),
prop: "namespace",
default: true,
label: t("namespace"),
prop: "namespace",
default: true,
description: t("filter.table_column.triggers.namespace")
},
{
label: t("current execution"),
prop: "executionId",
default: false,
label: t("current execution"),
prop: "executionId",
default: false,
description: t("filter.table_column.triggers.current execution")
},
{
label: t("workerId"),
prop: "workerId",
default: false,
label: t("workerId"),
prop: "workerId",
default: false,
description: t("filter.table_column.triggers.workerId")
},
{
label: t("last trigger date"),
prop: "date",
default: true,
label: t("last trigger date"),
prop: "date",
default: true,
description: t("filter.table_column.triggers.last trigger date")
},
{
label: t("context updated date"),
prop: "updatedDate",
default: false,
label: t("context updated date"),
prop: "updatedDate",
default: false,
description: t("filter.table_column.triggers.context updated date")
},
{
label: t("next evaluation date"),
prop: "nextExecutionDate",
default: false,
label: t("next evaluation date"),
prop: "nextExecutionDate",
default: false,
description: t("filter.table_column.triggers.next evaluation date")
},
{
label: t("evaluation lock date"),
prop: "evaluateRunningDate",
default: false,
label: t("evaluation lock date"),
prop: "evaluateRunningDate",
default: false,
description: t("filter.table_column.triggers.evaluation lock date")
}
]);
@@ -468,7 +430,7 @@
initialVisibleColumns: optionalColumns.value.filter(col => col.default).map(col => col.prop)
});
const visibleColumns = computed(() =>
const visibleColumns = computed(() =>
displayColumns.value
.map(prop => optionalColumns.value.find(c => c.prop === prop))
.filter(Boolean) as ColumnConfig[]
@@ -506,7 +468,7 @@
});
const {
queryBulkAction,
queryBulkAction,
selection,
handleSelectionChange,
toggleAllUnselected,
@@ -582,7 +544,7 @@
const disabledEndDate = (time: Date) => {
return new Date() < time || (backfill.value.start && backfill.value.start > time);
};
const triggerLoadDataAfterBulkEditAction = () => {
loadData();
setTimeout(() => loadData(), 200);
@@ -639,44 +601,9 @@
});
};
const confirmDeleteTrigger = (trigger) => {
const genericConfirmAction = (toastKey: string, queryAction: string, byIdAction: string, success: string, data?: any) => {
toast.confirm(
t("delete trigger confirmation", {id: trigger.id}),
() => triggerStore.delete({
namespace: trigger.namespace,
flowId: trigger.flowId,
triggerId: trigger.triggerId
}).then(() => {
toast.success(t("delete trigger success", {id: trigger.id}));
loadData();
}).catch(error => {
toast.error(t("delete trigger error", {id: trigger.id}));
console.error(error);
}),
"warning"
);
};
const deleteTriggers = () => {
genericConfirmAction(
"bulk delete triggers",
"deleteByQuery",
"deleteByTriggers",
"bulk success delete triggers",
null,
"WARNING: deleting triggers may lead to duplicate executions if the triggers are still active in flows"
);
};
const genericConfirmAction = (toastKey: string, queryAction: string, byIdAction: string, success: string, data?: any, extraWarning = null) => {
let message = t(toastKey, {"count": queryBulkAction.value ? total.value : selection.value?.length}) + ". " + t("bulk action async warning");
if (extraWarning) {
message += "<br><br><strong>" + extraWarning + "</strong>";
}
toast.confirm(
message,
t(toastKey, {"count": queryBulkAction.value ? total.value : selection.value?.length}) + ". " + t("bulk action async warning"),
() => genericConfirmCallback(queryAction, byIdAction, success, data)
);
};
@@ -693,8 +620,6 @@
"unlockByTriggers": () => triggerStore.unlockByTriggers,
"setDisabledByQuery": () => triggerStore.setDisabledByQuery,
"setDisabledByTriggers": () => triggerStore.setDisabledByTriggers,
"deleteByQuery": () => triggerStore.deleteByQuery,
"deleteByTriggers": () => triggerStore.deleteByTriggers,
};
if (queryBulkAction.value) {
@@ -771,16 +696,7 @@
};
const loadQuery = (base: any) => {
const queryFilter = queryWithFilter();
const timeRange = queryFilter["filters[timeRange][EQUALS]"];
if (timeRange) {
const end = new Date();
const start = new Date(end.getTime() - moment.duration(timeRange).asMilliseconds());
queryFilter["filters[startDate][GREATER_THAN_OR_EQUAL_TO]"] = start.toISOString();
queryFilter["filters[endDate][LESS_THAN_OR_EQUAL_TO]"] = end.toISOString();
delete queryFilter["filters[timeRange][EQUALS]"];
}
const queryFilter = queryWithFilter("triggers");
return _merge(base, queryFilter);
};
@@ -836,86 +752,86 @@
</script>
<style scoped lang="scss">
.data-table-wrapper {
margin-left: 0 !important;
padding-left: 0 !important;
.data-table-wrapper {
margin-left: 0 !important;
padding-left: 0 !important;
}
.backfillContainer {
display: flex;
align-items: center;
}
.statusIcon {
font-size: large;
}
.trigger-issue-icon {
color: var(--ks-content-warning);
font-size: 1.4em;
}
.alert-circle-icon {
color: var(--ks-content-warning);
font-size: 1.4em;
}
:deep(.el-table__expand-icon) {
pointer-events: none;
.el-icon {
display: none;
}
}
:deep(.el-switch) {
.is-text {
padding: 0 3px;
color: inherit;
}
.backfillContainer {
display: flex;
align-items: center;
}
.statusIcon {
font-size: large;
}
.trigger-issue-icon {
color: var(--ks-content-warning);
font-size: 1.4em;
}
.alert-circle-icon {
color: var(--ks-content-warning);
font-size: 1.4em;
}
:deep(.el-table__expand-icon) {
pointer-events: none;
.el-icon {
display: none;
}
}
:deep(.el-switch) {
&.is-checked {
.is-text {
padding: 0 3px;
color: inherit;
}
&.is-checked {
.is-text {
color: #ffffff;
}
color: #ffffff;
}
}
}
.el-table {
a {
color: var(--ks-content-link);
}
.el-table {
a {
color: var(--ks-content-link);
}
}
.wide-tooltip {
max-width: 400px;
white-space: normal;
word-break: break-word;
color: var(--ks-content-primary) !important;
}
:deep(.el-collapse) {
border-radius: var(--bs-border-radius-lg);
border: 1px solid var(--ks-border-primary);
background: var(--bs-gray-100);
.el-collapse-item__header {
background: transparent;
border-bottom: 1px solid var(--ks-border-primary);
font-size: var(--bs-font-size-sm);
}
.wide-tooltip {
max-width: 400px;
white-space: normal;
word-break: break-word;
color: var(--ks-content-primary) !important;
}
:deep(.el-collapse) {
border-radius: var(--bs-border-radius-lg);
border: 1px solid var(--ks-border-primary);
.el-collapse-item__content {
background: var(--bs-gray-100);
border-bottom: 1px solid var(--ks-border-primary);
}
.el-collapse-item__header {
background: transparent;
border-bottom: 1px solid var(--ks-border-primary);
font-size: var(--bs-font-size-sm);
}
.el-collapse-item__content {
background: var(--bs-gray-100);
border-bottom: 1px solid var(--ks-border-primary);
}
.el-collapse-item__header,
.el-collapse-item__content {
&:last-child {
border-bottom-left-radius: var(--bs-border-radius-lg);
border-bottom-right-radius: var(--bs-border-radius-lg);
}
.el-collapse-item__header,
.el-collapse-item__content {
&:last-child {
border-bottom-left-radius: var(--bs-border-radius-lg);
border-bottom-right-radius: var(--bs-border-radius-lg);
}
}
</style>
}
</style>

View File

@@ -194,10 +194,6 @@
showClose: false
})
})
} else {
ElMessage.error({
message: t("setup.validation.incorrect_creds")
})
}
} catch {
ElMessage.error({
@@ -270,7 +266,6 @@
.basic-auth-login {
flex-shrink: 1;
width: 400px;
container-type: inline-size;
.logo {
width: 250px;
@@ -311,20 +306,5 @@
}
}
}
@media (max-width: 640px) {
width: 100%;
padding: 1rem;
.logo {
width: 200px;
margin-bottom: 1.5rem;
}
.el-form {
max-width: 100%;
padding: 1.5rem;
}
}
}
</style>

View File

@@ -18,7 +18,7 @@
</template>
<script setup lang="ts">
import {computed, onBeforeMount, ref, useTemplateRef, watch} from "vue";
import {computed, onBeforeMount, ref, useTemplateRef} from "vue";
import {stringify, parse} from "@kestra-io/ui-libs/flow-yaml-utils";
import type {Dashboard, Chart} from "./composables/useDashboards";
@@ -89,16 +89,9 @@
}
if (!props.isFlow && !props.isNamespace) {
// Preserve timeRange filter when switching dashboards
const preservedQuery = Object.fromEntries(
Object.entries(route.query).filter(([key]) =>
key.includes("timeRange")
)
);
router.replace({
params: {...route.params, dashboard: id},
query: route.params.dashboard !== id ? preservedQuery : {...route.query},
query: route.params.dashboard !== id ? {} : {...route.query},
});
}
@@ -109,22 +102,8 @@
onBeforeMount(() => {
const ID = getDashboard(route, "id");
if (props.isFlow) {
load(ID, processFlowYaml(YAML_FLOW, route.params.namespace as string, route.params.id as string));
} else if (props.isNamespace) {
load(ID, YAML_NAMESPACE);
}
});
watch(() => getDashboard(route, "id"), (newId, oldId) => {
if (newId !== oldId) {
const defaultYAML = props.isFlow
? processFlowYaml(YAML_FLOW, route.params.namespace as string, route.params.id as string)
: props.isNamespace
? YAML_NAMESPACE
: YAML_MAIN;
load(newId, defaultYAML);
}
if (props.isFlow && ID === "default") load("default", processFlowYaml(YAML_FLOW, route.params.namespace as string, route.params.id as string));
else if (props.isNamespace && ID === "default") load("default", YAML_NAMESPACE);
});
</script>

View File

@@ -96,19 +96,14 @@
return [DEFAULT, ...dashboards.value].filter((d) => !search.value || d.title.toLowerCase().includes(search.value.toLowerCase()));
});
const STORAGE_KEY = getDashboard(route, "key");
const selected = ref<string | null>(null);
const ID = getDashboard(route, "id") as string;
const selected = ref(null);
const select = (dashboard: any) => {
selected.value = dashboard?.title;
if (STORAGE_KEY) {
if (dashboard?.id) {
localStorage.setItem(STORAGE_KEY, dashboard.id);
} else {
localStorage.removeItem(STORAGE_KEY);
}
}
if (dashboard?.id) localStorage.setItem(ID, dashboard.id)
else localStorage.removeItem(ID);
emits("dashboard", dashboard.id);
};
@@ -126,7 +121,7 @@
});
};
const getStoredDashboard = () => STORAGE_KEY ? localStorage.getItem(STORAGE_KEY) : null;
const fetchLast = () => localStorage.getItem(ID);
const fetchDashboards = () => {
dashboardStore
.list({})
@@ -134,17 +129,13 @@
dashboards.value = response.results;
const creation = Boolean(route.query.created);
const lastSelected = creation
? (route.params?.dashboard ?? getStoredDashboard())
: (getStoredDashboard() ?? route.params?.dashboard);
const lastSelected = creation ? (route.params?.dashboard ?? fetchLast()) : (fetchLast() ?? route.params?.dashboard);
if (lastSelected) {
const dashboard = dashboards.value.find((d) => d.id === lastSelected);
if (dashboard) {
selected.value = dashboard.title;
emits("dashboard", dashboard.id);
} else {
if (dashboard) select(dashboard);
else {
selected.value = null;
emits("dashboard", "default");
}
@@ -154,19 +145,15 @@
onBeforeMount(() => fetchDashboards());
const tenant = ref();
watch(() => route.params.tenant, (t) => {
if (tenant.value !== t) {
fetchDashboards();
tenant.value = t;
}
}, {immediate: true});
watch(() => route.params?.dashboard, (val) => {
if(route.name === "home" && STORAGE_KEY) {
localStorage.setItem(STORAGE_KEY, val as string);
}
}, {immediate: true});
const tenant = ref(route.params.tenant);
watch(route, (r) => {
if (tenant.value !== r.params.tenant) {
fetchDashboards();
tenant.value = r.params.tenant;
}
},
{deep: true},
);
</script>
<style scoped lang="scss">
@@ -174,6 +161,14 @@
span{
font-size: 14px;
}
:deep(svg){
color: var(--ks-content-tertiary);
font-size: 1.10rem;
position: absolute;
bottom: -0.10rem;
right: 0.08rem;
}
}
.dropdown {
width: 300px;

View File

@@ -121,32 +121,16 @@
const dataset = chart.data.datasets[0];
const meta = chart.getDatasetMeta(0);
//dynamically calculate thickness based on chart size
const chartArea = chart.chartArea;
if (!chartArea || !meta || !meta.data) return;
// Available radius = half of the smaller dimension (width or height)
const availableRadius = Math.min(chartArea.width, chartArea.height) / 2;
// define thickness bounds relative to available radius
const minThicknessPx = Math.max(6, availableRadius * 0.05); // >0
const maxThicknessPx = Math.max(12, availableRadius * 0.3); // >0
// Reading weights from dataset with fallback weight(1)
const weights: number[] = (dataset.thicknessWeight && Array.isArray(dataset.thicknessWeight))? dataset.thicknessWeight.map((w: any) =>
{
const n = Number(w);
return Number.isFinite(n) ? Math.min(Math.max(n, 0), 1) : 1;
})
: meta.data.map(() => 1);
for (let i = 0; i < meta.data.length; i++) {
const arc = meta.data[i];
const w = weights[i] ?? 1;
const thicknessPx = minThicknessPx + w * (maxThicknessPx - minThicknessPx);
const thicknessScale = dataset.thicknessScale;
const baseRadius = arc.innerRadius ?? Math.max(0, availableRadius - thicknessPx);
arc.outerRadius = baseRadius + thicknessPx;
meta.data.forEach((arc, index) => {
const baseRadius = arc.innerRadius;
const additionalThickness = thicknessScale[index];
arc.outerRadius = baseRadius + additionalThickness;
arc.innerRadius = baseRadius;
arc.draw(ctx);
}
});
},
};
@@ -215,9 +199,9 @@
</script>
<style scoped lang="scss">
.chart {
height: 100% !important;
width: 100% !important;
$height: 200px;
.chart {
max-height: $height;
}
</style>

View File

@@ -46,7 +46,7 @@
import Duration from "./table/columns/Duration.vue";
import Link from "./table/columns/Link.vue";
import Namespace from "./table/columns/Namespace.vue";
import {Status} from "@kestra-io/ui-libs";
import Status from "../../Status.vue";
import Pagination from "../../layout/Pagination.vue";
import NoData from "../../layout/NoData.vue";

View File

@@ -60,7 +60,7 @@
import type cytoscape from "cytoscape";
import Link from "./Link.vue";
import {Status} from "@kestra-io/ui-libs";
import Status from "../../Status.vue";
import OpenInNew from "vue-material-design-icons/OpenInNew.vue";

View File

@@ -18,6 +18,7 @@
import TopNavBar from "../layout/TopNavBar.vue";
import {useDocStore} from "../../stores/doc";
import DocsLayout from "./DocsLayout.vue";
//@ts-expect-error no declaration
import Toc from "./Toc.vue";
import {computed,ref,watch,getCurrentInstance} from "vue";
import {useRoute} from "vue-router";

View File

@@ -72,7 +72,9 @@
import StateMachine from "vue-material-design-icons/StateMachine.vue";
import {State, Status} from "@kestra-io/ui-libs";
import Status from "../../components/Status.vue";
import {State} from "@kestra-io/ui-libs";
import * as ExecutionUtils from "../../utils/executionUtils";
import permission from "../../models/permission";
import action from "../../models/action";

View File

@@ -68,7 +68,8 @@
import {useExecutionsStore} from "../../stores/executions";
import permission from "../../models/permission";
import action from "../../models/action";
import {State, Status} from "@kestra-io/ui-libs"
import {State} from "@kestra-io/ui-libs"
import Status from "../../components/Status.vue";
import * as ExecutionUtils from "../../utils/executionUtils";
import {shallowRef} from "vue";
import {useAuthStore} from "override/stores/auth"

View File

@@ -406,7 +406,7 @@
import Id from "../Id.vue";
import Kicon from "../Kicon.vue";
import {State, Status} from "@kestra-io/ui-libs";
import Status from "../Status.vue";
import Labels from "../layout/Labels.vue";
import DateAgo from "../layout/DateAgo.vue";
import DataTable from "../layout/DataTable.vue";
@@ -420,6 +420,7 @@
//@ts-expect-error no declaration file
import TriggerFlow from "../../components/flows/TriggerFlow.vue";
import {State} from "@kestra-io/ui-libs";
import {filterValidLabels} from "./utils";
import {useToast} from "../../utils/toast";
import {storageKeys} from "../../utils/constants";

View File

@@ -59,13 +59,13 @@
</span>
</template>
<div
:style="item.parentEndPercent !== undefined ? {left: `${item.start}%`, width: `${item.parentEndPercent - item.start}%`} : {left: `${item.start}%`, width: `${Math.max(item.width, 3)}%`}"
:style="{left: `${item.start}%`, width: `${Math.max(item.width, 3)}%`}"
class="task-progress"
>
<div class="progress">
<div
:style="{left: `${Math.min(item.left, 90)}%`, width: `${Math.max(100 - item.left, 10)}%`}"
class="progress-bar"
:style="{left: `${Math.min(item.left, 90)}%`, width: `${Math.max(100 - item.left, 10)}%`}"
:class="'bg-' + item.color + (item.running ? ' progress-bar-striped progress-bar-animated' : '')"
role="progressbar"
/>
@@ -195,7 +195,7 @@
const sortedTasks = []
const tasksById = {}
for (let task of (this.execution.taskRunList || [])) {
const taskWrapper = {task, depth: task.parentTaskRunId ? undefined : 0}
const taskWrapper = {task}
if (task.parentTaskRunId) {
childTasks.push(taskWrapper)
} else {
@@ -208,7 +208,6 @@
const taskWrapper = childTasks[i];
const parentTask = tasksById[taskWrapper.task.parentTaskRunId]
if (parentTask) {
taskWrapper.depth = parentTask.depth + 1
tasksById[taskWrapper.task.id] = taskWrapper
if (!parentTask.children) {
parentTask.children = []
@@ -223,7 +222,7 @@
return nodeStart(n1) > nodeStart(n2) ? 1 : -1
})
for (let node of nodes) {
sortedTasks.push(node)
sortedTasks.push(node.task)
if (node.children) {
childrenSort(node.children)
}
@@ -266,11 +265,8 @@
}
const series = [];
const executionDelta = this.delta();
const taskMap = {};
for (let taskWrapper of this.tasks) {
let task = taskWrapper.task
const executionDelta = this.delta(); //caching this value matters
for (let task of this.tasks) {
let stopTs;
if (State.isRunning(task.state.current)) {
stopTs = ts(new Date());
@@ -299,21 +295,13 @@
let width = (stop / executionDelta) * 100
if (State.isRunning(task.state.current)) {
width = ((this.stop() - startTs) / executionDelta) * 100
width = ((this.stop() - startTs) / executionDelta) * 100 //(stop / executionDelta) * 100
}
let startPercent = (start / executionDelta) * 100;
let parentEndPercent = undefined;
if (task.parentTaskRunId && taskMap[task.parentTaskRunId]) {
const parent = taskMap[task.parentTaskRunId];
parentEndPercent = parent.start + parent.width;
}
const seriesItem = {
series.push({
id: task.id,
name: task.taskId,
start: startPercent,
start: (start / executionDelta) * 100,
width,
left: left,
tooltip,
@@ -323,13 +311,8 @@
flowId: task.flowId,
namespace: task.namespace,
executionId: task.outputs && task.outputs.executionId,
attempts: task.attempts ? task.attempts.length : 1,
depth: taskWrapper.depth,
parentEndPercent: parentEndPercent
};
taskMap[task.id] = seriesItem;
series.push(seriesItem);
attempts: task.attempts ? task.attempts.length : 1
});
}
this.series = series;
},

View File

@@ -204,7 +204,7 @@
</template>
<script>
import {Status} from "@kestra-io/ui-libs";
import Status from "../Status.vue";
import SetLabels from "./SetLabels.vue";
import Restart from "./Restart.vue";
import Resume from "./Resume.vue";

View File

@@ -111,7 +111,7 @@
:readOnly="true"
component="el-dropdown-item"
:taskId="currentTaskRun.taskId"
section="tasks"
:section="SECTIONS.TASKS"
:flowId="followedExecution.flowId"
:namespace="followedExecution.namespace"
:revision="followedExecution.flowRevision"
@@ -175,7 +175,7 @@
<script>
import Restart from "./Restart.vue";
import Metrics from "./Metrics.vue";
import {State, Status} from "@kestra-io/ui-libs";
import Status from "../Status.vue";
import ChangeStatus from "./ChangeStatus.vue";
import TaskEdit from "../flows/TaskEdit.vue";
import SubFlowLink from "../flows/SubFlowLink.vue";
@@ -189,6 +189,7 @@
import Download from "vue-material-design-icons/Download.vue";
import WorkerInfo from "./WorkerInfo.vue";
import AiIcon from "../ai/AiIcon.vue";
import {State} from "@kestra-io/ui-libs"
import FlowUtils from "../../utils/flowUtils";
import _groupBy from "lodash/groupBy";
import {TaskIcon, SECTIONS} from "@kestra-io/ui-libs";

View File

@@ -49,7 +49,8 @@
import {useExecutionsStore} from "../../stores/executions";
import permission from "../../models/permission";
import action from "../../models/action";
import {State, Status} from "@kestra-io/ui-libs"
import {State} from "@kestra-io/ui-libs"
import Status from "../../components/Status.vue";
import {useAuthStore} from "override/stores/auth"
import {useI18n} from "vue-i18n";
import {useToast} from "../../utils/toast";

View File

@@ -32,18 +32,6 @@
<span v-else-if="value === null">
<em>null</em>
</span>
<div v-else-if="isComplexValue(value)">
<Editor
:readOnly="true"
:input="true"
:fullHeight="false"
:customHeight="Math.min(20, Math.max(5, JSON.stringify(getDisplayValue(value), null, 2).split('\n').length))"
:navbar="false"
:modelValue="JSON.stringify(getDisplayValue(value), null, 2)"
lang="json"
class="complex-value-editor"
/>
</div>
<span v-else>
{{ value }}
</span>
@@ -54,7 +42,6 @@
import Download from "vue-material-design-icons/Download.vue";
import OpenInNew from "vue-material-design-icons/OpenInNew.vue";
import FilePreview from "./FilePreview.vue";
import Editor from "../inputs/Editor.vue";
import {apiUrl} from "override/utils/route";
import Utils from "../../utils/utils";
@@ -101,42 +88,6 @@
}
};
const isComplexValue = (value: unknown): boolean => {
if ((typeof value === "object" && value !== null) || Array.isArray(value)) {
return true;
}
if (typeof value === "string") {
try {
const parsed = JSON.parse(value);
return (typeof parsed === "object" && parsed !== null) || Array.isArray(parsed);
} catch {
return false;
}
}
return false;
};
const getDisplayValue = (value: unknown): unknown => {
if ((typeof value === "object" && value !== null) || Array.isArray(value)) {
return value;
}
if (typeof value === "string") {
try {
const parsed = JSON.parse(value);
if ((typeof parsed === "object" && parsed !== null) || Array.isArray(parsed)) {
return parsed;
}
} catch {
return value;
}
}
return value;
};
const itemUrl = (value: string): string => {
return `${apiUrl()}/executions/${props.execution?.id}/file?path=${encodeURI(value)}`;
};
@@ -167,11 +118,3 @@
getFileSize();
});
</script>
<style scoped lang="scss">
.complex-value-editor {
margin-top: 0.5rem;
border: 1px solid var(--ks-border-primary);
border-radius: 4px;
}
</style>

View File

@@ -183,4 +183,4 @@ export function useExecutionRoot() {
getBaseTabs,
setupLifecycle
};
}
}

View File

@@ -225,15 +225,14 @@
const debugOutput = JSON.stringify(parsedResult, null, 2);
debugExpression.value = debugOutput;
if (response.status === 200 && debugOutput !== null && debugOutput !== undefined) {
selected.value.push(debugOutput);
}
selected.value.push(debugOutput);
isJSON.value = true;
} catch {
debugExpression.value = response.data.result;
// Parsing failed, therefore, copy raw result
if (response.status === 200 && response.data.result !== null && response.data.result !== undefined)
if (response.status === 200 && response.data.result)
selected.value.push(response.data.result);
}

View File

@@ -71,7 +71,7 @@
<script setup lang="ts">
import {computed, ref} from "vue";
import {Magnify, InformationOutline} from "../../utils/icons";
import {Status} from "@kestra-io/ui-libs";
import Status from "../../../Status.vue";
import Checkbox from "../../../layout/Checkbox.vue";
const props = defineProps<{
@@ -243,7 +243,7 @@
}
}
button.status-button {
button.el-button {
width: 10rem;
}
</style>

View File

@@ -36,7 +36,8 @@
import {ref, computed} from "vue";
import Executions from "../executions/Executions.vue";
import Empty from "../layout/empty/Empty.vue";
import {State, Status} from "@kestra-io/ui-libs";
import {State} from "@kestra-io/ui-libs";
import Status from "../Status.vue";
import {useFlowStore} from "../../stores/flow";
defineOptions({inheritAttrs: false});

View File

@@ -33,7 +33,7 @@
import FlowRootTopBar from "./FlowRootTopBar.vue";
import FlowConcurrency from "./FlowConcurrency.vue";
import DemoAuditLogs from "../demo/AuditLogs.vue";
import {useAuthStore} from "override/stores/auth";
import {useAuthStore} from "override/stores/auth"
import {useMiscStore} from "override/stores/misc";
export default {
@@ -59,12 +59,13 @@
"$route.params.tab": {
immediate: true,
handler: function (newTab) {
if (newTab === "overview" || newTab === "executions") {
if (newTab === "overview") {
const dateTimeKeys = ["startDate", "endDate", "timeRange"];
if (!Object.keys(this.$route.query).some((key) => dateTimeKeys.some((dateTimeKey) => key.includes(dateTimeKey)))) {
const DEFAULT_DURATION = this.miscStore.configs?.chartDefaultDuration ?? "P30D";
const newQuery = {...this.$route.query, "filters[timeRange][EQUALS]": DEFAULT_DURATION};
const miscStore = useMiscStore();
const defaultDuration = miscStore.configs?.chartDefaultDuration || "P30D";
const newQuery = {...this.$route.query, "filters[timeRange][EQUALS]": defaultDuration};
this.$router.replace({name: this.$route.name, params: this.$route.params, query: newQuery});
}
}
@@ -313,7 +314,7 @@
}
},
computed: {
...mapStores(useCoreStore, useFlowStore, useAuthStore, useMiscStore),
...mapStores(useCoreStore, useFlowStore, useAuthStore),
routeInfo() {
return {
title: this.$route.params.id,

View File

@@ -204,7 +204,6 @@
<template #default="scope">
<TimeSeries
:chart="mappedChart(scope.row.id, scope.row.namespace)"
:filters="chartFilters()"
showDefault
short
/>
@@ -267,7 +266,7 @@
import FileDocumentRemoveOutline from "vue-material-design-icons/FileDocumentRemoveOutline.vue";
import Kicon from "../Kicon.vue";
import {Status} from "@kestra-io/ui-libs";
import Status from "../Status.vue";
import Labels from "../layout/Labels.vue";
import DateAgo from "../layout/DateAgo.vue";
import TriggerAvatar from "./TriggerAvatar.vue";
@@ -288,7 +287,6 @@
import {useFlowStore} from "../../stores/flow";
import {useAuthStore} from "override/stores/auth";
import {useMiscStore} from "override/stores/misc";
import {useExecutionsStore} from "../../stores/executions";
import {useTableColumns} from "../../composables/useTableColumns";
@@ -309,7 +307,6 @@
const flowStore = useFlowStore();
const authStore = useAuthStore();
const executionsStore = useExecutionsStore();
const miscStore = useMiscStore();
const route = useRoute();
const router = useRouter();
@@ -625,15 +622,6 @@
return MAPPED_CHARTS;
}
function chartFilters() {
const DEFAULT_DURATION = miscStore.configs?.chartDefaultDuration ?? "P30D";
return [{
field: "timeRange",
value: DEFAULT_DURATION,
operation: "EQUALS"
}];
}
onMounted(() => {
const query = {...route.query};
const queryKeys = Object.keys(query);

View File

@@ -169,14 +169,9 @@
? revisions.value?.[props.revision - 1]?.source
: flowStore.flow?.source;
});
// Methods
const load = async (taskId: string) => {
await flowStore.loadFlow({
namespace: props.namespace,
id: props.flowId,
revision: props.revision?.toString(),
});
if (props.revision) {
if (!revisions.value?.[props.revision - 1]) {
revisions.value = await flowStore.loadRevisions({

View File

@@ -34,7 +34,7 @@
import Markdown from "../layout/Markdown.vue";
import Cron from "../layout/Cron.vue";
const {t, te} = useI18n();
const {t} = useI18n();
defineProps<{
data: Record<string, any>;
@@ -51,10 +51,9 @@
"date": "last trigger date",
"updatedDate": "context updated date",
"evaluateRunningDate": "evaluation lock date",
"states": "trigger_states",
};
const translationKey = mappings[key] ?? key;
return te(translationKey) && t(translationKey) || translationKey;
return t(translationKey);
};
</script>

View File

@@ -18,7 +18,7 @@
</template>
<script setup lang="ts">
import {Status} from "@kestra-io/ui-libs";
import Status from "../../Status.vue";
import {date, humanizeDuration} from "../../../utils/filters";
import {Execution, useExecutionsStore} from "../../../stores/executions";

View File

@@ -292,8 +292,6 @@
return editor?.getEditorType() === monacoEditor.value?.monaco.editor.EditorType.ICodeEditor
}
const scrollMemory = props.scrollKey ? useScrollMemory(ref(props.scrollKey)) : null;
function editorDidMount(monacoMounted?: monaco.editor.IStandaloneCodeEditor | monaco.editor.IStandaloneDiffEditor) {
const monacoRef = monacoEditor.value
@@ -305,6 +303,8 @@
return;
}
// avoid double import of monaco editor, use a reference
const KeyCode = monacoRef.monaco.KeyCode;
const KeyMod = monacoRef.monaco.KeyMod;
@@ -316,7 +316,7 @@
}
const codeEditor = editor as monaco.editor.IStandaloneCodeEditor;
const scrollMemory = props.scrollKey ? useScrollMemory(ref(props.scrollKey)) : null;
if (props.scrollKey && scrollMemory) {
const savedState = scrollMemory.loadData<monaco.editor.ICodeEditorViewState>("viewState");

View File

@@ -275,10 +275,6 @@
}, 1000);
}
onBeforeUnmount(() => {
clearTimeout(timeout.value);
});
function updatePluginDocumentation(event: any) {
const source = event.model.getValue();

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@@ -20,7 +20,8 @@
</el-select>
</template>
<script>
import {State, Status} from "@kestra-io/ui-libs"
import {State} from "@kestra-io/ui-libs"
import Status from "../Status.vue";
export default {
components: {Status},

View File

@@ -31,7 +31,6 @@
const namespace = computed(() => route.params?.id) as Ref<string>;
const miscStore = useMiscStore();
const namespacesStore = useNamespacesStore();
watch(namespace, (newID) => {
@@ -41,12 +40,13 @@
});
watch(() => route.params.tab, (newTab) => {
if (newTab === "overview" || newTab === "executions") {
if (newTab === "overview") {
const dateTimeKeys = ["startDate", "endDate", "timeRange"];
if (!Object.keys(route.query).some((key) => dateTimeKeys.some((dateTimeKey) => key.includes(dateTimeKey)))) {
const DEFAULT_DURATION = miscStore.configs?.chartDefaultDuration ?? "P30D";
const newQuery = {...route.query, "filters[timeRange][EQUALS]": DEFAULT_DURATION};
const miscStore = useMiscStore();
const defaultDuration = miscStore.configs?.chartDefaultDuration || "P30D";
const newQuery = {...route.query, "filters[timeRange][EQUALS]": defaultDuration};
router.replace({name: route.name, params: route.params, query: newQuery});
}
}

View File

@@ -8,15 +8,10 @@
</template>
<script setup lang="ts">
import {computed} from "vue";
import {computed, getCurrentInstance} from "vue";
import {useI18n} from "vue-i18n";
import InputText from "../inputs/InputText.vue";
import {useMiscStore} from "override/stores/misc";
const miscStore = useMiscStore();
const disabled = computed(() => miscStore.configs?.edition === "OSS")
const {t} = useI18n()
const modelValue = defineModel<string>({default: ""})
@@ -25,4 +20,8 @@
name: "TaskVersion",
inheritAttrs: false,
})
const disabled = computed(() => {
return Boolean(getCurrentInstance()?.appContext.config.globalProperties.$isOss)
})
</script>

View File

@@ -31,18 +31,13 @@
noUrlChange
>
<template #markdown="{content}">
<!-- Plugin schema content: search disabled -->
<Markdown
font-size-var="font-size-base"
:source="content"
/>
<EnhancedMarkdown font-size-var="font-size-base" :source="content" :showSearch="false" />
</template>
</SchemaToHtml>
</Suspense>
</template>
<Markdown
v-else-if="introContent"
<EnhancedMarkdown
v-else
:source="introContent"
:class="{'position-absolute': absolute}"
:showSearch="true"
@@ -54,7 +49,7 @@
<script setup lang="ts">
import {computed} from "vue";
import Markdown from "../layout/Markdown.vue";
import EnhancedMarkdown from "../layout/EnhancedMarkdown.vue";
import {SchemaToHtml, TaskIcon} from "@kestra-io/ui-libs";
import {getPluginReleaseUrl} from "../../utils/pluginUtils";
import {useMiscStore} from "override/stores/misc";

View File

@@ -4,18 +4,23 @@
</div>
</template>
<script setup lang="ts">
import {computed} from "vue";
const props = defineProps<{
encodedSvg?: string;
}>();
const styles = computed(() => ({
backgroundImage: props.encodedSvg
? `url(data:image/svg+xml;base64,${props.encodedSvg})`
: "none",
}));
<script>
export default {
name: "SvgDisplay",
props: {
encodedSvg: {
type: String,
default: undefined
}
},
computed: {
styles() {
return {
backgroundImage: `url(data:image/svg+xml;base64,${this.encodedSvg})`
}
},
}
}
</script>
<style scoped lang="scss">

View File

@@ -78,7 +78,6 @@ export const useBaseNamespacesStore = () => {
}
const data = response.data;
const contentLength = response.headers?.["content-length"];
if (contentLength === (data.length + 2).toString()) {
return `"${data}"`;
}

View File

@@ -27,13 +27,13 @@ export function useScrollMemory(keyRef: Ref<string>, elementRef?: Ref<HTMLElemen
: () => { if (elementRef?.value) elementRef.value.scrollTo({top: scrollTop, behavior: "smooth"}) }
setTimeout(applyScroll, 10)
}
onActivated(() => nextTick(restoreScroll))
const throttledSave = useThrottleFn((top: number) => saveScroll(top), 100)
if (useWindow) {
const {y} = useWindowScroll({throttle: 16, onScroll: () => throttledSave(y.value)})
watch(keyRef, () => nextTick(restoreScroll), {immediate: true})
onActivated(() => nextTick(restoreScroll))
} else {
useScroll(elementRef || ref(null), {
throttle: 16,
@@ -42,6 +42,7 @@ export function useScrollMemory(keyRef: Ref<string>, elementRef?: Ref<HTMLElemen
watch([keyRef, () => elementRef?.value], ([newKey, newElement]) => {
if (newElement && newKey) nextTick(restoreScroll)
}, {immediate: true})
onActivated(() => nextTick(restoreScroll))
}
return {saveData: saveToStorage, loadData: loadFromStorage}

View File

@@ -91,6 +91,8 @@ initApp(app, routes, null, en).then(({router, piniaStore}) => {
});
}, null, router, true);
app.config.globalProperties.$isOss = true; // Set to true for OSS version
// mount
app.mount("#app")
});

View File

@@ -12,7 +12,7 @@
import Auth from "../../override/components/auth/Auth.vue";
withDefaults(defineProps<{
showLink?: boolean
showLink: boolean
}>(), {
showLink: true
});

View File

@@ -1,8 +1,4 @@
<template>
<Dashboards
v-if="tab === 'overview' && ALLOWED_CREATION_ROUTES.includes(String(route.name))"
@dashboard="onSelectDashboard"
/>
<Action
v-if="deleted"
type="default"
@@ -38,20 +34,12 @@
import Action from "../../../components/namespaces/components/buttons/Action.vue";
// @ts-expect-error does not have types
import TriggerFlow from "../../../components/flows/TriggerFlow.vue";
import Dashboards from "../../../components/dashboard/components/selector/Selector.vue";
import {ALLOWED_CREATION_ROUTES} from "../../../components/dashboard/composables/useDashboards";
import permission from "../../../models/permission";
import action from "../../../models/action";
import {useAuthStore} from "override/stores/auth";
const {t} = useI18n();
const onSelectDashboard = (value: any) => {
router.replace({
params: {...route.params, dashboard: value}
});
};
const coreStore = useCoreStore();
const flowStore = useFlowStore();
const router = useRouter();

View File

@@ -1,9 +1,4 @@
<template>
<Dashboards
v-if="tab === 'overview' && ALLOWED_CREATION_ROUTES.includes(String(route.name))"
@dashboard="onSelectDashboard"
/>
<Action
v-if="tab === 'flows'"
:label="t('create_flow')"
@@ -26,25 +21,17 @@
<script setup lang="ts">
import {computed, Ref} from "vue";
import {useRoute, useRouter} from "vue-router";
import {useRoute} from "vue-router";
import {useI18n} from "vue-i18n";
import {useNamespacesStore} from "override/stores/namespaces";
import Action from "../../../components/namespaces/components/buttons/Action.vue";
import Dashboards from "../../../components/dashboard/components/selector/Selector.vue";
import {ALLOWED_CREATION_ROUTES} from "../../../components/dashboard/composables/useDashboards";
import FamilyTree from "vue-material-design-icons/FamilyTree.vue";
const route = useRoute();
const router = useRouter();
const {t} = useI18n({useScope: "global"});
const namespacesStore = useNamespacesStore();
const onSelectDashboard = (value: any) => {
router.replace({
params: {...route.params, dashboard: value}
});
};
const tab = computed(() => route.params?.tab);
const namespace = computed(() => route.params?.id) as Ref<string>;
</script>

View File

@@ -54,6 +54,7 @@ interface FlowValidations {
export interface Flow {
id: string;
namespace: string;
disabled?: boolean;
source: string;
revision?: number;
deleted?: boolean;
@@ -96,8 +97,6 @@ export const useFlowStore = defineStore("flow", () => {
const axios = useAxios();
const coreStore = useCoreStore();
const t = (key: string, values?: Record<string, any>) => {
if (!globalI18n.value) {
return key;
@@ -167,7 +166,7 @@ export const useFlowStore = defineStore("flow", () => {
if (flowBeforeEdit &&
(flowOnValidation.id !== flowBeforeEdit.id ||
flowOnValidation.namespace !== flowBeforeEdit.namespace)) {
const coreStore = useCoreStore();
coreStore.message = {
variant: "error",
title: t("readonly property"),
@@ -184,6 +183,7 @@ export const useFlowStore = defineStore("flow", () => {
}
}
const coreStore = useCoreStore();
coreStore.unsavedChange = true;
return validateFlow({
@@ -210,6 +210,7 @@ export const useFlowStore = defineStore("flow", () => {
const flowSource = flowYaml.value ?? "";
if (flowParsed.value === undefined) {
const coreStore = useCoreStore();
coreStore.message = {
variant: "error",
title: t("invalid flow"),
@@ -250,6 +251,7 @@ export const useFlowStore = defineStore("flow", () => {
await createFlow({flow: flowSource ?? ""})
.then((response: Flow) => {
toast.saved(response.id);
const coreStore = useCoreStore();
coreStore.unsavedChange = false;
isCreating.value = false;
});
@@ -257,6 +259,7 @@ export const useFlowStore = defineStore("flow", () => {
await saveFlow({flow: flowSource})
.then((response: Flow) => {
toast.saved(response.id);
const coreStore = useCoreStore();
coreStore.unsavedChange = false;
});
}
@@ -351,6 +354,7 @@ export const useFlowStore = defineStore("flow", () => {
})
.then((response: any) => {
if (response.data.exception) {
const coreStore = useCoreStore();
coreStore.message = {
title: "Invalid source code",
message: response.data.exception,
@@ -550,6 +554,7 @@ function deleteFlowAndDependencies() {
}
if ([404, 422].includes(error.response?.status) && config?.params?.subflows?.length > 0) {
const coreStore = useCoreStore();
coreStore.message = {
title: "Couldn't expand subflow",
message: error.response.data.message,

View File

@@ -4,7 +4,7 @@ import {apiUrl} from "override/utils/route";
export const useKvStore = defineStore("kv", () => {
async function find(params: {page: number, size: number, filters: {[key: string]: {EQUALS: string}}}) {
const {data} = await axios.get(`${apiUrl()}/kv`, {withCredentials: true, params});
const {data} = await axios.get(`${apiUrl()}/kv`, {params});
return data;
}

View File

@@ -1,5 +1,5 @@
import {defineStore} from "pinia"
import {ref, computed, toRaw, nextTick} from "vue";
import {ref, computed, toRaw} from "vue";
import {trackPluginDocumentationView} from "../utils/tabTracking";
import {apiUrlWithoutTenants} from "override/utils/route";
import semver from "semver";
@@ -74,6 +74,7 @@ export const usePluginsStore = defineStore("plugins", () => {
const inputSchema = ref<any>();
const inputsType = ref<any>();
const schemaType = ref<Record<string, any>>();
const currentlyLoading = ref<{type?: string; version?: string}>();
const forceIncludeProperties = ref<string[]>();
const _iconsPromise = ref<Promise<Record<string, string>>>();
@@ -165,9 +166,7 @@ export const usePluginsStore = defineStore("plugins", () => {
const id = options.version ? `${options.cls}/${options.version}` : options.cls;
const cachedPluginDoc = pluginsDocumentation.value[options.hash ? options.hash + id : id];
if (!options.all && cachedPluginDoc) {
nextTick(() => {
plugin.value = cachedPluginDoc;
})
plugin.value = cachedPluginDoc;
return cachedPluginDoc;
}
@@ -277,19 +276,17 @@ export const usePluginsStore = defineStore("plugins", () => {
});
}
let currentlyLoading: {type?: string; version?: string} | undefined = undefined;
async function updateDocumentation(pluginElement?: ({type: string, version?: string, forceRefresh?: boolean} & Record<string, any>) | undefined) {
if (!pluginElement?.type || !allTypes.value.includes(pluginElement.type)) {
editorPlugin.value = undefined;
currentlyLoading = undefined;
currentlyLoading.value = undefined;
return;
}
const {type, version, forceRefresh = false} = pluginElement;
if (currentlyLoading?.type === type &&
currentlyLoading?.version === version &&
if (currentlyLoading.value?.type === type &&
currentlyLoading.value?.version === version &&
!forceRefresh) {
return
}
@@ -314,22 +311,22 @@ export const usePluginsStore = defineStore("plugins", () => {
}
}
currentlyLoading = {
currentlyLoading.value = {
type,
version,
};
const pluginData = await load(payload);
editorPlugin.value = {
cls: type,
version,
...pluginData,
};
load(payload).then((pluginData) => {
editorPlugin.value = {
cls: type,
version,
...pluginData,
};
trackPluginDocumentationView(type);
trackPluginDocumentationView(type);
forceIncludeProperties.value = Object.keys(pluginElement).filter(k => k !== "type" && k !== "version" && k !== "forceRefresh");
forceIncludeProperties.value = Object.keys(pluginElement).filter(k => k !== "type" && k !== "version" && k !== "forceRefresh");
});
}
return {

View File

@@ -4,7 +4,7 @@ import {apiUrl} from "override/utils/route";
export const useSecretsStore = defineStore("secrets", () => {
async function find(params: {page: number, size: number, filters: {[key: string]: {EQUALS: string}}}) {
const {data} = await axios.get(`${apiUrl()}/secrets`, {withCredentials: true, params});
const {data} = await axios.get(`${apiUrl()}/secrets`, {params});
return data;
}

View File

@@ -36,12 +36,6 @@ interface TriggerBulkOptions {
[key: string]: any;
}
interface TriggerDeleteOptions {
namespace: string;
flowId: string;
triggerId: string;
}
export const useTriggerStore = defineStore("trigger", {
state: () => ({}),
@@ -138,21 +132,6 @@ export const useTriggerStore = defineStore("trigger", {
async setDisabledByTriggers(options: TriggerBulkOptions) {
const response = await this.$http.post(`${apiUrl()}/triggers/set-disabled/by-triggers`, options);
return response.data;
},
async delete(options: TriggerDeleteOptions) {
const response = await this.$http.delete(`${apiUrl()}/triggers/${options.namespace}/${options.flowId}/${options.triggerId}`);
return response.data;
},
async deleteByQuery(options: TriggerBulkOptions) {
const response = await this.$http.post(`${apiUrl()}/triggers/delete/by-query`, null, {params: options});
return response.data;
},
async deleteByTriggers(options: TriggerBulkOptions) {
const response = await this.$http.delete(`${apiUrl()}/triggers/delete/by-triggers`, options);
return response.data;
},
}
}
});

View File

@@ -1029,7 +1029,7 @@ form.ks-horizontal {
@include res(lg) {
width: 35%;
min-width: auto;
min-width: 800px;
}
}
}

View File

@@ -98,7 +98,6 @@
"bulk change state": "Sind Sie sicher, dass Sie den Zustand von <code>{executionCount}</code> Ausführung(en) ändern möchten?",
"bulk delete": "Sind Sie sicher, dass Sie <code>{executionCount}</code> Ausführung(en) löschen möchten?",
"bulk delete backfills": "{count} Backfill löschen",
"bulk delete triggers": "Möchten Sie wirklich {count} Trigger löschen?",
"bulk disabled status": {
"false": "{count} Trigger aktivieren",
"true": "{count} Trigger deaktivieren"
@@ -112,7 +111,6 @@
"bulk resume": "Sind Sie sicher, dass Sie <code>{executionCount}</code> Ausführung(en) fortsetzen möchten?",
"bulk set labels": "Sind Sie sicher, dass Sie Labels für <code>{executionCount}</code> Ausführung(en) festlegen möchten?",
"bulk success delete backfills": "{count} Backfills gelöscht",
"bulk success delete triggers": "{count} Trigger wurden erfolgreich gelöscht",
"bulk success disabled status": {
"false": "{count} Trigger aktiviert",
"true": "{count} Trigger deaktiviert"
@@ -295,11 +293,6 @@
"delete logs": "Logs löschen",
"delete ok": "ist gelöscht",
"delete task confirm": "Möchten Sie den Task <code>{taskId}</code> wirklich löschen?",
"delete trigger": "Trigger löschen",
"delete trigger confirmation": "Möchten Sie den Trigger {id} wirklich löschen? WARNING: Das Löschen eines Triggers kann zu doppelten Ausführungen führen, wenn der Trigger noch in einem flow aktiv ist.",
"delete trigger error": "Fehler beim Löschen des Triggers {id}",
"delete trigger success": "Trigger {id} wurde erfolgreich gelöscht",
"delete triggers": "Trigger löschen",
"delete_all_logs": "Sind Sie sicher, dass Sie alle Logs löschen möchten?",
"delete_log": "Sind Sie sicher, dass Sie das Log löschen möchten?",
"deleted": "Erfolgreich gelöscht",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "Die Verwendung von Trigger-Variablen funktioniert nicht bei manueller Flow-Ausführung. Fügen Sie den `??`-Koaleszenz-Operator hinzu, um das Problem zu lösen. Verwenden Sie zum Beispiel anstelle von `trigger.date` den Ausdruck `trigger.date ?? execution.startDate`.",
"trigger_id_exists": "Trigger-ID existiert bereits",
"trigger_id_message": "Trigger-ID ${existingTrigger} existiert bereits im Flow.",
"trigger_states": "Zustände",
"triggered": "Ausführungstrigger",
"triggered done": "Ausführung <em>{name}</em> ist erfolgreich ausgelöst",
"triggerflow disabled": "Trigger ist in der Flow-Definition deaktiviert",

View File

@@ -375,7 +375,6 @@
"trigger details": "Trigger details",
"triggerflow disabled": "Trigger is disabled from flow definition",
"last trigger date": "Last triggered date",
"trigger_states": "States",
"context updated date": "Context updated date",
"next evaluation date": "Next evaluation date",
"last trigger date tooltip": "When the trigger last executed. May be in the past when running a backfill.",
@@ -1836,13 +1835,6 @@
"search_blueprints": "Search blueprints",
"search_plugins": "Search {count}+ plugins"
}
},
"delete trigger": "Delete trigger",
"delete triggers": "Delete triggers",
"bulk delete triggers": "Are you sure you want to delete {count} triggers?",
"bulk success delete triggers": "{count} triggers have been deleted successfully",
"delete trigger confirmation": "Are you sure you want to delete trigger {id}? WARNING: deleting a trigger may lead to duplicate executions if the trigger is still active in a flow",
"delete trigger success": "Trigger {id} has been deleted successfully",
"delete trigger error": "Error deleting trigger {id}"
}
}
}

View File

@@ -98,7 +98,6 @@
"bulk change state": "¿Está seguro de que desea cambiar el estado de <code>{executionCount}</code> ejecución(es)?",
"bulk delete": "¿Estás seguro de que quieres eliminar <code>{executionCount}</code> ejecución(es)?",
"bulk delete backfills": "Eliminar {count} backfill",
"bulk delete triggers": "¿Está seguro de que desea eliminar {count} triggers?",
"bulk disabled status": {
"false": "Habilitar {count} triggers",
"true": "Deshabilitar {count} triggers"
@@ -112,7 +111,6 @@
"bulk resume": "¿Estás seguro de que quieres reanudar <code>{executionCount}</code> ejecución(es)?",
"bulk set labels": "¿Estás seguro de que quieres establecer etiquetas a <code>{executionCount}</code> ejecución(es)?",
"bulk success delete backfills": "{count} backfills eliminados",
"bulk success delete triggers": "{count} triggers han sido eliminados exitosamente",
"bulk success disabled status": {
"false": "{count} trigger(s) habilitado(s)",
"true": "{count} trigger(s) deshabilitado(s)"
@@ -295,11 +293,6 @@
"delete logs": "Eliminar logs",
"delete ok": "está eliminado",
"delete task confirm": "¿Quieres eliminar la task <code>{taskId}</code>?",
"delete trigger": "Eliminar trigger",
"delete trigger confirmation": "¿Está seguro de que desea eliminar el trigger {id}? WARNING: eliminar un trigger puede llevar a ejecuciones duplicadas si el trigger aún está activo en un flow.",
"delete trigger error": "Error al eliminar trigger {id}",
"delete trigger success": "El trigger {id} ha sido eliminado con éxito",
"delete triggers": "Eliminar triggers",
"delete_all_logs": "¿Estás seguro de que quieres eliminar todos los logs?",
"delete_log": "¿Está seguro de que desea eliminar el log?",
"deleted": "Eliminado exitosamente",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "Al usar variables de trigger no funcionará con la ejecución manual de flow. Agrega el operador de coalescencia `??` para resolver el problema. Por ejemplo, en lugar de `trigger.date`, usa `trigger.date ?? execution.startDate`.",
"trigger_id_exists": "El Trigger Id ya existe",
"trigger_id_message": "El Trigger Id ${existingTrigger} ya existe en el flow.",
"trigger_states": "Estados",
"triggered": "Trigger de ejecución",
"triggered done": "Ejecución <em>{name}</em> se ha activado exitosamente",
"triggerflow disabled": "El trigger está deshabilitado en la definición del flujo",

View File

@@ -98,7 +98,6 @@
"bulk change state": "Êtes-vous sûr de vouloir changer l'état de <code>{executionCount}</code> exécution(s) ?",
"bulk delete": "Êtes-vous sur de vouloir supprimer <code>{executionCount}</code> exécution(s) ?",
"bulk delete backfills": "Supprimer {count} backfill",
"bulk delete triggers": "Êtes-vous sûr de vouloir supprimer {count} triggers ?",
"bulk disabled status": {
"false": "Activer {count} triggers",
"true": "Désactiver {count} triggers"
@@ -112,7 +111,6 @@
"bulk resume": "Êtes-vous sur de vouloir reprendre <code>{executionCount}</code> exécution(s)?",
"bulk set labels": "Êtes-vous sûr de vouloir ajouter des labels à <code>{executionCount}</code> exécutions(s)?",
"bulk success delete backfills": "{count} backfills supprimés",
"bulk success delete triggers": "{count} triggers ont été supprimés avec succès",
"bulk success disabled status": {
"false": "{count} triggers activés",
"true": "{count} triggers désactivés"
@@ -295,11 +293,6 @@
"delete logs": "Supprimer les logs",
"delete ok": "est effacé",
"delete task confirm": "Êtes-vous sûr de vouloir supprimer la tâche <code>{taskId}</code> ?",
"delete trigger": "Supprimer le trigger",
"delete trigger confirmation": "Êtes-vous sûr de vouloir supprimer le trigger {id} ? WARNING : supprimer un trigger peut entraîner des exécutions en double si le trigger est toujours actif dans un flow.",
"delete trigger error": "Erreur lors de la suppression du trigger {id}",
"delete trigger success": "Le trigger {id} a été supprimé avec succès",
"delete triggers": "Supprimer les triggers",
"delete_all_logs": "Êtes-vous sûr de vouloir supprimer tous les journaux ?",
"delete_log": "Êtes-vous sûr de vouloir supprimer le log ?",
"deleted": "Effacement réussi",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "L'utilisation des variables de trigger ne fonctionnera pas avec l'exécution manuelle du flow. Ajoutez l'opérateur de coalescence `??` pour résoudre le problème. Par exemple, au lieu de `trigger.date`, utilisez `trigger.date ?? execution.startDate`.",
"trigger_id_exists": "L'identifiant de trigger existe déjà",
"trigger_id_message": "L'ID de trigger ${existingTrigger} existe déjà dans le flow.",
"trigger_states": "États",
"triggered": "Exécution déclenchée",
"triggered done": "L'exécution <em>{name}</em> est déclenchée !",
"triggerflow disabled": "Déclencheur désactivé depuis la définition du flow",

View File

@@ -98,7 +98,6 @@
"bulk change state": "क्या आप सुनिश्चित हैं कि आप <code>{executionCount}</code> execution(s) की स्थिति बदलना चाहते हैं?",
"bulk delete": "क्या आप वाकई <code>{executionCount}</code> निष्पादन को हटाना चाहते हैं?",
"bulk delete backfills": "{count} backfill को हटाएं",
"bulk delete triggers": "क्या आप वाकई {count} triggers को हटाना चाहते हैं?",
"bulk disabled status": {
"false": "{count} triggers को सक्षम करें",
"true": "{count} triggers को अक्षम करें"
@@ -112,7 +111,6 @@
"bulk resume": "क्या आप वाकई <code>{executionCount}</code> निष्पादन को फिर से शुरू करना चाहते हैं?",
"bulk set labels": "क्या आप वाकई <code>{executionCount}</code> निष्पादन के लिए लेबल्स सेट करना चाहते हैं?",
"bulk success delete backfills": "{count} backfills हटाए गए",
"bulk success delete triggers": "{count} triggers सफलतापूर्वक हटा दिए गए हैं",
"bulk success disabled status": {
"false": "{count} trigger(s) सक्षम किए गए",
"true": "{count} trigger(s) अक्षम किए गए"
@@ -295,11 +293,6 @@
"delete logs": "Logs हटाएं",
"delete ok": "हटा दिया गया है",
"delete task confirm": "क्या आप वाकई Task <code>{taskId}</code> को हटाना चाहते हैं?",
"delete trigger": "ट्रिगर हटाएं",
"delete trigger confirmation": "क्या आप वाकई trigger {id} को हटाना चाहते हैं? WARNING: यदि trigger अभी भी किसी flow में सक्रिय है, तो उसे हटाने से duplicate executions हो सकते हैं।",
"delete trigger error": "ट्रिगर {id} को हटाने में त्रुटि",
"delete trigger success": "ट्रिगर {id} को सफलतापूर्वक हटा दिया गया है",
"delete triggers": "ट्रिगर्स हटाएं",
"delete_all_logs": "क्या आप वाकई सभी Logs हटाना चाहते हैं?",
"delete_log": "क्या आप वाकई log को हटाना चाहते हैं?",
"deleted": "सफलतापूर्वक हटाया गया",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "ट्रिगर वेरिएबल्स का उपयोग मैनुअल flow निष्पादन के साथ काम नहीं करेगा। समस्या को हल करने के लिए `??` कोएलस ऑपरेटर जोड़ें। उदाहरण के लिए, `trigger.date` के बजाय `trigger.date ?? execution.startDate` का उपयोग करें।",
"trigger_id_exists": "ट्रिगर Id पहले से मौजूद है",
"trigger_id_message": "ट्रिगर Id ${existingTrigger} पहले से ही flow में मौजूद है।",
"trigger_states": "राज्य",
"triggered": "निष्पादन trigger",
"triggered done": "निष्पादन <em>{name}</em> सफलतापूर्वक ट्रिगर किया गया है",
"triggerflow disabled": "Flow परिभाषा से Trigger अक्षम है",

View File

@@ -98,7 +98,6 @@
"bulk change state": "Sei sicuro di voler cambiare lo stato di <code>{executionCount}</code> esecuzione/i?",
"bulk delete": "Sei sicuro di voler eliminare <code>{executionCount}</code> esecuzione/i?",
"bulk delete backfills": "Elimina {count} backfill",
"bulk delete triggers": "Sei sicuro di voler eliminare {count} trigger?",
"bulk disabled status": {
"false": "Abilita {count} triggers",
"true": "Disabilita {count} triggers"
@@ -112,7 +111,6 @@
"bulk resume": "Sei sicuro di voler riprendere <code>{executionCount}</code> esecuzione/i?",
"bulk set labels": "Sei sicuro di voler impostare etichette a <code>{executionCount}</code> esecuzione/i?",
"bulk success delete backfills": "{count} backfills eliminati",
"bulk success delete triggers": "{count} trigger sono stati eliminati con successo",
"bulk success disabled status": {
"false": "{count} trigger(s) abilitati",
"true": "{count} trigger(s) disabilitati"
@@ -295,11 +293,6 @@
"delete logs": "Elimina logs",
"delete ok": "è eliminato",
"delete task confirm": "Vuoi eliminare il task <code>{taskId}</code>?",
"delete trigger": "Elimina trigger",
"delete trigger confirmation": "Sei sicuro di voler eliminare il trigger {id}? WARNING: eliminare un trigger potrebbe portare a esecuzioni duplicate se il trigger è ancora attivo in un flow.",
"delete trigger error": "Errore durante l'eliminazione del trigger {id}",
"delete trigger success": "Il trigger {id} è stato eliminato con successo",
"delete triggers": "Elimina trigger",
"delete_all_logs": "Sei sicuro di voler eliminare tutti i logs?",
"delete_log": "Sei sicuro di voler eliminare il log?",
"deleted": "Eliminato con successo",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "Utilizzare le variabili trigger non funzionerà con l'esecuzione manuale del flow. Aggiungi l'operatore di coalescenza `??` per risolvere il problema. Ad esempio, invece di `trigger.date`, usa `trigger.date ?? execution.startDate`.",
"trigger_id_exists": "L'ID del trigger esiste già",
"trigger_id_message": "L'ID del trigger ${existingTrigger} esiste già nel flow.",
"trigger_states": "Stati",
"triggered": "Execution trigger",
"triggered done": "L'esecuzione <em>{name}</em> è stata attivata con successo",
"triggerflow disabled": "Il trigger è disabilitato dalla definizione del flow",

View File

@@ -98,7 +98,6 @@
"bulk change state": "以下の実行の状態を変更してもよろしいですか?<code>{executionCount}</code> 実行。",
"bulk delete": "<code>{executionCount}</code>件の実行を削除してもよろしいですか?",
"bulk delete backfills": "{count}個のbackfillを削除",
"bulk delete triggers": "{count} 個のtriggerを削除してもよろしいですか",
"bulk disabled status": {
"false": "{count}個のtriggersを有効化",
"true": "{count}個のtriggersを無効化"
@@ -112,7 +111,6 @@
"bulk resume": "<code>{executionCount}</code>件の実行を再開してもよろしいですか?",
"bulk set labels": "<code>{executionCount}</code>件の実行にラベルを設定してもよろしいですか?",
"bulk success delete backfills": "{count}個のbackfillsが削除されました",
"bulk success delete triggers": "{count} 個のtriggerが正常に削除されました",
"bulk success disabled status": {
"false": "{count}個のtrigger(s)が有効化されました",
"true": "{count}個のtrigger(s)が無効化されました"
@@ -295,11 +293,6 @@
"delete logs": "Logsを削除",
"delete ok": "削除されました",
"delete task confirm": "task<code>{taskId}</code>を削除してもよろしいですか?",
"delete trigger": "トリガーを削除",
"delete trigger confirmation": "トリガー {id} を削除してもよろしいですかWARNING: トリガーがまだflowでアクティブな場合、削除すると重複した実行が発生する可能性があります。",
"delete trigger error": "トリガー {id} の削除エラー",
"delete trigger success": "トリガー{id}が正常に削除されました",
"delete triggers": "トリガーを削除",
"delete_all_logs": "すべてのLogsを削除してもよろしいですか",
"delete_log": "ログを削除してもよろしいですか?",
"deleted": "正常に削除されました",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "トリガー変数を使用すると、手動でのflow実行では機能しません。問題を解決するために`??`コアレス演算子を追加してください。例えば、`trigger.date`の代わりに`trigger.date ?? execution.startDate`を使用してください。",
"trigger_id_exists": "トリガーIDはすでに存在します。",
"trigger_id_message": "フロー内に既存のTrigger Id ${existingTrigger}が既に存在します。",
"trigger_states": "状態",
"triggered": "実行trigger",
"triggered done": "実行<em>{name}</em>が正常にトリガーされました",
"triggerflow disabled": "Flow定義からTriggerが無効になっています",

View File

@@ -98,7 +98,6 @@
"bulk change state": "<code>{executionCount}</code> 실행의 상태를 변경하시겠습니까?",
"bulk delete": "<code>{executionCount}</code> 개의 실행을 삭제하시겠습니까?",
"bulk delete backfills": "{count} backfill 삭제",
"bulk delete triggers": "{count}개의 trigger를 삭제하시겠습니까?",
"bulk disabled status": {
"false": "{count} triggers 활성화",
"true": "{count} triggers 비활성화"
@@ -112,7 +111,6 @@
"bulk resume": "<code>{executionCount}</code> 개의 실행을 재개하시겠습니까?",
"bulk set labels": "<code>{executionCount}</code> 개의 실행에 라벨을 설정하시겠습니까?",
"bulk success delete backfills": "{count} backfills 삭제됨",
"bulk success delete triggers": "{count} triggers가 성공적으로 삭제되었습니다",
"bulk success disabled status": {
"false": "{count} trigger(s) 활성화됨",
"true": "{count} trigger(s) 비활성화됨"
@@ -295,11 +293,6 @@
"delete logs": "Logs 삭제",
"delete ok": "삭제됨",
"delete task confirm": "task <code>{taskId}</code>을(를) 삭제하시겠습니까?",
"delete trigger": "트리거 삭제",
"delete trigger confirmation": "트리거 {id}를 삭제하시겠습니까? WARNING: 트리거가 flow에서 여전히 활성 상태인 경우 삭제하면 중복 실행이 발생할 수 있습니다.",
"delete trigger error": "트리거 {id} 삭제 오류",
"delete trigger success": "{id} 트리거가 성공적으로 삭제되었습니다",
"delete triggers": "트리거 삭제",
"delete_all_logs": "모든 logs를 삭제하시겠습니까?",
"delete_log": "로그를 삭제하시겠습니까?",
"deleted": "성공적으로 삭제됨",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "트리거 변수를 사용하는 것은 수동 flow 실행과 함께 작동하지 않습니다. 문제를 해결하려면 `??` 병합 연산자를 추가하세요. 예를 들어, `trigger.date` 대신 `trigger.date ?? execution.startDate`를 사용하세요.",
"trigger_id_exists": "트리거 ID가 이미 존재합니다.",
"trigger_id_message": "Flow에 Trigger Id ${existingTrigger}가 이미 존재합니다.",
"trigger_states": "상태",
"triggered": "Trigger 실행",
"triggered done": "<em>{name}</em>이(가) 성공적으로 Trigger 되었습니다.",
"triggerflow disabled": "Flow 정의에서 trigger가 비활성화되었습니다.",

View File

@@ -98,7 +98,6 @@
"bulk change state": "Czy na pewno chcesz zmienić stan <code>{executionCount}</code> wykonania/wykonań?",
"bulk delete": "Czy na pewno chcesz usunąć <code>{executionCount}</code> wykonanie(a)?",
"bulk delete backfills": "Usuń {count} backfill",
"bulk delete triggers": "Czy na pewno chcesz usunąć {count} triggerów?",
"bulk disabled status": {
"false": "Włącz {count} triggers",
"true": "Wyłącz {count} triggers"
@@ -112,7 +111,6 @@
"bulk resume": "Czy na pewno chcesz wznowić <code>{executionCount}</code> wykonanie(a)?",
"bulk set labels": "Czy na pewno chcesz ustawić etykiety dla <code>{executionCount}</code> wykonanie(a)?",
"bulk success delete backfills": "{count} backfills usunięte",
"bulk success delete triggers": "Pomyślnie usunięto {count} triggerów",
"bulk success disabled status": {
"false": "{count} trigger(s) włączone",
"true": "{count} trigger(s) wyłączone"
@@ -295,11 +293,6 @@
"delete logs": "Usuń logs",
"delete ok": "jest usunięty",
"delete task confirm": "Czy chcesz usunąć task <code>{taskId}</code>?",
"delete trigger": "Usuń trigger",
"delete trigger confirmation": "Czy na pewno chcesz usunąć trigger {id}? WARNING: usunięcie triggera może prowadzić do zduplikowanych wykonania, jeśli trigger jest nadal aktywny w flow.",
"delete trigger error": "Błąd podczas usuwania triggera {id}",
"delete trigger success": "Trigger {id} został pomyślnie usunięty",
"delete triggers": "Usuń triggery",
"delete_all_logs": "Czy na pewno chcesz usunąć wszystkie logs?",
"delete_log": "Czy na pewno chcesz usunąć log?",
"deleted": "Pomyślnie usunięto",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "Używanie zmiennych trigger nie zadziała przy ręcznym wykonaniu flow. Dodaj operator koalescencji `??`, aby rozwiązać problem. Na przykład, zamiast `trigger.date`, użyj `trigger.date ?? execution.startDate`.",
"trigger_id_exists": "Id triggera już istnieje",
"trigger_id_message": "Identyfikator Trigger ${existingTrigger} już istnieje w flow.",
"trigger_states": "Stany",
"triggered": "Execution trigger",
"triggered done": "Wykonanie <em>{name}</em> zostało pomyślnie uruchomione",
"triggerflow disabled": "Trigger jest wyłączony w definicji flow",

View File

@@ -98,7 +98,6 @@
"bulk change state": "Tem certeza de que deseja alterar o estado de <code>{executionCount}</code> execução(ões)?",
"bulk delete": "Tem certeza de que deseja deletar <code>{executionCount}</code> execução(ões)?",
"bulk delete backfills": "Excluir {count} backfill",
"bulk delete triggers": "Tem certeza de que deseja excluir {count} triggers?",
"bulk disabled status": {
"false": "Ativar {count} triggers",
"true": "Desativar {count} triggers"
@@ -112,7 +111,6 @@
"bulk resume": "Tem certeza de que deseja retomar <code>{executionCount}</code> execução(ões)?",
"bulk set labels": "Tem certeza de que deseja definir labels para <code>{executionCount}</code> execução(ões)?",
"bulk success delete backfills": "{count} backfills excluídos",
"bulk success delete triggers": "{count} triggers foram excluídos com sucesso",
"bulk success disabled status": {
"false": "{count} trigger(s) ativado(s)",
"true": "{count} trigger(s) desativado(s)"
@@ -295,11 +293,6 @@
"delete logs": "Deletar logs",
"delete ok": "está deletado",
"delete task confirm": "Deseja deletar a task <code>{taskId}</code>?",
"delete trigger": "Excluir trigger",
"delete trigger confirmation": "Tem certeza de que deseja excluir o trigger {id}? WARNING: excluir um trigger pode levar a execuções duplicadas se o trigger ainda estiver ativo em um flow.",
"delete trigger error": "Erro ao excluir trigger {id}",
"delete trigger success": "O Trigger {id} foi excluído com sucesso",
"delete triggers": "Excluir triggers",
"delete_all_logs": "Tem certeza de que deseja deletar todos os logs?",
"delete_log": "Tem certeza de que deseja excluir o log?",
"deleted": "Deletado com sucesso",
@@ -1752,7 +1745,6 @@
"trigger_check_warning": "Usar variáveis de trigger não funcionará com a execução manual de flow. Adicione o operador de coalescência `??` para resolver o problema. Por exemplo, em vez de `trigger.date`, use `trigger.date ?? execution.startDate`.",
"trigger_id_exists": "Id de Trigger já existe",
"trigger_id_message": "O Trigger Id ${existingTrigger} já existe no flow.",
"trigger_states": "Estados",
"triggered": "Trigger de execução",
"triggered done": "Execução <em>{name}</em> foi acionada com sucesso",
"triggerflow disabled": "Trigger está desativado na definição do flow",

Some files were not shown because too many files have changed in this diff Show More