2022-12-20 01:52:01 -08:00
|
|
|
<script lang="ts" setup>
|
|
|
|
import { computed, onMounted, ref } from 'vue';
|
2023-07-28 00:51:07 -07:00
|
|
|
import { useRoute, useRouter } from 'vue-router';
|
2023-05-05 01:41:54 -07:00
|
|
|
import type { UsageTelemetry } from '@/stores/usage.store';
|
|
|
|
import { useUsageStore } from '@/stores/usage.store';
|
2022-12-20 01:52:01 -08:00
|
|
|
import { telemetry } from '@/plugins/telemetry';
|
|
|
|
import { i18n as locale } from '@/plugins/i18n';
|
2023-11-28 03:15:08 -08:00
|
|
|
import { useUIStore } from '@/stores/ui.store';
|
2022-12-29 00:42:38 -08:00
|
|
|
import { N8N_PRICING_PAGE_URL } from '@/constants';
|
2023-11-28 03:15:08 -08:00
|
|
|
import { useToast } from '@/composables/useToast';
|
|
|
|
import { ROLE } from '@/utils/userUtils';
|
2023-11-23 03:22:47 -08:00
|
|
|
import { hasPermission } from '@/rbac/permissions';
|
2022-12-20 01:52:01 -08:00
|
|
|
|
|
|
|
const usageStore = useUsageStore();
|
|
|
|
const route = useRoute();
|
|
|
|
const router = useRouter();
|
2023-06-05 10:39:04 -07:00
|
|
|
const uiStore = useUIStore();
|
2023-07-28 00:51:07 -07:00
|
|
|
const toast = useToast();
|
2022-12-20 01:52:01 -08:00
|
|
|
|
|
|
|
const queryParamCallback = ref<string>(
|
|
|
|
`callback=${encodeURIComponent(`${window.location.origin}${window.location.pathname}`)}`,
|
|
|
|
);
|
2023-01-09 04:57:51 -08:00
|
|
|
const viewPlansUrl = computed(
|
|
|
|
() => `${usageStore.viewPlansUrl}&${queryParamCallback.value}&source=usage_page`,
|
|
|
|
);
|
2022-12-20 01:52:01 -08:00
|
|
|
const managePlanUrl = computed(() => `${usageStore.managePlanUrl}&${queryParamCallback.value}`);
|
|
|
|
const activationKeyModal = ref(false);
|
|
|
|
const activationKey = ref('');
|
|
|
|
const activationKeyInput = ref<HTMLInputElement | null>(null);
|
|
|
|
|
2023-11-23 03:22:47 -08:00
|
|
|
const canUserActivateLicense = computed(() =>
|
|
|
|
hasPermission(['role'], {
|
|
|
|
role: [ROLE.Owner],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
2022-12-20 01:52:01 -08:00
|
|
|
const showActivationSuccess = () => {
|
2023-07-28 00:51:07 -07:00
|
|
|
toast.showMessage({
|
|
|
|
type: 'success',
|
2022-12-20 01:52:01 -08:00
|
|
|
title: locale.baseText('settings.usageAndPlan.license.activation.success.title'),
|
|
|
|
message: locale.baseText('settings.usageAndPlan.license.activation.success.message', {
|
|
|
|
interpolate: {
|
|
|
|
name: usageStore.planName,
|
|
|
|
type: usageStore.planId
|
|
|
|
? locale.baseText('settings.usageAndPlan.plan')
|
|
|
|
: locale.baseText('settings.usageAndPlan.edition'),
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const showActivationError = (error: Error) => {
|
2023-07-28 00:51:07 -07:00
|
|
|
toast.showError(
|
|
|
|
error,
|
|
|
|
locale.baseText('settings.usageAndPlan.license.activation.error.title'),
|
|
|
|
error.message,
|
|
|
|
);
|
2022-12-20 01:52:01 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
const onLicenseActivation = async () => {
|
|
|
|
try {
|
|
|
|
await usageStore.activateLicense(activationKey.value);
|
|
|
|
activationKeyModal.value = false;
|
|
|
|
showActivationSuccess();
|
|
|
|
} catch (error) {
|
|
|
|
showActivationError(error);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
onMounted(async () => {
|
2022-12-29 00:42:38 -08:00
|
|
|
if (usageStore.isDesktop) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
usageStore.setLoading(true);
|
|
|
|
if (route.query.key) {
|
2022-12-20 01:52:01 -08:00
|
|
|
try {
|
2022-12-29 00:42:38 -08:00
|
|
|
await usageStore.activateLicense(route.query.key as string);
|
|
|
|
await router.replace({ query: {} });
|
|
|
|
showActivationSuccess();
|
2022-12-20 01:52:01 -08:00
|
|
|
usageStore.setLoading(false);
|
2022-12-29 00:42:38 -08:00
|
|
|
return;
|
2022-12-20 01:52:01 -08:00
|
|
|
} catch (error) {
|
2022-12-29 00:42:38 -08:00
|
|
|
showActivationError(error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
try {
|
2023-11-23 03:22:47 -08:00
|
|
|
if (!route.query.key && canUserActivateLicense.value) {
|
2022-12-29 00:42:38 -08:00
|
|
|
await usageStore.refreshLicenseManagementToken();
|
|
|
|
} else {
|
|
|
|
await usageStore.getLicenseInfo();
|
|
|
|
}
|
|
|
|
usageStore.setLoading(false);
|
|
|
|
} catch (error) {
|
|
|
|
if (!error.name) {
|
|
|
|
error.name = locale.baseText('settings.usageAndPlan.error');
|
2022-12-20 01:52:01 -08:00
|
|
|
}
|
2023-07-28 00:51:07 -07:00
|
|
|
toast.showError(error, error.name, error.message);
|
2022-12-20 01:52:01 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const sendUsageTelemetry = (action: UsageTelemetry['action']) => {
|
|
|
|
const telemetryPayload = usageStore.telemetryPayload;
|
|
|
|
telemetryPayload.action = action;
|
|
|
|
telemetry.track('User clicked button on usage page', telemetryPayload);
|
|
|
|
};
|
|
|
|
|
|
|
|
const onAddActivationKey = () => {
|
|
|
|
activationKeyModal.value = true;
|
|
|
|
sendUsageTelemetry('add_activation_key');
|
|
|
|
};
|
|
|
|
|
|
|
|
const onViewPlans = () => {
|
2023-10-06 04:16:27 -07:00
|
|
|
void uiStore.goToUpgrade('usage_page', 'open');
|
2022-12-20 01:52:01 -08:00
|
|
|
sendUsageTelemetry('view_plans');
|
|
|
|
};
|
|
|
|
|
|
|
|
const onManagePlan = () => {
|
|
|
|
sendUsageTelemetry('manage_plan');
|
|
|
|
};
|
|
|
|
|
|
|
|
const onDialogClosed = () => {
|
|
|
|
activationKey.value = '';
|
|
|
|
};
|
|
|
|
|
|
|
|
const onDialogOpened = () => {
|
|
|
|
activationKeyInput.value?.focus();
|
|
|
|
};
|
2022-12-28 08:07:34 -08:00
|
|
|
|
|
|
|
const openPricingPage = () => {
|
|
|
|
sendUsageTelemetry('desktop_view_plans');
|
2022-12-29 00:42:38 -08:00
|
|
|
window.open(N8N_PRICING_PAGE_URL, '_blank');
|
2022-12-28 08:07:34 -08:00
|
|
|
};
|
2022-12-20 01:52:01 -08:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
2023-07-28 00:51:07 -07:00
|
|
|
<div class="settings-usage-and-plan">
|
2022-12-20 01:52:01 -08:00
|
|
|
<n8n-heading size="2xlarge">{{ locale.baseText('settings.usageAndPlan.title') }}</n8n-heading>
|
2022-12-28 08:07:34 -08:00
|
|
|
<n8n-action-box
|
|
|
|
v-if="usageStore.isDesktop"
|
|
|
|
:class="$style.actionBox"
|
|
|
|
:heading="locale.baseText('settings.usageAndPlan.desktop.title')"
|
|
|
|
:description="locale.baseText('settings.usageAndPlan.desktop.description')"
|
|
|
|
:buttonText="locale.baseText('settings.usageAndPlan.button.plans')"
|
2023-07-28 00:51:07 -07:00
|
|
|
@click:button="openPricingPage"
|
2022-12-28 08:07:34 -08:00
|
|
|
/>
|
|
|
|
<div v-if="!usageStore.isDesktop && !usageStore.isLoading">
|
|
|
|
<n8n-heading :class="$style.title" size="large">
|
2023-07-28 00:51:07 -07:00
|
|
|
<i18n-t keypath="settings.usageAndPlan.description" tag="span">
|
2022-12-28 08:07:34 -08:00
|
|
|
<template #name>{{ usageStore.planName }}</template>
|
|
|
|
<template #type>
|
|
|
|
<span v-if="usageStore.planId">{{
|
|
|
|
locale.baseText('settings.usageAndPlan.plan')
|
2022-12-20 01:52:01 -08:00
|
|
|
}}</span>
|
2022-12-28 08:07:34 -08:00
|
|
|
<span v-else>{{ locale.baseText('settings.usageAndPlan.edition') }}</span>
|
2022-12-20 01:52:01 -08:00
|
|
|
</template>
|
2023-07-28 00:51:07 -07:00
|
|
|
</i18n-t>
|
2022-12-28 08:07:34 -08:00
|
|
|
</n8n-heading>
|
|
|
|
|
|
|
|
<div :class="$style.quota">
|
|
|
|
<n8n-text size="medium" color="text-light">
|
|
|
|
{{ locale.baseText('settings.usageAndPlan.activeWorkflows') }}
|
|
|
|
</n8n-text>
|
|
|
|
<div :class="$style.chart">
|
|
|
|
<span v-if="usageStore.executionLimit > 0" :class="$style.chartLine">
|
|
|
|
<span
|
|
|
|
:class="$style.chartBar"
|
|
|
|
:style="{ width: `${usageStore.executionPercentage}%` }"
|
|
|
|
></span>
|
|
|
|
</span>
|
2023-07-28 00:51:07 -07:00
|
|
|
<i18n-t
|
|
|
|
tag="span"
|
|
|
|
:class="$style.count"
|
|
|
|
keypath="settings.usageAndPlan.activeWorkflows.count"
|
|
|
|
>
|
2022-12-28 08:07:34 -08:00
|
|
|
<template #count>{{ usageStore.executionCount }}</template>
|
|
|
|
<template #limit>
|
|
|
|
<span v-if="usageStore.executionLimit < 0">{{
|
|
|
|
locale.baseText('settings.usageAndPlan.activeWorkflows.unlimited')
|
|
|
|
}}</span>
|
|
|
|
<span v-else>{{ usageStore.executionLimit }}</span>
|
|
|
|
</template>
|
2023-07-28 00:51:07 -07:00
|
|
|
</i18n-t>
|
2022-12-28 08:07:34 -08:00
|
|
|
</div>
|
2022-12-20 01:52:01 -08:00
|
|
|
</div>
|
|
|
|
|
2022-12-28 08:07:34 -08:00
|
|
|
<n8n-info-tip>{{
|
|
|
|
locale.baseText('settings.usageAndPlan.activeWorkflows.hint')
|
|
|
|
}}</n8n-info-tip>
|
2022-12-20 01:52:01 -08:00
|
|
|
|
2022-12-28 08:07:34 -08:00
|
|
|
<div :class="$style.buttons">
|
|
|
|
<n8n-button
|
|
|
|
:class="$style.buttonTertiary"
|
|
|
|
@click="onAddActivationKey"
|
2023-11-23 03:22:47 -08:00
|
|
|
v-if="canUserActivateLicense"
|
2022-12-28 08:07:34 -08:00
|
|
|
type="tertiary"
|
|
|
|
size="large"
|
|
|
|
>
|
2023-11-01 05:33:36 -07:00
|
|
|
<span>{{ locale.baseText('settings.usageAndPlan.button.activation') }}</span>
|
2022-12-28 08:07:34 -08:00
|
|
|
</n8n-button>
|
|
|
|
<n8n-button v-if="usageStore.managementToken" @click="onManagePlan" size="large">
|
|
|
|
<a :href="managePlanUrl" target="_blank">{{
|
|
|
|
locale.baseText('settings.usageAndPlan.button.manage')
|
|
|
|
}}</a>
|
2022-12-20 01:52:01 -08:00
|
|
|
</n8n-button>
|
fix(editor): Open only one tab with plans page (#7377)
## Issue
In community edition, clicking on "View plans" button on "Settings" ->
"Usage and plan" page (e.g. http://127.0.0.1:5678/settings/usage) opens
two new tabs with n8n pricing (one of them with UTM tracking, another
without).
This was introduced in #6317 , when click handler of "View plans" link
container [started
calling](https://github.com/n8n-io/n8n/pull/6317/files#diff-0bf26afac8a06e03b3d39d0668f22408859355b585a9ab420800c125e33f0691R109)
`uiStore.goToUpgrade(...)` which opens n8n pricing in a new tab, while
browser opens another tab for the link URL.
The simplest fix, implemented in this PR, is to prevent default event
handling (so that, after `onViewPlans` is called, browser will not
attempt to process the click additionally as clicking on the link),
similarly to how it is prevented on some other pages. It only solves the
immediate problem of browser opening two new tabs on clicking "View
plans".
Note that **I didn't implement any tests for the changed behavior**,
because it was not covered by tests before, and I couldn't quite figure
out how to cover it now within the existing test approach (considering
that testing the fact that only one new tab is open will likely require
to write entirely new tests relying on puppeteer; as far as I can see,
no existing `editor-ui` tests are doing anything like that). I'll gladly
implement tests for the new behavior if you tell me how you would like
them to look.
The existing tests for `editor-ui` still pass; I didn't run tests for
other subpackages (see "additional contribution notes" below).
## Additional notes on the issue.
I'm not sure that the change in this PR is the correct long-term
solution for the issue, because the URLs for these two methods (custom
click handler for link container and default link handling) are slightly
different:
* Custom click handler calls `useTelemetryStore().track('User clicked
upgrade CTA', ...)`; then calls `sendUsageTelemetry('view_plans')` (it
feels weird that two calls to telemetry are made); then opens new tab
for `https://n8n.io/pricing?utm_campaign=open&source=usage_page` (note
that prior to #7316 the second call to telemetry was done after the new
tab is opened, not before);
* Link itself refers to another page, with slightly different tracking
parameters:
`https://subscription.n8n.io/?instanceid=[REDACTED]&version=1.10.0&callback=http%3A%2F%2F127.0.0.1%3A5678%2Fsettings%2Fusage&source=usage_page`;
but this page redirects to `https://n8n.io/pricing/`.
It is not clear which one of the two is the right way of doing things.
Although `goToUpgrade` is called in 20 places throughout `editor-ui`,
while `viewPlansUrl`, as far as I can see, is used for this button only.
Additionally, since Settings pages don't work without JS anyway, I can
only think of two separate scenarios where any tab would be opened:
* Left-clicking the link (or Ctrl-clicking, or pressing Space or Enter
when the link is focused, or tapping): previously, both custom click
handler was executed and link's `href` was opened; in this PR, only
custom click handler is executed (similarly to how it is done in the
other places where `goToUpgrade` is called);
* Right-clicking (or long tapping, or opening context menu in any other
way) and selecting "open link in new tab" (or similar): opens a new tab
for URL from the `href` attribute (and does not send any telemetry at
all).
I'd say that the better permanent solution would probably be to get rid
of one of these methods entirely, and only rely on another in all cases
(for me, as an outside contributor, the preferred way would be for
custom click handler to only send telemetry, while letting my browser
handle the actual navigation). However, that would be a large change,
much more than one line in this PR.
Additionally, other similar places where `goToUpgrade` is currently
called (directly or indirectly) would also need to be adapted for this
change.
## Additional contribution notes
As a first-time contributor, I've encountered several things I didn't
expect; I'm not sure if they should be expected or are issues:
1. Tests for the entire monorepo consume a lot of RAM; 20GB free RAM was
not enough, so I couldn't run tests for the entire monorepo and had to
only run them for `packages/editor-ui`;
2. Linting is very slow; `pnpm lint` in `packages/editor-ui` takes ten
minutes to complete;
3. It seems that types are not checked. Code OSS highlights numerous
errors in code files: for example, `'debug'` is incompatible with
`CloudUpdateLinkSourceType` expected by `goToUpgrade` here:
https://github.com/n8n-io/n8n/blob/3e7a4d3b2cc12fcb1b011fccd0773bb807986884/packages/editor-ui/src/composables/useExecutionDebugging.ts#L128
However, I'm not getting any errors during build. There is a `typecheck`
script defined in `package.json`, but `pnpm typecheck` fails with:
```
n8n-toy-demo:~/projects/n8n/packages/editor-ui$ pnpm typecheck
> n8n-editor-ui@1.10.0 typecheck
/home/inga/projects/n8n/packages/editor-ui
> vue-tsc --emitDeclarationOnly
error TS5069: Option 'emitDeclarationOnly' cannot be specified without
specifying
option 'declaration' or option 'composite'.
Found 1 error.
ELIFECYCLE Command failed with exit code 1.
n8n-toy-demo:~/projects/n8n/packages/editor-ui$
```
Replacing `--emitDeclarationsOnly` with `--noEmit` in `package.json`
unblocks typechecking and results in seemingly, at first glance, correct
"Found 1924 errors in 306 files" (at least several of the reported
errors that I've checked seem to be correct).
But maybe I'm missing something and there are not in fact two thousands
type errors in `editor-ui`?
2023-10-13 05:14:26 -07:00
|
|
|
<n8n-button v-else @click.prevent="onViewPlans" size="large">
|
2022-12-28 08:07:34 -08:00
|
|
|
<a :href="viewPlansUrl" target="_blank">{{
|
|
|
|
locale.baseText('settings.usageAndPlan.button.plans')
|
|
|
|
}}</a>
|
2022-12-20 01:52:01 -08:00
|
|
|
</n8n-button>
|
2022-12-28 08:07:34 -08:00
|
|
|
</div>
|
|
|
|
|
|
|
|
<el-dialog
|
|
|
|
width="480px"
|
|
|
|
top="0"
|
|
|
|
@closed="onDialogClosed"
|
|
|
|
@opened="onDialogOpened"
|
2023-07-28 00:51:07 -07:00
|
|
|
v-model="activationKeyModal"
|
2022-12-28 08:07:34 -08:00
|
|
|
:title="locale.baseText('settings.usageAndPlan.dialog.activation.title')"
|
2023-08-01 04:52:33 -07:00
|
|
|
:modal-class="$style.center"
|
2022-12-28 08:07:34 -08:00
|
|
|
>
|
|
|
|
<template #default>
|
|
|
|
<n8n-input
|
|
|
|
ref="activationKeyInput"
|
|
|
|
v-model="activationKey"
|
|
|
|
:placeholder="locale.baseText('settings.usageAndPlan.dialog.activation.label')"
|
|
|
|
/>
|
|
|
|
</template>
|
|
|
|
<template #footer>
|
2023-07-28 00:51:07 -07:00
|
|
|
<n8n-button @click="activationKeyModal = false" type="secondary">
|
2022-12-28 08:07:34 -08:00
|
|
|
{{ locale.baseText('settings.usageAndPlan.dialog.activation.cancel') }}
|
|
|
|
</n8n-button>
|
2023-07-28 00:51:07 -07:00
|
|
|
<n8n-button @click="onLicenseActivation">
|
2022-12-28 08:07:34 -08:00
|
|
|
{{ locale.baseText('settings.usageAndPlan.dialog.activation.activate') }}
|
|
|
|
</n8n-button>
|
|
|
|
</template>
|
|
|
|
</el-dialog>
|
|
|
|
</div>
|
2022-12-20 01:52:01 -08:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<style lang="scss" module>
|
|
|
|
@import '@/styles/css-animation-helpers.scss';
|
|
|
|
|
2023-08-01 04:52:33 -07:00
|
|
|
.center > div {
|
|
|
|
justify-content: center;
|
|
|
|
}
|
|
|
|
|
2022-12-28 08:07:34 -08:00
|
|
|
.actionBox {
|
|
|
|
margin: var(--spacing-2xl) 0 0;
|
|
|
|
}
|
|
|
|
|
2022-12-20 01:52:01 -08:00
|
|
|
.spacedFlex {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
.title {
|
|
|
|
display: block;
|
|
|
|
padding: var(--spacing-2xl) 0 var(--spacing-m);
|
|
|
|
}
|
|
|
|
|
|
|
|
.quota {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
align-items: center;
|
|
|
|
height: 54px;
|
|
|
|
padding: 0 var(--spacing-s);
|
|
|
|
margin: 0 0 var(--spacing-xs);
|
|
|
|
background: var(--color-background-xlight);
|
|
|
|
border-radius: var(--border-radius-large);
|
2023-11-14 08:13:30 -08:00
|
|
|
border: 1px solid var(--color-foreground-base);
|
2022-12-20 01:52:01 -08:00
|
|
|
white-space: nowrap;
|
|
|
|
|
|
|
|
.count {
|
|
|
|
text-transform: lowercase;
|
|
|
|
font-size: var(--font-size-s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.buttons {
|
|
|
|
display: flex;
|
|
|
|
justify-content: flex-end;
|
|
|
|
padding: var(--spacing-xl) 0 0;
|
|
|
|
|
|
|
|
button {
|
|
|
|
margin-left: var(--spacing-xs);
|
|
|
|
|
|
|
|
a {
|
|
|
|
display: inline-block;
|
|
|
|
color: inherit;
|
|
|
|
text-decoration: none;
|
|
|
|
padding: var(--spacing-xs) var(--spacing-m);
|
|
|
|
margin: calc(var(--spacing-xs) * -1) calc(var(--spacing-m) * -1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.chart {
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
justify-content: flex-end;
|
|
|
|
flex-grow: 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
.chartLine {
|
|
|
|
display: block;
|
|
|
|
height: 10px;
|
|
|
|
width: 100%;
|
|
|
|
max-width: 260px;
|
|
|
|
margin: 0 var(--spacing-m);
|
|
|
|
border-radius: 10px;
|
|
|
|
background: var(--color-background-base);
|
|
|
|
}
|
|
|
|
|
|
|
|
.chartBar {
|
|
|
|
float: left;
|
|
|
|
height: 100%;
|
|
|
|
max-width: 100%;
|
|
|
|
background: var(--color-secondary);
|
|
|
|
border-radius: 10px;
|
|
|
|
transition: width 0.2s $ease-out-expo;
|
|
|
|
}
|
|
|
|
|
|
|
|
div[class*='info'] > span > span:last-child {
|
|
|
|
line-height: 1.4;
|
|
|
|
padding: 0 0 0 var(--spacing-4xs);
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
2023-07-28 00:51:07 -07:00
|
|
|
.settings-usage-and-plan {
|
|
|
|
:deep(.el-dialog__wrapper) {
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
justify-content: center;
|
|
|
|
|
|
|
|
.el-dialog {
|
|
|
|
margin: 0;
|
|
|
|
|
|
|
|
.el-dialog__footer {
|
|
|
|
button {
|
|
|
|
margin-left: var(--spacing-xs);
|
|
|
|
}
|
2022-12-20 01:52:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|