mirror of
https://github.com/umami-software/umami.git
synced 2026-02-18 19:45:35 +01:00
Merge branch 'umami-software:master' into master
This commit is contained in:
commit
fc10af35c3
166 changed files with 5605 additions and 2228 deletions
|
|
@ -16,7 +16,8 @@
|
||||||
"react/display-name": "off",
|
"react/display-name": "off",
|
||||||
"react/react-in-jsx-scope": "off",
|
"react/react-in-jsx-scope": "off",
|
||||||
"react/prop-types": "off",
|
"react/prop-types": "off",
|
||||||
"import/no-anonymous-default-export": "off"
|
"import/no-anonymous-default-export": "off",
|
||||||
|
"@next/next/no-img-element": "off"
|
||||||
},
|
},
|
||||||
"globals": {
|
"globals": {
|
||||||
"React": "writable"
|
"React": "writable"
|
||||||
|
|
|
||||||
2
.github/workflows/cd-manual.yml
vendored
2
.github/workflows/cd-manual.yml
vendored
|
|
@ -26,6 +26,6 @@ jobs:
|
||||||
image: umami
|
image: umami
|
||||||
tags: ${{ matrix.db-type }}-${{ inputs.version }}, ${{ matrix.db-type }}-latest
|
tags: ${{ matrix.db-type }}-${{ inputs.version }}, ${{ matrix.db-type }}-latest
|
||||||
buildArgs: DATABASE_TYPE=${{ matrix.db-type }}
|
buildArgs: DATABASE_TYPE=${{ matrix.db-type }}
|
||||||
registry: ghcr.io/${{ github.actor }}
|
registry: ghcr.io
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
|
||||||
2
.github/workflows/cd.yml
vendored
2
.github/workflows/cd.yml
vendored
|
|
@ -25,6 +25,6 @@ jobs:
|
||||||
image: umami
|
image: umami
|
||||||
tags: ${{ matrix.db-type }}-${{ env.RELEASE_VERSION }}, ${{ matrix.db-type }}-latest
|
tags: ${{ matrix.db-type }}-${{ env.RELEASE_VERSION }}, ${{ matrix.db-type }}-latest
|
||||||
buildArgs: DATABASE_TYPE=${{ matrix.db-type }}
|
buildArgs: DATABASE_TYPE=${{ matrix.db-type }}
|
||||||
registry: ghcr.io/${{ github.actor }}
|
registry: ghcr.io
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
|
||||||
4
.github/workflows/ci.yml
vendored
4
.github/workflows/ci.yml
vendored
|
|
@ -15,10 +15,6 @@ jobs:
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
- node-version: 12.x
|
|
||||||
db-type: postgresql
|
|
||||||
- node-version: 12.x
|
|
||||||
db-type: mysql
|
|
||||||
- node-version: 14.x
|
- node-version: 14.x
|
||||||
db-type: postgresql
|
db-type: postgresql
|
||||||
- node-version: 14.x
|
- node-version: 14.x
|
||||||
|
|
|
||||||
|
|
@ -22,7 +22,7 @@ npm install -g yarn
|
||||||
### Get the source code and install packages
|
### Get the source code and install packages
|
||||||
|
|
||||||
```
|
```
|
||||||
git clone https://github.com/mikecao/umami.git
|
git clone https://github.com/umami-software/umami.git
|
||||||
cd umami
|
cd umami
|
||||||
yarn install
|
yarn install
|
||||||
```
|
```
|
||||||
|
|
@ -32,7 +32,7 @@ yarn install
|
||||||
Create an `.env` file with the following
|
Create an `.env` file with the following
|
||||||
|
|
||||||
```
|
```
|
||||||
DATABASE_URL=(connection url)
|
DATABASE_URL=connection-url
|
||||||
```
|
```
|
||||||
|
|
||||||
The connection url is in the following format:
|
The connection url is in the following format:
|
||||||
|
|
@ -76,12 +76,12 @@ docker-compose up
|
||||||
|
|
||||||
Alternatively, to pull just the Umami Docker image with PostgreSQL support:
|
Alternatively, to pull just the Umami Docker image with PostgreSQL support:
|
||||||
```bash
|
```bash
|
||||||
docker pull ghcr.io/mikecao/umami:postgresql-latest
|
docker pull docker.umami.is/umami-software/umami:postgresql-latest
|
||||||
```
|
```
|
||||||
|
|
||||||
Or with MySQL support:
|
Or with MySQL support:
|
||||||
```bash
|
```bash
|
||||||
docker pull ghcr.io/mikecao/umami:mysql-latest
|
docker pull docker.umami.is/umami-software/umami:mysql-latest
|
||||||
```
|
```
|
||||||
|
|
||||||
## Getting updates
|
## Getting updates
|
||||||
|
|
|
||||||
2
app.json
2
app.json
|
|
@ -3,7 +3,7 @@
|
||||||
"description": "Umami is a simple, fast, website analytics alternative to Google Analytics.",
|
"description": "Umami is a simple, fast, website analytics alternative to Google Analytics.",
|
||||||
"keywords": ["analytics", "charts", "statistics", "web-analytics"],
|
"keywords": ["analytics", "charts", "statistics", "web-analytics"],
|
||||||
"website": "https://umami.is",
|
"website": "https://umami.is",
|
||||||
"repository": "https://github.com/mikecao/umami",
|
"repository": "https://github.com/umami-software/umami",
|
||||||
"addons": ["heroku-postgresql"],
|
"addons": ["heroku-postgresql"],
|
||||||
"env": {
|
"env": {
|
||||||
"HASH_SALT": {
|
"HASH_SALT": {
|
||||||
|
|
|
||||||
|
|
@ -3,12 +3,12 @@ import { FormattedMessage } from 'react-intl';
|
||||||
import ButtonLayout from 'components/layout/ButtonLayout';
|
import ButtonLayout from 'components/layout/ButtonLayout';
|
||||||
import useStore, { checkVersion } from 'store/version';
|
import useStore, { checkVersion } from 'store/version';
|
||||||
import { setItem } from 'lib/web';
|
import { setItem } from 'lib/web';
|
||||||
import { VERSION_CHECK, VERSION_URL } from 'lib/constants';
|
import { REPO_URL, VERSION_CHECK } from 'lib/constants';
|
||||||
import Button from './Button';
|
import Button from './Button';
|
||||||
import styles from './UpdateNotice.module.css';
|
import styles from './UpdateNotice.module.css';
|
||||||
|
|
||||||
export default function UpdateNotice() {
|
export default function UpdateNotice() {
|
||||||
const { latest, checked, hasUpdate } = useStore();
|
const { latest, checked, hasUpdate, releaseUrl } = useStore();
|
||||||
const [dismissed, setDismissed] = useState(false);
|
const [dismissed, setDismissed] = useState(false);
|
||||||
|
|
||||||
const updateCheck = useCallback(() => {
|
const updateCheck = useCallback(() => {
|
||||||
|
|
@ -18,7 +18,7 @@ export default function UpdateNotice() {
|
||||||
function handleViewClick() {
|
function handleViewClick() {
|
||||||
updateCheck();
|
updateCheck();
|
||||||
setDismissed(true);
|
setDismissed(true);
|
||||||
location.href = VERSION_URL;
|
location.href = releaseUrl || REPO_URL;
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleDismissClick() {
|
function handleDismissClick() {
|
||||||
|
|
|
||||||
|
|
@ -4,7 +4,7 @@ import { FormattedMessage } from 'react-intl';
|
||||||
import Link from 'components/common/Link';
|
import Link from 'components/common/Link';
|
||||||
import styles from './Footer.module.css';
|
import styles from './Footer.module.css';
|
||||||
import useStore from 'store/version';
|
import useStore from 'store/version';
|
||||||
import { HOMEPAGE_URL, VERSION_URL } from 'lib/constants';
|
import { HOMEPAGE_URL, REPO_URL } from 'lib/constants';
|
||||||
|
|
||||||
export default function Footer() {
|
export default function Footer() {
|
||||||
const { current } = useStore();
|
const { current } = useStore();
|
||||||
|
|
@ -26,8 +26,11 @@ export default function Footer() {
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className={classNames(styles.version, 'col-12 col-md-4')}>
|
<div className={classNames(styles.version, 'col-12 col-md-4')}>
|
||||||
<Link href={VERSION_URL}>{`v${current}`}</Link>
|
<Link href={REPO_URL}>{`v${current}`}</Link>
|
||||||
</div>
|
</div>
|
||||||
|
{!process.env.telemetryDisabled && (
|
||||||
|
<img src={`https://i.umami.is/a.png?v=${current}`} alt="" />
|
||||||
|
)}
|
||||||
</footer>
|
</footer>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -2,6 +2,7 @@ import React from 'react';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import Button from 'components/common/Button';
|
import Button from 'components/common/Button';
|
||||||
import Times from 'assets/times.svg';
|
import Times from 'assets/times.svg';
|
||||||
|
import { safeDecodeURI } from 'lib/url';
|
||||||
import styles from './FilterTags.module.css';
|
import styles from './FilterTags.module.css';
|
||||||
|
|
||||||
export default function FilterTags({ params, onClick }) {
|
export default function FilterTags({ params, onClick }) {
|
||||||
|
|
@ -17,7 +18,7 @@ export default function FilterTags({ params, onClick }) {
|
||||||
return (
|
return (
|
||||||
<div key={key} className={styles.tag}>
|
<div key={key} className={styles.tag}>
|
||||||
<Button icon={<Times />} onClick={() => onClick(key)} variant="action" iconRight>
|
<Button icon={<Times />} onClick={() => onClick(key)} variant="action" iconRight>
|
||||||
{`${key}: ${params[key]}`}
|
{`${key}: ${safeDecodeURI(params[key])}`}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
|
||||||
|
|
@ -22,6 +22,7 @@ export default function MetricsTable({
|
||||||
filterOptions,
|
filterOptions,
|
||||||
limit,
|
limit,
|
||||||
onDataLoad,
|
onDataLoad,
|
||||||
|
delay = null,
|
||||||
...props
|
...props
|
||||||
}) {
|
}) {
|
||||||
const [{ startDate, endDate, modified }] = useDateRange(websiteId);
|
const [{ startDate, endDate, modified }] = useDateRange(websiteId);
|
||||||
|
|
@ -46,9 +47,9 @@ export default function MetricsTable({
|
||||||
country,
|
country,
|
||||||
},
|
},
|
||||||
onDataLoad,
|
onDataLoad,
|
||||||
delay: DEFAULT_ANIMATION_DURATION,
|
delay: delay || DEFAULT_ANIMATION_DURATION,
|
||||||
},
|
},
|
||||||
[modified, url, referrer, os, browser, device, country],
|
[type, modified, url, referrer, os, browser, device, country],
|
||||||
);
|
);
|
||||||
|
|
||||||
const filteredData = useMemo(() => {
|
const filteredData = useMemo(() => {
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { useIntl, defineMessage } from 'react-intl';
|
||||||
import FilterLink from 'components/common/FilterLink';
|
import FilterLink from 'components/common/FilterLink';
|
||||||
import FilterButtons from 'components/common/FilterButtons';
|
import FilterButtons from 'components/common/FilterButtons';
|
||||||
import { urlFilter } from 'lib/filters';
|
import { urlFilter } from 'lib/filters';
|
||||||
|
|
@ -8,15 +8,26 @@ import MetricsTable from './MetricsTable';
|
||||||
export const FILTER_COMBINED = 0;
|
export const FILTER_COMBINED = 0;
|
||||||
export const FILTER_RAW = 1;
|
export const FILTER_RAW = 1;
|
||||||
|
|
||||||
export default function PagesTable({ websiteId, websiteDomain, showFilters, ...props }) {
|
const messages = defineMessage({
|
||||||
|
combined: { id: 'metrics.filter.combined', defaultMessage: 'Combined' },
|
||||||
|
raw: { id: 'metrics.filter.raw', defaultMessage: 'Raw' },
|
||||||
|
pages: { id: 'metrics.pages', defaultMessage: 'Pages' },
|
||||||
|
views: { id: 'metrics.views', defaultMessage: 'View' },
|
||||||
|
});
|
||||||
|
|
||||||
|
export default function PagesTable({ websiteId, showFilters, ...props }) {
|
||||||
const [filter, setFilter] = useState(FILTER_COMBINED);
|
const [filter, setFilter] = useState(FILTER_COMBINED);
|
||||||
|
const { formatMessage } = useIntl();
|
||||||
|
|
||||||
const buttons = [
|
const buttons = [
|
||||||
{
|
{
|
||||||
label: <FormattedMessage id="metrics.filter.combined" defaultMessage="Combined" />,
|
label: formatMessage(messages.combined),
|
||||||
value: FILTER_COMBINED,
|
value: FILTER_COMBINED,
|
||||||
},
|
},
|
||||||
{ label: <FormattedMessage id="metrics.filter.raw" defaultMessage="Raw" />, value: FILTER_RAW },
|
{
|
||||||
|
label: formatMessage(messages.raw),
|
||||||
|
value: FILTER_RAW,
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const renderLink = ({ x: url }) => {
|
const renderLink = ({ x: url }) => {
|
||||||
|
|
@ -27,12 +38,11 @@ export default function PagesTable({ websiteId, websiteDomain, showFilters, ...p
|
||||||
<>
|
<>
|
||||||
{showFilters && <FilterButtons buttons={buttons} selected={filter} onClick={setFilter} />}
|
{showFilters && <FilterButtons buttons={buttons} selected={filter} onClick={setFilter} />}
|
||||||
<MetricsTable
|
<MetricsTable
|
||||||
title={<FormattedMessage id="metrics.pages" defaultMessage="Pages" />}
|
title={formatMessage(messages.pages)}
|
||||||
type="url"
|
type="url"
|
||||||
metric={<FormattedMessage id="metrics.views" defaultMessage="Views" />}
|
metric={formatMessage(messages.views)}
|
||||||
websiteId={websiteId}
|
websiteId={websiteId}
|
||||||
dataFilter={urlFilter}
|
dataFilter={filter !== FILTER_RAW ? urlFilter : null}
|
||||||
filterOptions={{ domain: websiteDomain, raw: filter === FILTER_RAW }}
|
|
||||||
renderLabel={renderLink}
|
renderLabel={renderLink}
|
||||||
{...props}
|
{...props}
|
||||||
/>
|
/>
|
||||||
|
|
|
||||||
|
|
@ -1,31 +1,40 @@
|
||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { useIntl, defineMessages } from 'react-intl';
|
||||||
import MetricsTable from './MetricsTable';
|
import MetricsTable from './MetricsTable';
|
||||||
import FilterButtons from 'components/common/FilterButtons';
|
import FilterButtons from 'components/common/FilterButtons';
|
||||||
import FilterLink from 'components/common/FilterLink';
|
import FilterLink from 'components/common/FilterLink';
|
||||||
import { refFilter } from 'lib/filters';
|
import { refFilter } from 'lib/filters';
|
||||||
|
|
||||||
export const FILTER_DOMAIN_ONLY = 0;
|
export const FILTER_COMBINED = 0;
|
||||||
export const FILTER_COMBINED = 1;
|
export const FILTER_RAW = 1;
|
||||||
export const FILTER_RAW = 2;
|
|
||||||
|
|
||||||
export default function ReferrersTable({ websiteId, websiteDomain, showFilters, ...props }) {
|
const messages = defineMessages({
|
||||||
|
combined: { id: 'metrics.filter.combined', defaultMessage: 'Combined' },
|
||||||
|
raw: { id: 'metrics.filter.raw', defaultMessage: 'Raw' },
|
||||||
|
referrers: { id: 'metrics.referrers', defaultMessage: 'Referrers' },
|
||||||
|
views: { id: 'metrics.views', defaultMessage: 'Views' },
|
||||||
|
none: { id: 'label.none', defaultMessage: 'None' },
|
||||||
|
});
|
||||||
|
|
||||||
|
export default function ReferrersTable({ websiteId, showFilters, ...props }) {
|
||||||
const [filter, setFilter] = useState(FILTER_COMBINED);
|
const [filter, setFilter] = useState(FILTER_COMBINED);
|
||||||
|
const { formatMessage } = useIntl();
|
||||||
|
const none = formatMessage(messages.none);
|
||||||
|
|
||||||
const buttons = [
|
const buttons = [
|
||||||
{
|
{
|
||||||
label: <FormattedMessage id="metrics.filter.domain-only" defaultMessage="Domain only" />,
|
label: formatMessage(messages.combined),
|
||||||
value: FILTER_DOMAIN_ONLY,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: <FormattedMessage id="metrics.filter.combined" defaultMessage="Combined" />,
|
|
||||||
value: FILTER_COMBINED,
|
value: FILTER_COMBINED,
|
||||||
},
|
},
|
||||||
{ label: <FormattedMessage id="metrics.filter.raw" defaultMessage="Raw" />, value: FILTER_RAW },
|
{ label: formatMessage(messages.raw), value: FILTER_RAW },
|
||||||
];
|
];
|
||||||
|
|
||||||
const renderLink = ({ w: link, x: referrer }) => {
|
const renderLink = ({ w: link, x: referrer }) => {
|
||||||
return <FilterLink id="referrer" value={referrer} externalUrl={link} />;
|
return referrer ? (
|
||||||
|
<FilterLink id="referrer" value={referrer} externalUrl={link} />
|
||||||
|
) : (
|
||||||
|
`(${none})`
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
@ -33,16 +42,11 @@ export default function ReferrersTable({ websiteId, websiteDomain, showFilters,
|
||||||
{showFilters && <FilterButtons buttons={buttons} selected={filter} onClick={setFilter} />}
|
{showFilters && <FilterButtons buttons={buttons} selected={filter} onClick={setFilter} />}
|
||||||
<MetricsTable
|
<MetricsTable
|
||||||
{...props}
|
{...props}
|
||||||
title={<FormattedMessage id="metrics.referrers" defaultMessage="Referrers" />}
|
title={formatMessage(messages.referrers)}
|
||||||
type="referrer"
|
type="referrer"
|
||||||
metric={<FormattedMessage id="metrics.views" defaultMessage="Views" />}
|
metric={formatMessage(messages.views)}
|
||||||
websiteId={websiteId}
|
websiteId={websiteId}
|
||||||
dataFilter={refFilter}
|
dataFilter={filter !== FILTER_RAW ? refFilter : null}
|
||||||
filterOptions={{
|
|
||||||
domain: websiteDomain,
|
|
||||||
domainOnly: filter === FILTER_DOMAIN_ONLY,
|
|
||||||
raw: filter === FILTER_RAW,
|
|
||||||
}}
|
|
||||||
renderLabel={renderLink}
|
renderLabel={renderLink}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
|
|
|
||||||
15
components/metrics/ScreenTable.js
Normal file
15
components/metrics/ScreenTable.js
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
import React from 'react';
|
||||||
|
import MetricsTable from './MetricsTable';
|
||||||
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
|
||||||
|
export default function ScreenTable({ websiteId, ...props }) {
|
||||||
|
return (
|
||||||
|
<MetricsTable
|
||||||
|
{...props}
|
||||||
|
title={<FormattedMessage id="metrics.screens" defaultMessage="Screen" />}
|
||||||
|
type="screen"
|
||||||
|
metric={<FormattedMessage id="metrics.visitors" defaultMessage="Visitors" />}
|
||||||
|
websiteId={websiteId}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
47
components/metrics/UTMTable.js
Normal file
47
components/metrics/UTMTable.js
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
import { useIntl, defineMessages } from 'react-intl';
|
||||||
|
import MetricsTable from './MetricsTable';
|
||||||
|
import FilterButtons from 'components/common/FilterButtons';
|
||||||
|
|
||||||
|
export const UTM_SOURCE = 'utm_source';
|
||||||
|
export const UTM_MEDIUM = 'utm_medium';
|
||||||
|
export const UTM_CAMPAIGN = 'utm_campaign';
|
||||||
|
export const UTM_CONTENT = 'utm_content';
|
||||||
|
export const UTM_TERM = 'utm_term';
|
||||||
|
|
||||||
|
const messages = defineMessages({
|
||||||
|
utm_source: { id: 'metrics.utm_source', defaultMessage: 'UTM Source' },
|
||||||
|
utm_medium: { id: 'metrics.utm_medium', defaultMessage: 'UTM Medium' },
|
||||||
|
utm_campaign: { id: 'metrics.utm_campaign', defaultMessage: 'UTM Campaign' },
|
||||||
|
utm_content: { id: 'metrics.utm_content', defaultMessage: 'UTM Content' },
|
||||||
|
utm_term: { id: 'metrics.utm_term', defaultMessage: 'UTM Term' },
|
||||||
|
views: { id: 'metrics.views', defaultMessage: 'Views' },
|
||||||
|
none: { id: 'label.none', defaultMessage: 'None' },
|
||||||
|
});
|
||||||
|
|
||||||
|
export default function UTMTable({ websiteId, showFilters, ...props }) {
|
||||||
|
const [type, setType] = useState(UTM_SOURCE);
|
||||||
|
const { formatMessage } = useIntl();
|
||||||
|
|
||||||
|
const buttons = [
|
||||||
|
{ label: formatMessage(messages.utm_source), value: UTM_SOURCE },
|
||||||
|
{ label: formatMessage(messages.utm_medium), value: UTM_MEDIUM },
|
||||||
|
{ label: formatMessage(messages.utm_campaign), value: UTM_CAMPAIGN },
|
||||||
|
{ label: formatMessage(messages.utm_content), value: UTM_CONTENT },
|
||||||
|
{ label: formatMessage(messages.utm_term), value: UTM_TERM },
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{showFilters && <FilterButtons buttons={buttons} selected={type} onClick={setType} />}
|
||||||
|
<MetricsTable
|
||||||
|
{...props}
|
||||||
|
title={formatMessage(messages[type])}
|
||||||
|
type={type}
|
||||||
|
metric={formatMessage(messages.views)}
|
||||||
|
websiteId={websiteId}
|
||||||
|
delay={0}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
@ -22,6 +22,8 @@ import useFetch from 'hooks/useFetch';
|
||||||
import usePageQuery from 'hooks/usePageQuery';
|
import usePageQuery from 'hooks/usePageQuery';
|
||||||
import { DEFAULT_ANIMATION_DURATION } from 'lib/constants';
|
import { DEFAULT_ANIMATION_DURATION } from 'lib/constants';
|
||||||
import styles from './WebsiteDetails.module.css';
|
import styles from './WebsiteDetails.module.css';
|
||||||
|
import ScreenTable from 'components/metrics/ScreenTable';
|
||||||
|
import UTMTable from 'components/metrics/UTMTable';
|
||||||
|
|
||||||
const views = {
|
const views = {
|
||||||
url: PagesTable,
|
url: PagesTable,
|
||||||
|
|
@ -29,9 +31,11 @@ const views = {
|
||||||
browser: BrowsersTable,
|
browser: BrowsersTable,
|
||||||
os: OSTable,
|
os: OSTable,
|
||||||
device: DevicesTable,
|
device: DevicesTable,
|
||||||
|
screen: ScreenTable,
|
||||||
country: CountriesTable,
|
country: CountriesTable,
|
||||||
language: LanguagesTable,
|
language: LanguagesTable,
|
||||||
event: EventsTable,
|
event: EventsTable,
|
||||||
|
utm: UTMTable,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function WebsiteDetails({ websiteId }) {
|
export default function WebsiteDetails({ websiteId }) {
|
||||||
|
|
@ -64,6 +68,10 @@ export default function WebsiteDetails({ websiteId }) {
|
||||||
label: <FormattedMessage id="metrics.referrers" defaultMessage="Referrers" />,
|
label: <FormattedMessage id="metrics.referrers" defaultMessage="Referrers" />,
|
||||||
value: resolve({ view: 'referrer' }),
|
value: resolve({ view: 'referrer' }),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: <FormattedMessage id="metrics.screens" defaultMessage="Screens" />,
|
||||||
|
value: resolve({ view: 'screen' }),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: <FormattedMessage id="metrics.browsers" defaultMessage="Browsers" />,
|
label: <FormattedMessage id="metrics.browsers" defaultMessage="Browsers" />,
|
||||||
value: resolve({ view: 'browser' }),
|
value: resolve({ view: 'browser' }),
|
||||||
|
|
@ -88,6 +96,10 @@ export default function WebsiteDetails({ websiteId }) {
|
||||||
label: <FormattedMessage id="metrics.events" defaultMessage="Events" />,
|
label: <FormattedMessage id="metrics.events" defaultMessage="Events" />,
|
||||||
value: resolve({ view: 'event' }),
|
value: resolve({ view: 'event' }),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: <FormattedMessage id="metrics.utm" defaultMessage="UTM" />,
|
||||||
|
value: resolve({ view: 'utm' }),
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const tableProps = {
|
const tableProps = {
|
||||||
|
|
|
||||||
|
|
@ -9,7 +9,7 @@ CREATE TABLE `account` (
|
||||||
|
|
||||||
UNIQUE INDEX `username`(`username`),
|
UNIQUE INDEX `username`(`username`),
|
||||||
PRIMARY KEY (`user_id`)
|
PRIMARY KEY (`user_id`)
|
||||||
) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
||||||
|
|
||||||
-- CreateTable
|
-- CreateTable
|
||||||
CREATE TABLE `event` (
|
CREATE TABLE `event` (
|
||||||
|
|
@ -25,7 +25,7 @@ CREATE TABLE `event` (
|
||||||
INDEX `event_session_id_idx`(`session_id`),
|
INDEX `event_session_id_idx`(`session_id`),
|
||||||
INDEX `event_website_id_idx`(`website_id`),
|
INDEX `event_website_id_idx`(`website_id`),
|
||||||
PRIMARY KEY (`event_id`)
|
PRIMARY KEY (`event_id`)
|
||||||
) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
||||||
|
|
||||||
-- CreateTable
|
-- CreateTable
|
||||||
CREATE TABLE `pageview` (
|
CREATE TABLE `pageview` (
|
||||||
|
|
@ -42,7 +42,7 @@ CREATE TABLE `pageview` (
|
||||||
INDEX `pageview_website_id_idx`(`website_id`),
|
INDEX `pageview_website_id_idx`(`website_id`),
|
||||||
INDEX `pageview_website_id_session_id_created_at_idx`(`website_id`, `session_id`, `created_at`),
|
INDEX `pageview_website_id_session_id_created_at_idx`(`website_id`, `session_id`, `created_at`),
|
||||||
PRIMARY KEY (`view_id`)
|
PRIMARY KEY (`view_id`)
|
||||||
) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
||||||
|
|
||||||
-- CreateTable
|
-- CreateTable
|
||||||
CREATE TABLE `session` (
|
CREATE TABLE `session` (
|
||||||
|
|
@ -62,7 +62,7 @@ CREATE TABLE `session` (
|
||||||
INDEX `session_created_at_idx`(`created_at`),
|
INDEX `session_created_at_idx`(`created_at`),
|
||||||
INDEX `session_website_id_idx`(`website_id`),
|
INDEX `session_website_id_idx`(`website_id`),
|
||||||
PRIMARY KEY (`session_id`)
|
PRIMARY KEY (`session_id`)
|
||||||
) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
||||||
|
|
||||||
-- CreateTable
|
-- CreateTable
|
||||||
CREATE TABLE `website` (
|
CREATE TABLE `website` (
|
||||||
|
|
@ -78,7 +78,7 @@ CREATE TABLE `website` (
|
||||||
UNIQUE INDEX `share_id`(`share_id`),
|
UNIQUE INDEX `share_id`(`share_id`),
|
||||||
INDEX `website_user_id_idx`(`user_id`),
|
INDEX `website_user_id_idx`(`user_id`),
|
||||||
PRIMARY KEY (`website_id`)
|
PRIMARY KEY (`website_id`)
|
||||||
) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
||||||
|
|
||||||
-- AddForeignKey
|
-- AddForeignKey
|
||||||
ALTER TABLE `event` ADD CONSTRAINT `event_ibfk_2` FOREIGN KEY (`session_id`) REFERENCES `session`(`session_id`) ON DELETE CASCADE ON UPDATE NO ACTION;
|
ALTER TABLE `event` ADD CONSTRAINT `event_ibfk_2` FOREIGN KEY (`session_id`) REFERENCES `session`(`session_id`) ON DELETE CASCADE ON UPDATE NO ACTION;
|
||||||
|
|
@ -97,3 +97,6 @@ ALTER TABLE `session` ADD CONSTRAINT `session_ibfk_1` FOREIGN KEY (`website_id`)
|
||||||
|
|
||||||
-- AddForeignKey
|
-- AddForeignKey
|
||||||
ALTER TABLE `website` ADD CONSTRAINT `website_ibfk_1` FOREIGN KEY (`user_id`) REFERENCES `account`(`user_id`) ON DELETE CASCADE ON UPDATE NO ACTION;
|
ALTER TABLE `website` ADD CONSTRAINT `website_ibfk_1` FOREIGN KEY (`user_id`) REFERENCES `account`(`user_id`) ON DELETE CASCADE ON UPDATE NO ACTION;
|
||||||
|
|
||||||
|
-- CreateAdminUser
|
||||||
|
INSERT INTO account (username, password, is_admin) values ('admin', '$2b$10$BUli0c.muyCW1ErNJc3jL.vFRFtFJWrT8/GcR4A.sUdCznaXiqFXa', true);
|
||||||
|
|
|
||||||
|
|
@ -9,7 +9,7 @@ datasource db {
|
||||||
|
|
||||||
model account {
|
model account {
|
||||||
user_id Int @id @default(autoincrement())
|
user_id Int @id @default(autoincrement())
|
||||||
username String @unique(map: "account.username_unique") @db.VarChar(255)
|
username String @unique @db.VarChar(255)
|
||||||
password String @db.VarChar(60)
|
password String @db.VarChar(60)
|
||||||
is_admin Boolean @default(false)
|
is_admin Boolean @default(false)
|
||||||
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
||||||
|
|
@ -25,8 +25,8 @@ model event {
|
||||||
url String @db.VarChar(500)
|
url String @db.VarChar(500)
|
||||||
event_type String @db.VarChar(50)
|
event_type String @db.VarChar(50)
|
||||||
event_value String @db.VarChar(50)
|
event_value String @db.VarChar(50)
|
||||||
session session @relation(fields: [session_id], references: [session_id], onDelete: Cascade)
|
session session @relation(fields: [session_id], references: [session_id], onDelete: Cascade, onUpdate: NoAction)
|
||||||
website website @relation(fields: [website_id], references: [website_id], onDelete: Cascade)
|
website website @relation(fields: [website_id], references: [website_id], onDelete: Cascade, onUpdate: NoAction)
|
||||||
|
|
||||||
@@index([created_at])
|
@@index([created_at])
|
||||||
@@index([session_id])
|
@@index([session_id])
|
||||||
|
|
@ -40,8 +40,8 @@ model pageview {
|
||||||
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
||||||
url String @db.VarChar(500)
|
url String @db.VarChar(500)
|
||||||
referrer String? @db.VarChar(500)
|
referrer String? @db.VarChar(500)
|
||||||
session session @relation(fields: [session_id], references: [session_id], onDelete: Cascade)
|
session session @relation(fields: [session_id], references: [session_id], onDelete: Cascade, onUpdate: NoAction)
|
||||||
website website @relation(fields: [website_id], references: [website_id], onDelete: Cascade)
|
website website @relation(fields: [website_id], references: [website_id], onDelete: Cascade, onUpdate: NoAction)
|
||||||
|
|
||||||
@@index([created_at])
|
@@index([created_at])
|
||||||
@@index([session_id])
|
@@index([session_id])
|
||||||
|
|
@ -52,17 +52,17 @@ model pageview {
|
||||||
|
|
||||||
model session {
|
model session {
|
||||||
session_id Int @id @default(autoincrement())
|
session_id Int @id @default(autoincrement())
|
||||||
session_uuid String @unique(map: "session.session_uuid_unique") @db.Uuid
|
session_uuid String @unique @db.Uuid
|
||||||
website_id Int
|
website_id Int
|
||||||
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
||||||
hostname String? @db.VarChar(100)
|
hostname String? @db.VarChar(100)
|
||||||
browser String? @db.VarChar(20)
|
browser String? @db.VarChar(20)
|
||||||
os String? @db.VarChar(20)
|
os String? @db.VarChar(20)
|
||||||
device String? @db.VarChar(20)
|
|
||||||
screen String? @db.VarChar(11)
|
screen String? @db.VarChar(11)
|
||||||
language String? @db.VarChar(35)
|
language String? @db.VarChar(35)
|
||||||
country String? @db.Char(2)
|
country String? @db.Char(2)
|
||||||
website website @relation(fields: [website_id], references: [website_id], onDelete: Cascade)
|
device String? @db.VarChar(20)
|
||||||
|
website website @relation(fields: [website_id], references: [website_id], onDelete: Cascade, onUpdate: NoAction)
|
||||||
event event[]
|
event event[]
|
||||||
pageview pageview[]
|
pageview pageview[]
|
||||||
|
|
||||||
|
|
@ -72,16 +72,16 @@ model session {
|
||||||
|
|
||||||
model website {
|
model website {
|
||||||
website_id Int @id @default(autoincrement())
|
website_id Int @id @default(autoincrement())
|
||||||
website_uuid String @unique(map: "website.website_uuid_unique") @db.Uuid
|
website_uuid String @unique @db.Uuid
|
||||||
user_id Int
|
|
||||||
name String @db.VarChar(100)
|
name String @db.VarChar(100)
|
||||||
domain String? @db.VarChar(500)
|
|
||||||
share_id String? @unique(map: "website.share_id_unique") @db.VarChar(64)
|
|
||||||
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
created_at DateTime? @default(now()) @db.Timestamptz(6)
|
||||||
account account @relation(fields: [user_id], references: [user_id], onDelete: Cascade)
|
user_id Int
|
||||||
|
domain String? @db.VarChar(500)
|
||||||
|
share_id String? @unique(map: "website_share_id_idx") @db.VarChar(64)
|
||||||
|
account account @relation(fields: [user_id], references: [user_id], onDelete: NoAction, onUpdate: NoAction)
|
||||||
event event[]
|
event event[]
|
||||||
pageview pageview[]
|
pageview pageview[]
|
||||||
session session[]
|
session session[]
|
||||||
|
|
||||||
@@index([user_id])
|
@@index([user_id])
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,7 @@
|
||||||
version: '3'
|
version: '3'
|
||||||
services:
|
services:
|
||||||
umami:
|
umami:
|
||||||
image: ghcr.io/mikecao/umami:postgresql-latest
|
image: ghcr.io/umami-software/umami:postgresql-latest
|
||||||
ports:
|
ports:
|
||||||
- "3000:3000"
|
- "3000:3000"
|
||||||
environment:
|
environment:
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "المزيد",
|
"label.more": "المزيد",
|
||||||
"label.name": "الإسم",
|
"label.name": "الإسم",
|
||||||
"label.new-password": "كلمة مرور جديدة",
|
"label.new-password": "كلمة مرور جديدة",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "كلمة المرور",
|
"label.password": "كلمة المرور",
|
||||||
"label.passwords-dont-match": "كلمة المرور غير متطابقة",
|
"label.passwords-dont-match": "كلمة المرور غير متطابقة",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "الأجهزة",
|
"metrics.devices": "الأجهزة",
|
||||||
"metrics.events": "الأحداث",
|
"metrics.events": "الأحداث",
|
||||||
"metrics.filter.combined": "مجمعة",
|
"metrics.filter.combined": "مجمعة",
|
||||||
"metrics.filter.domain-only": "نطاق فقط",
|
|
||||||
"metrics.filter.raw": "مفصلة",
|
"metrics.filter.raw": "مفصلة",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "نظام التشغيل",
|
"metrics.operating-systems": "نظام التشغيل",
|
||||||
"metrics.page-views": "مشاهدات الصفحة",
|
"metrics.page-views": "مشاهدات الصفحة",
|
||||||
"metrics.pages": "الصفحات",
|
"metrics.pages": "الصفحات",
|
||||||
"metrics.referrers": "التحويلات",
|
"metrics.referrers": "التحويلات",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "زائرون فريدون",
|
"metrics.unique-visitors": "زائرون فريدون",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "مشاهدات",
|
"metrics.views": "مشاهدات",
|
||||||
"metrics.visitors": "زوار"
|
"metrics.visitors": "زوار"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Més",
|
"label.more": "Més",
|
||||||
"label.name": "Nom",
|
"label.name": "Nom",
|
||||||
"label.new-password": "Contrasenya nova",
|
"label.new-password": "Contrasenya nova",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Propietari",
|
"label.owner": "Propietari",
|
||||||
"label.password": "Contrasenya",
|
"label.password": "Contrasenya",
|
||||||
"label.passwords-dont-match": "Les contrasenyes no coincideixen",
|
"label.passwords-dont-match": "Les contrasenyes no coincideixen",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispositius",
|
"metrics.devices": "Dispositius",
|
||||||
"metrics.events": "Esdeveniments",
|
"metrics.events": "Esdeveniments",
|
||||||
"metrics.filter.combined": "Combinat",
|
"metrics.filter.combined": "Combinat",
|
||||||
"metrics.filter.domain-only": "Només domini",
|
|
||||||
"metrics.filter.raw": "En cru",
|
"metrics.filter.raw": "En cru",
|
||||||
"metrics.languages": "Llengües",
|
"metrics.languages": "Llengües",
|
||||||
"metrics.operating-systems": "Sistemes operatius",
|
"metrics.operating-systems": "Sistemes operatius",
|
||||||
"metrics.page-views": "Pàgines vistes",
|
"metrics.page-views": "Pàgines vistes",
|
||||||
"metrics.pages": "Pàgines",
|
"metrics.pages": "Pàgines",
|
||||||
"metrics.referrers": "Referents",
|
"metrics.referrers": "Referents",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Visitants únics",
|
"metrics.unique-visitors": "Visitants únics",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Vistes",
|
"metrics.views": "Vistes",
|
||||||
"metrics.visitors": "Visitants"
|
"metrics.visitors": "Visitants"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Více",
|
"label.more": "Více",
|
||||||
"label.name": "Jméno",
|
"label.name": "Jméno",
|
||||||
"label.new-password": "Nové heslo",
|
"label.new-password": "Nové heslo",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Heslo",
|
"label.password": "Heslo",
|
||||||
"label.passwords-dont-match": "Hesla se neschodují",
|
"label.passwords-dont-match": "Hesla se neschodují",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Zařízení",
|
"metrics.devices": "Zařízení",
|
||||||
"metrics.events": "Události",
|
"metrics.events": "Události",
|
||||||
"metrics.filter.combined": "Kombinace",
|
"metrics.filter.combined": "Kombinace",
|
||||||
"metrics.filter.domain-only": "Domény",
|
|
||||||
"metrics.filter.raw": "Nezpracované",
|
"metrics.filter.raw": "Nezpracované",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Operační systém",
|
"metrics.operating-systems": "Operační systém",
|
||||||
"metrics.page-views": "Zobrazení stránek",
|
"metrics.page-views": "Zobrazení stránek",
|
||||||
"metrics.pages": "Stránky",
|
"metrics.pages": "Stránky",
|
||||||
"metrics.referrers": "Odkazy",
|
"metrics.referrers": "Odkazy",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Jedinečné návštěvy",
|
"metrics.unique-visitors": "Jedinečné návštěvy",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Zobrazení",
|
"metrics.views": "Zobrazení",
|
||||||
"metrics.visitors": "Návštěvy"
|
"metrics.visitors": "Návštěvy"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -4,9 +4,9 @@
|
||||||
"label.add-website": "Tilføj hjemmeside",
|
"label.add-website": "Tilføj hjemmeside",
|
||||||
"label.administrator": "Administrator",
|
"label.administrator": "Administrator",
|
||||||
"label.all": "Alle",
|
"label.all": "Alle",
|
||||||
"label.all-events": "All events",
|
"label.all-events": "Alle hændelser",
|
||||||
"label.all-time": "All time",
|
"label.all-time": "Altid",
|
||||||
"label.all-websites": "Alle websites",
|
"label.all-websites": "Alle hjemmesider",
|
||||||
"label.back": "Tilbage",
|
"label.back": "Tilbage",
|
||||||
"label.cancel": "Afvis",
|
"label.cancel": "Afvis",
|
||||||
"label.change-password": "Skift adgangskode",
|
"label.change-password": "Skift adgangskode",
|
||||||
|
|
@ -28,30 +28,31 @@
|
||||||
"label.enable-share-url": "Aktivér delings-URL",
|
"label.enable-share-url": "Aktivér delings-URL",
|
||||||
"label.invalid": "Ugyldig",
|
"label.invalid": "Ugyldig",
|
||||||
"label.invalid-domain": "Ugyldigt domæne",
|
"label.invalid-domain": "Ugyldigt domæne",
|
||||||
"label.language": "Language",
|
"label.language": "Sprog",
|
||||||
"label.last-days": "Sidste {x} dage",
|
"label.last-days": "Sidste {x} dage",
|
||||||
"label.last-hours": "Sidste {x} timer",
|
"label.last-hours": "Sidste {x} timer",
|
||||||
"label.logged-in-as": "Loggede ind som {username}",
|
"label.logged-in-as": "Logget ind som {username}",
|
||||||
"label.login": "Log ind",
|
"label.login": "Log ind",
|
||||||
"label.logout": "Log ud",
|
"label.logout": "Log ud",
|
||||||
"label.more": "Mere",
|
"label.more": "Mere",
|
||||||
"label.name": "Navn",
|
"label.name": "Navn",
|
||||||
"label.new-password": "Ny adgangskode",
|
"label.new-password": "Ny adgangskode",
|
||||||
"label.owner": "Owner",
|
"label.none": "None",
|
||||||
|
"label.owner": "Ejer",
|
||||||
"label.password": "Adgangskode",
|
"label.password": "Adgangskode",
|
||||||
"label.passwords-dont-match": "Adgangskoder matcher ikke",
|
"label.passwords-dont-match": "Adgangskoderne matcher ikke",
|
||||||
"label.profile": "Profil",
|
"label.profile": "Profil",
|
||||||
"label.realtime": "Realtid",
|
"label.realtime": "Realtid",
|
||||||
"label.realtime-logs": "Realtid logs",
|
"label.realtime-logs": "Realtid logs",
|
||||||
"label.refresh": "Opdater",
|
"label.refresh": "Opdater",
|
||||||
"label.required": "Påkrævet",
|
"label.required": "Påkrævet",
|
||||||
"label.reset": "Reset",
|
"label.reset": "Nulstil",
|
||||||
"label.reset-website": "Reset statistics",
|
"label.reset-website": "Nulstil statistikker",
|
||||||
"label.save": "Gem",
|
"label.save": "Gem",
|
||||||
"label.settings": "Indstillinger",
|
"label.settings": "Indstillinger",
|
||||||
"label.share-url": "Del URL",
|
"label.share-url": "Del URL",
|
||||||
"label.single-day": "Enkelt dag",
|
"label.single-day": "Enkelt dag",
|
||||||
"label.theme": "Theme",
|
"label.theme": "Tema",
|
||||||
"label.this-month": "Denne måned",
|
"label.this-month": "Denne måned",
|
||||||
"label.this-week": "Denne uge",
|
"label.this-week": "Denne uge",
|
||||||
"label.this-year": "Dette år",
|
"label.this-year": "Dette år",
|
||||||
|
|
@ -64,7 +65,7 @@
|
||||||
"label.websites": "Hjemmesider",
|
"label.websites": "Hjemmesider",
|
||||||
"message.active-users": "{x} nuværende {x, plural, one {bruger} other {brugere}}",
|
"message.active-users": "{x} nuværende {x, plural, one {bruger} other {brugere}}",
|
||||||
"message.confirm-delete": "Er du sikker på at du vil slette {target}?",
|
"message.confirm-delete": "Er du sikker på at du vil slette {target}?",
|
||||||
"message.confirm-reset": "Are your sure you want to reset {target}'s statistics?",
|
"message.confirm-reset": "Er du sikker på at du ville nulstille {target}'s statistikker?",
|
||||||
"message.copied": "Kopieret!",
|
"message.copied": "Kopieret!",
|
||||||
"message.delete-warning": "Alle tilknyttede data slettes også.",
|
"message.delete-warning": "Alle tilknyttede data slettes også.",
|
||||||
"message.failure": "Noget gik galt.",
|
"message.failure": "Noget gik galt.",
|
||||||
|
|
@ -75,14 +76,14 @@
|
||||||
"message.log.visitor": "Besøgende fra {country} bruger {browser} på {os} {device}",
|
"message.log.visitor": "Besøgende fra {country} bruger {browser} på {os} {device}",
|
||||||
"message.new-version-available": "Ny udgave af Umami {version} er tilgængelig!",
|
"message.new-version-available": "Ny udgave af Umami {version} er tilgængelig!",
|
||||||
"message.no-data-available": "Ingen data tilgængelig.",
|
"message.no-data-available": "Ingen data tilgængelig.",
|
||||||
"message.no-websites-configured": "Du har ikke konfigureret nogen websteder.",
|
"message.no-websites-configured": "Du har ikke konfigureret nogen hjemmesider.",
|
||||||
"message.page-not-found": "Side ikke fundet.",
|
"message.page-not-found": "Side ikke fundet.",
|
||||||
"message.powered-by": "Drevet af {name}",
|
"message.powered-by": "Drevet af {name}",
|
||||||
"message.reset-warning": "All statistics for this website will be deleted, but your tracking code will remain intact.",
|
"message.reset-warning": "Alle statistikker for denne hjemmeside ville blive slettet, men sporingskode ville forblive intakt.",
|
||||||
"message.save-success": "Gemt!",
|
"message.save-success": "Gemt!",
|
||||||
"message.share-url": "Dette er den offentligt delings-URL til {target}.",
|
"message.share-url": "Dette er den offentlige delings-URL til {target}.",
|
||||||
"message.toggle-charts": "Toggle charts",
|
"message.toggle-charts": "Ændre graf",
|
||||||
"message.track-stats": "For at spore statistik for {target} skal du placere følgende kode i {head} sektionen på dit websted.",
|
"message.track-stats": "For at spore statistik for {target} skal du placere følgende kode i {head} sektionen på din hjemmeside.",
|
||||||
"message.type-delete": "Skriv {delete} i boksen nedenfor, for at bekræfte.",
|
"message.type-delete": "Skriv {delete} i boksen nedenfor, for at bekræfte.",
|
||||||
"message.type-reset": "Skriv {reset} i boksen nedenfor, for at bekræfte.",
|
"message.type-reset": "Skriv {reset} i boksen nedenfor, for at bekræfte.",
|
||||||
"metrics.actions": "Handlinger",
|
"metrics.actions": "Handlinger",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Enheder",
|
"metrics.devices": "Enheder",
|
||||||
"metrics.events": "Hændelser",
|
"metrics.events": "Hændelser",
|
||||||
"metrics.filter.combined": "Kombineret",
|
"metrics.filter.combined": "Kombineret",
|
||||||
"metrics.filter.domain-only": "Kun domæne",
|
|
||||||
"metrics.filter.raw": "Rå",
|
"metrics.filter.raw": "Rå",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Sprog",
|
||||||
"metrics.operating-systems": "Operativsystemer",
|
"metrics.operating-systems": "Operativsystemer",
|
||||||
"metrics.page-views": "Sidevisninger",
|
"metrics.page-views": "Sidevisninger",
|
||||||
"metrics.pages": "Sider",
|
"metrics.pages": "Sider",
|
||||||
"metrics.referrers": "Henvisninger",
|
"metrics.referrers": "Henvisninger",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unikke besøgende",
|
"metrics.unique-visitors": "Unikke besøgende",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visninger",
|
"metrics.views": "Visninger",
|
||||||
"metrics.visitors": "Besøgende"
|
"metrics.visitors": "Besøgende"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Mehr",
|
"label.more": "Mehr",
|
||||||
"label.name": "Name",
|
"label.name": "Name",
|
||||||
"label.new-password": "Neues Passwort",
|
"label.new-password": "Neues Passwort",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Besitzer",
|
"label.owner": "Besitzer",
|
||||||
"label.password": "Passwort",
|
"label.password": "Passwort",
|
||||||
"label.passwords-dont-match": "Passwörter stimmen nicht überein",
|
"label.passwords-dont-match": "Passwörter stimmen nicht überein",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Geräte",
|
"metrics.devices": "Geräte",
|
||||||
"metrics.events": "Ereignisse",
|
"metrics.events": "Ereignisse",
|
||||||
"metrics.filter.combined": "Kombiniert",
|
"metrics.filter.combined": "Kombiniert",
|
||||||
"metrics.filter.domain-only": "Nur diese Domain",
|
|
||||||
"metrics.filter.raw": "Rohdaten",
|
"metrics.filter.raw": "Rohdaten",
|
||||||
"metrics.languages": "Sprachen",
|
"metrics.languages": "Sprachen",
|
||||||
"metrics.operating-systems": "Betriebssysteme",
|
"metrics.operating-systems": "Betriebssysteme",
|
||||||
"metrics.page-views": "Seitenaufrufe",
|
"metrics.page-views": "Seitenaufrufe",
|
||||||
"metrics.pages": "Seiten",
|
"metrics.pages": "Seiten",
|
||||||
"metrics.referrers": "Referrer",
|
"metrics.referrers": "Referrer",
|
||||||
|
"metrics.screens": "Bildschirmauflösungen",
|
||||||
"metrics.unique-visitors": "Eindeutige Besucher",
|
"metrics.unique-visitors": "Eindeutige Besucher",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Aufrufe",
|
"metrics.views": "Aufrufe",
|
||||||
"metrics.visitors": "Besucher"
|
"metrics.visitors": "Besucher"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Περισσότερα",
|
"label.more": "Περισσότερα",
|
||||||
"label.name": "Όνομα",
|
"label.name": "Όνομα",
|
||||||
"label.new-password": "Νέος κωδικός",
|
"label.new-password": "Νέος κωδικός",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Κωδικός",
|
"label.password": "Κωδικός",
|
||||||
"label.passwords-dont-match": "Οι κωδικοί πρόσβασης δεν ταιριάζουν",
|
"label.passwords-dont-match": "Οι κωδικοί πρόσβασης δεν ταιριάζουν",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Συσκευές",
|
"metrics.devices": "Συσκευές",
|
||||||
"metrics.events": "Γεγονότα",
|
"metrics.events": "Γεγονότα",
|
||||||
"metrics.filter.combined": "Σε συνδυασμό",
|
"metrics.filter.combined": "Σε συνδυασμό",
|
||||||
"metrics.filter.domain-only": "Μόνο τομέας",
|
|
||||||
"metrics.filter.raw": "Ακατέργαστο",
|
"metrics.filter.raw": "Ακατέργαστο",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Λειτουργικά συστήματα",
|
"metrics.operating-systems": "Λειτουργικά συστήματα",
|
||||||
"metrics.page-views": "Προβολές σελίδας",
|
"metrics.page-views": "Προβολές σελίδας",
|
||||||
"metrics.pages": "Σελίδες",
|
"metrics.pages": "Σελίδες",
|
||||||
"metrics.referrers": "Παραπομπές",
|
"metrics.referrers": "Παραπομπές",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Μοναδικοί επισκέπτες",
|
"metrics.unique-visitors": "Μοναδικοί επισκέπτες",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Προβολές",
|
"metrics.views": "Προβολές",
|
||||||
"metrics.visitors": "Επισκέπτες"
|
"metrics.visitors": "Επισκέπτες"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "More",
|
"label.more": "More",
|
||||||
"label.name": "Name",
|
"label.name": "Name",
|
||||||
"label.new-password": "New password",
|
"label.new-password": "New password",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Password",
|
"label.password": "Password",
|
||||||
"label.passwords-dont-match": "Passwords don't match",
|
"label.passwords-dont-match": "Passwords don't match",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Devices",
|
"metrics.devices": "Devices",
|
||||||
"metrics.events": "Events",
|
"metrics.events": "Events",
|
||||||
"metrics.filter.combined": "Combined",
|
"metrics.filter.combined": "Combined",
|
||||||
"metrics.filter.domain-only": "Domain only",
|
|
||||||
"metrics.filter.raw": "Raw",
|
"metrics.filter.raw": "Raw",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Operating systems",
|
"metrics.operating-systems": "Operating systems",
|
||||||
"metrics.page-views": "Page views",
|
"metrics.page-views": "Page views",
|
||||||
"metrics.pages": "Pages",
|
"metrics.pages": "Pages",
|
||||||
"metrics.referrers": "Referrers",
|
"metrics.referrers": "Referrers",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unique visitors",
|
"metrics.unique-visitors": "Unique visitors",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Views",
|
"metrics.views": "Views",
|
||||||
"metrics.visitors": "Visitors"
|
"metrics.visitors": "Visitors"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "More",
|
"label.more": "More",
|
||||||
"label.name": "Name",
|
"label.name": "Name",
|
||||||
"label.new-password": "New password",
|
"label.new-password": "New password",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Password",
|
"label.password": "Password",
|
||||||
"label.passwords-dont-match": "Passwords don't match",
|
"label.passwords-dont-match": "Passwords don't match",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Devices",
|
"metrics.devices": "Devices",
|
||||||
"metrics.events": "Events",
|
"metrics.events": "Events",
|
||||||
"metrics.filter.combined": "Combined",
|
"metrics.filter.combined": "Combined",
|
||||||
"metrics.filter.domain-only": "Domain only",
|
|
||||||
"metrics.filter.raw": "Raw",
|
"metrics.filter.raw": "Raw",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Operating systems",
|
"metrics.operating-systems": "Operating systems",
|
||||||
"metrics.page-views": "Page views",
|
"metrics.page-views": "Page views",
|
||||||
"metrics.pages": "Pages",
|
"metrics.pages": "Pages",
|
||||||
"metrics.referrers": "Referrers",
|
"metrics.referrers": "Referrers",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unique visitors",
|
"metrics.unique-visitors": "Unique visitors",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Views",
|
"metrics.views": "Views",
|
||||||
"metrics.visitors": "Visitors"
|
"metrics.visitors": "Visitors"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Más",
|
"label.more": "Más",
|
||||||
"label.name": "Nombre",
|
"label.name": "Nombre",
|
||||||
"label.new-password": "Nueva contraseña",
|
"label.new-password": "Nueva contraseña",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Propietario",
|
"label.owner": "Propietario",
|
||||||
"label.password": "Contraseña",
|
"label.password": "Contraseña",
|
||||||
"label.passwords-dont-match": "Las contraseñas no coinciden",
|
"label.passwords-dont-match": "Las contraseñas no coinciden",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispositivos",
|
"metrics.devices": "Dispositivos",
|
||||||
"metrics.events": "Eventos",
|
"metrics.events": "Eventos",
|
||||||
"metrics.filter.combined": "Combinado",
|
"metrics.filter.combined": "Combinado",
|
||||||
"metrics.filter.domain-only": "Únicamente dominio",
|
|
||||||
"metrics.filter.raw": "Personalizado",
|
"metrics.filter.raw": "Personalizado",
|
||||||
"metrics.languages": "Idiomas",
|
"metrics.languages": "Idiomas",
|
||||||
"metrics.operating-systems": "Sistemas operativos",
|
"metrics.operating-systems": "Sistemas operativos",
|
||||||
"metrics.page-views": "Vistas",
|
"metrics.page-views": "Vistas",
|
||||||
"metrics.pages": "Páginas",
|
"metrics.pages": "Páginas",
|
||||||
"metrics.referrers": "Referentes",
|
"metrics.referrers": "Referentes",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Visitantes únicos",
|
"metrics.unique-visitors": "Visitantes únicos",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Vistas",
|
"metrics.views": "Vistas",
|
||||||
"metrics.visitors": "Visitantes"
|
"metrics.visitors": "Visitantes"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "بیشتر",
|
"label.more": "بیشتر",
|
||||||
"label.name": "نام",
|
"label.name": "نام",
|
||||||
"label.new-password": "رمز جدید",
|
"label.new-password": "رمز جدید",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "ایجاد شده توسط",
|
"label.owner": "ایجاد شده توسط",
|
||||||
"label.password": "رمز",
|
"label.password": "رمز",
|
||||||
"label.passwords-dont-match": "رمزها یکسان نیستند",
|
"label.passwords-dont-match": "رمزها یکسان نیستند",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "دستگاهها",
|
"metrics.devices": "دستگاهها",
|
||||||
"metrics.events": "رویدادها",
|
"metrics.events": "رویدادها",
|
||||||
"metrics.filter.combined": "ترکیب شده",
|
"metrics.filter.combined": "ترکیب شده",
|
||||||
"metrics.filter.domain-only": "فقط دامنه",
|
|
||||||
"metrics.filter.raw": "خام",
|
"metrics.filter.raw": "خام",
|
||||||
"metrics.languages": "زبانها",
|
"metrics.languages": "زبانها",
|
||||||
"metrics.operating-systems": "سیستمعاملها",
|
"metrics.operating-systems": "سیستمعاملها",
|
||||||
"metrics.page-views": "بازدید صفحه",
|
"metrics.page-views": "بازدید صفحه",
|
||||||
"metrics.pages": "صفحهها",
|
"metrics.pages": "صفحهها",
|
||||||
"metrics.referrers": "ارجاع دهندگان",
|
"metrics.referrers": "ارجاع دهندگان",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "بازدیدکنندههای یکتا",
|
"metrics.unique-visitors": "بازدیدکنندههای یکتا",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "بازدید",
|
"metrics.views": "بازدید",
|
||||||
"metrics.visitors": "بازدیدکننده"
|
"metrics.visitors": "بازدیدکننده"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Lisää",
|
"label.more": "Lisää",
|
||||||
"label.name": "Nimi",
|
"label.name": "Nimi",
|
||||||
"label.new-password": "Uusi salasana",
|
"label.new-password": "Uusi salasana",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Omistaja",
|
"label.owner": "Omistaja",
|
||||||
"label.password": "Salasana",
|
"label.password": "Salasana",
|
||||||
"label.passwords-dont-match": "Salasanat eivät täsmää",
|
"label.passwords-dont-match": "Salasanat eivät täsmää",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Laitteet",
|
"metrics.devices": "Laitteet",
|
||||||
"metrics.events": "Tapahtumat",
|
"metrics.events": "Tapahtumat",
|
||||||
"metrics.filter.combined": "Yhdistetty",
|
"metrics.filter.combined": "Yhdistetty",
|
||||||
"metrics.filter.domain-only": "Vain verkkotunnus",
|
|
||||||
"metrics.filter.raw": "Käsittelemätön",
|
"metrics.filter.raw": "Käsittelemätön",
|
||||||
"metrics.languages": "Kielet",
|
"metrics.languages": "Kielet",
|
||||||
"metrics.operating-systems": "Käyttöjärjestelmät",
|
"metrics.operating-systems": "Käyttöjärjestelmät",
|
||||||
"metrics.page-views": "Sivun näyttökerrat",
|
"metrics.page-views": "Sivun näyttökerrat",
|
||||||
"metrics.pages": "Sivut",
|
"metrics.pages": "Sivut",
|
||||||
"metrics.referrers": "Viittaajat",
|
"metrics.referrers": "Viittaajat",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Yksittäiset kävijät",
|
"metrics.unique-visitors": "Yksittäiset kävijät",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Näyttökerrat",
|
"metrics.views": "Näyttökerrat",
|
||||||
"metrics.visitors": "Vierailijat"
|
"metrics.visitors": "Vierailijat"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Meira",
|
"label.more": "Meira",
|
||||||
"label.name": "Navn",
|
"label.name": "Navn",
|
||||||
"label.new-password": "Nýtt loyniorð",
|
"label.new-password": "Nýtt loyniorð",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Loyniorð",
|
"label.password": "Loyniorð",
|
||||||
"label.passwords-dont-match": "Loyniorðini eru ikki eins",
|
"label.passwords-dont-match": "Loyniorðini eru ikki eins",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Tóleindir",
|
"metrics.devices": "Tóleindir",
|
||||||
"metrics.events": "Hendingar/tiltøk",
|
"metrics.events": "Hendingar/tiltøk",
|
||||||
"metrics.filter.combined": "Samansett",
|
"metrics.filter.combined": "Samansett",
|
||||||
"metrics.filter.domain-only": "Bara økisnavn",
|
|
||||||
"metrics.filter.raw": "Óviðgjørt",
|
"metrics.filter.raw": "Óviðgjørt",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Stýrikervir",
|
"metrics.operating-systems": "Stýrikervir",
|
||||||
"metrics.page-views": "Opnaðar síðir",
|
"metrics.page-views": "Opnaðar síðir",
|
||||||
"metrics.pages": "Síðir",
|
"metrics.pages": "Síðir",
|
||||||
"metrics.referrers": "Framsendingar",
|
"metrics.referrers": "Framsendingar",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Einsýna vitjanir",
|
"metrics.unique-visitors": "Einsýna vitjanir",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Sýningar",
|
"metrics.views": "Sýningar",
|
||||||
"metrics.visitors": "Vitjandi"
|
"metrics.visitors": "Vitjandi"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Plus",
|
"label.more": "Plus",
|
||||||
"label.name": "Nom",
|
"label.name": "Nom",
|
||||||
"label.new-password": "Nouveau mot de passe",
|
"label.new-password": "Nouveau mot de passe",
|
||||||
|
"label.none": "Aucun·e",
|
||||||
"label.owner": "Propriétaire",
|
"label.owner": "Propriétaire",
|
||||||
"label.password": "Mot de passe",
|
"label.password": "Mot de passe",
|
||||||
"label.passwords-dont-match": "Les mots de passe ne correspondent pas",
|
"label.passwords-dont-match": "Les mots de passe ne correspondent pas",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Appareils",
|
"metrics.devices": "Appareils",
|
||||||
"metrics.events": "Événements",
|
"metrics.events": "Événements",
|
||||||
"metrics.filter.combined": "Combiné",
|
"metrics.filter.combined": "Combiné",
|
||||||
"metrics.filter.domain-only": "Domaine uniquement",
|
|
||||||
"metrics.filter.raw": "Brut",
|
"metrics.filter.raw": "Brut",
|
||||||
"metrics.languages": "Langages",
|
"metrics.languages": "Langages",
|
||||||
"metrics.operating-systems": "Systèmes d'exploitation",
|
"metrics.operating-systems": "Systèmes d'exploitation",
|
||||||
"metrics.page-views": "Pages vues",
|
"metrics.page-views": "Pages vues",
|
||||||
"metrics.pages": "Pages",
|
"metrics.pages": "Pages",
|
||||||
"metrics.referrers": "Sources",
|
"metrics.referrers": "Sources",
|
||||||
|
"metrics.screens": "Tailles d'écran",
|
||||||
"metrics.unique-visitors": "Visiteurs uniques",
|
"metrics.unique-visitors": "Visiteurs uniques",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Vues",
|
"metrics.views": "Vues",
|
||||||
"metrics.visitors": "Visiteurs"
|
"metrics.visitors": "Visiteurs"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Máis",
|
"label.more": "Máis",
|
||||||
"label.name": "Nome",
|
"label.name": "Nome",
|
||||||
"label.new-password": "Novo contrasinal",
|
"label.new-password": "Novo contrasinal",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Dona",
|
"label.owner": "Dona",
|
||||||
"label.password": "Contrasinal",
|
"label.password": "Contrasinal",
|
||||||
"label.passwords-dont-match": "Non concordan os contrasinais",
|
"label.passwords-dont-match": "Non concordan os contrasinais",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispositivos",
|
"metrics.devices": "Dispositivos",
|
||||||
"metrics.events": "Eventos",
|
"metrics.events": "Eventos",
|
||||||
"metrics.filter.combined": "Combinado",
|
"metrics.filter.combined": "Combinado",
|
||||||
"metrics.filter.domain-only": "Só dominio",
|
|
||||||
"metrics.filter.raw": "Raw",
|
"metrics.filter.raw": "Raw",
|
||||||
"metrics.languages": "Idiomas",
|
"metrics.languages": "Idiomas",
|
||||||
"metrics.operating-systems": "Sistemas operativos",
|
"metrics.operating-systems": "Sistemas operativos",
|
||||||
"metrics.page-views": "Vistas de páxinas",
|
"metrics.page-views": "Vistas de páxinas",
|
||||||
"metrics.pages": "Páxinas",
|
"metrics.pages": "Páxinas",
|
||||||
"metrics.referrers": "Orixes",
|
"metrics.referrers": "Orixes",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Visitas únicas",
|
"metrics.unique-visitors": "Visitas únicas",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visualizacións",
|
"metrics.views": "Visualizacións",
|
||||||
"metrics.visitors": "Visitantes"
|
"metrics.visitors": "Visitantes"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "עוד",
|
"label.more": "עוד",
|
||||||
"label.name": "שם",
|
"label.name": "שם",
|
||||||
"label.new-password": "סיסמה חדשה",
|
"label.new-password": "סיסמה חדשה",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "סיסמה",
|
"label.password": "סיסמה",
|
||||||
"label.passwords-dont-match": "סיסמאות לא תואמות",
|
"label.passwords-dont-match": "סיסמאות לא תואמות",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "מכשירים",
|
"metrics.devices": "מכשירים",
|
||||||
"metrics.events": "אירועים",
|
"metrics.events": "אירועים",
|
||||||
"metrics.filter.combined": "משותף",
|
"metrics.filter.combined": "משותף",
|
||||||
"metrics.filter.domain-only": "דומיין בלבד",
|
|
||||||
"metrics.filter.raw": "גולמי",
|
"metrics.filter.raw": "גולמי",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "מערכות הפעלה",
|
"metrics.operating-systems": "מערכות הפעלה",
|
||||||
"metrics.page-views": "צפיות בדפים",
|
"metrics.page-views": "צפיות בדפים",
|
||||||
"metrics.pages": "דפים",
|
"metrics.pages": "דפים",
|
||||||
"metrics.referrers": "מפנים",
|
"metrics.referrers": "מפנים",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "מבקרים ייחודיים",
|
"metrics.unique-visitors": "מבקרים ייחודיים",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "צפיות",
|
"metrics.views": "צפיות",
|
||||||
"metrics.visitors": "מבקרים"
|
"metrics.visitors": "מבקרים"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "और",
|
"label.more": "और",
|
||||||
"label.name": "नाम",
|
"label.name": "नाम",
|
||||||
"label.new-password": "नया पासवर्ड",
|
"label.new-password": "नया पासवर्ड",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "पासवर्ड",
|
"label.password": "पासवर्ड",
|
||||||
"label.passwords-dont-match": "पासवर्ड मेल नहीं खाते",
|
"label.passwords-dont-match": "पासवर्ड मेल नहीं खाते",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "उपकरण",
|
"metrics.devices": "उपकरण",
|
||||||
"metrics.events": "स्पर्धाएँ",
|
"metrics.events": "स्पर्धाएँ",
|
||||||
"metrics.filter.combined": "संयुक्त",
|
"metrics.filter.combined": "संयुक्त",
|
||||||
"metrics.filter.domain-only": "केवल डोमेन",
|
|
||||||
"metrics.filter.raw": "रॉ",
|
"metrics.filter.raw": "रॉ",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "ऑपरेटिंग सिस्टम",
|
"metrics.operating-systems": "ऑपरेटिंग सिस्टम",
|
||||||
"metrics.page-views": "पृष्ठ दृश्य",
|
"metrics.page-views": "पृष्ठ दृश्य",
|
||||||
"metrics.pages": "पृष्ठों",
|
"metrics.pages": "पृष्ठों",
|
||||||
"metrics.referrers": "सन्दर्भदाता",
|
"metrics.referrers": "सन्दर्भदाता",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "अद्वितीय आगंतुकों",
|
"metrics.unique-visitors": "अद्वितीय आगंतुकों",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "दृश्य",
|
"metrics.views": "दृश्य",
|
||||||
"metrics.visitors": "आगंतुकों"
|
"metrics.visitors": "आगंतुकों"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Bővebben",
|
"label.more": "Bővebben",
|
||||||
"label.name": "Név",
|
"label.name": "Név",
|
||||||
"label.new-password": "Új jelszó",
|
"label.new-password": "Új jelszó",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Jelszó",
|
"label.password": "Jelszó",
|
||||||
"label.passwords-dont-match": "A jelszavak nem egyeznek",
|
"label.passwords-dont-match": "A jelszavak nem egyeznek",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Eszközök",
|
"metrics.devices": "Eszközök",
|
||||||
"metrics.events": "Események",
|
"metrics.events": "Események",
|
||||||
"metrics.filter.combined": "Összevont",
|
"metrics.filter.combined": "Összevont",
|
||||||
"metrics.filter.domain-only": "Csak domain",
|
|
||||||
"metrics.filter.raw": "Nyers",
|
"metrics.filter.raw": "Nyers",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Operációs rendszerek",
|
"metrics.operating-systems": "Operációs rendszerek",
|
||||||
"metrics.page-views": "Oldalmegtekintések",
|
"metrics.page-views": "Oldalmegtekintések",
|
||||||
"metrics.pages": "Oldalak",
|
"metrics.pages": "Oldalak",
|
||||||
"metrics.referrers": "Hivatkozók",
|
"metrics.referrers": "Hivatkozók",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Egyedi látogatók",
|
"metrics.unique-visitors": "Egyedi látogatók",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Megtekintések",
|
"metrics.views": "Megtekintések",
|
||||||
"metrics.visitors": "Látogatók"
|
"metrics.visitors": "Látogatók"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Lebih banyak",
|
"label.more": "Lebih banyak",
|
||||||
"label.name": "Nama",
|
"label.name": "Nama",
|
||||||
"label.new-password": "Kata sandi baru",
|
"label.new-password": "Kata sandi baru",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Kata sandi",
|
"label.password": "Kata sandi",
|
||||||
"label.passwords-dont-match": "Kata sandi tidak cocok",
|
"label.passwords-dont-match": "Kata sandi tidak cocok",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Perangkat",
|
"metrics.devices": "Perangkat",
|
||||||
"metrics.events": "Perihal",
|
"metrics.events": "Perihal",
|
||||||
"metrics.filter.combined": "Gabungan",
|
"metrics.filter.combined": "Gabungan",
|
||||||
"metrics.filter.domain-only": "Hanya domain",
|
|
||||||
"metrics.filter.raw": "Mentah",
|
"metrics.filter.raw": "Mentah",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Sistem Operasi",
|
"metrics.operating-systems": "Sistem Operasi",
|
||||||
"metrics.page-views": "Tampilan halaman",
|
"metrics.page-views": "Tampilan halaman",
|
||||||
"metrics.pages": "Halaman",
|
"metrics.pages": "Halaman",
|
||||||
"metrics.referrers": "Perujuk",
|
"metrics.referrers": "Perujuk",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Pengunjung unik",
|
"metrics.unique-visitors": "Pengunjung unik",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Tampilan",
|
"metrics.views": "Tampilan",
|
||||||
"metrics.visitors": "Pengunjung"
|
"metrics.visitors": "Pengunjung"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Dettagli",
|
"label.more": "Dettagli",
|
||||||
"label.name": "Nome",
|
"label.name": "Nome",
|
||||||
"label.new-password": "Nuova password",
|
"label.new-password": "Nuova password",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Proprietario",
|
"label.owner": "Proprietario",
|
||||||
"label.password": "Password",
|
"label.password": "Password",
|
||||||
"label.passwords-dont-match": "Le password non corrispondono",
|
"label.passwords-dont-match": "Le password non corrispondono",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispositivi",
|
"metrics.devices": "Dispositivi",
|
||||||
"metrics.events": "Eventi",
|
"metrics.events": "Eventi",
|
||||||
"metrics.filter.combined": "Aggregati",
|
"metrics.filter.combined": "Aggregati",
|
||||||
"metrics.filter.domain-only": "Solo dominio",
|
|
||||||
"metrics.filter.raw": "Raw",
|
"metrics.filter.raw": "Raw",
|
||||||
"metrics.languages": "Lingue",
|
"metrics.languages": "Lingue",
|
||||||
"metrics.operating-systems": "Sistemi operativi",
|
"metrics.operating-systems": "Sistemi operativi",
|
||||||
"metrics.page-views": "Visualizzazioni di pagina",
|
"metrics.page-views": "Visualizzazioni di pagina",
|
||||||
"metrics.pages": "Pagine",
|
"metrics.pages": "Pagine",
|
||||||
"metrics.referrers": "Referrers",
|
"metrics.referrers": "Referrers",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Visitatori unici",
|
"metrics.unique-visitors": "Visitatori unici",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visualizzazioni",
|
"metrics.views": "Visualizzazioni",
|
||||||
"metrics.visitors": "Visitatori"
|
"metrics.visitors": "Visitatori"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "さらに表示",
|
"label.more": "さらに表示",
|
||||||
"label.name": "名前",
|
"label.name": "名前",
|
||||||
"label.new-password": "新しいパスワード",
|
"label.new-password": "新しいパスワード",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "パスワード",
|
"label.password": "パスワード",
|
||||||
"label.passwords-dont-match": "パスワードが一致しません",
|
"label.passwords-dont-match": "パスワードが一致しません",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "デバイス",
|
"metrics.devices": "デバイス",
|
||||||
"metrics.events": "イベント",
|
"metrics.events": "イベント",
|
||||||
"metrics.filter.combined": "パスまで",
|
"metrics.filter.combined": "パスまで",
|
||||||
"metrics.filter.domain-only": "ドメインのみ",
|
|
||||||
"metrics.filter.raw": "すべて表示",
|
"metrics.filter.raw": "すべて表示",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "OS",
|
"metrics.operating-systems": "OS",
|
||||||
"metrics.page-views": "閲覧数",
|
"metrics.page-views": "閲覧数",
|
||||||
"metrics.pages": "ページ",
|
"metrics.pages": "ページ",
|
||||||
"metrics.referrers": "リファラー",
|
"metrics.referrers": "リファラー",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "ユニーク訪問者数",
|
"metrics.unique-visitors": "ユニーク訪問者数",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "閲覧数",
|
"metrics.views": "閲覧数",
|
||||||
"metrics.visitors": "訪問者数"
|
"metrics.visitors": "訪問者数"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "더 보기",
|
"label.more": "더 보기",
|
||||||
"label.name": "이름",
|
"label.name": "이름",
|
||||||
"label.new-password": "새 비밀번호",
|
"label.new-password": "새 비밀번호",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "비밀번호",
|
"label.password": "비밀번호",
|
||||||
"label.passwords-dont-match": "비밀번호가 일치하지 않음",
|
"label.passwords-dont-match": "비밀번호가 일치하지 않음",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "기기",
|
"metrics.devices": "기기",
|
||||||
"metrics.events": "이벤트",
|
"metrics.events": "이벤트",
|
||||||
"metrics.filter.combined": "합쳐서 보기",
|
"metrics.filter.combined": "합쳐서 보기",
|
||||||
"metrics.filter.domain-only": "도메인만",
|
|
||||||
"metrics.filter.raw": "전체 보기",
|
"metrics.filter.raw": "전체 보기",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "운영체제",
|
"metrics.operating-systems": "운영체제",
|
||||||
"metrics.page-views": "페이지 뷰(PV)",
|
"metrics.page-views": "페이지 뷰(PV)",
|
||||||
"metrics.pages": "페이지",
|
"metrics.pages": "페이지",
|
||||||
"metrics.referrers": "리퍼러",
|
"metrics.referrers": "리퍼러",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "순방문자(UV)",
|
"metrics.unique-visitors": "순방문자(UV)",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "조회수",
|
"metrics.views": "조회수",
|
||||||
"metrics.visitors": "방문객"
|
"metrics.visitors": "방문객"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Daugiau",
|
"label.more": "Daugiau",
|
||||||
"label.name": "Pavadinimas",
|
"label.name": "Pavadinimas",
|
||||||
"label.new-password": "Naujas slaptažodis",
|
"label.new-password": "Naujas slaptažodis",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Savininkas",
|
"label.owner": "Savininkas",
|
||||||
"label.password": "Slaptažodis",
|
"label.password": "Slaptažodis",
|
||||||
"label.passwords-dont-match": "Slaptažodžiai nesutampa",
|
"label.passwords-dont-match": "Slaptažodžiai nesutampa",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Įrenginiai",
|
"metrics.devices": "Įrenginiai",
|
||||||
"metrics.events": "Įvykiai",
|
"metrics.events": "Įvykiai",
|
||||||
"metrics.filter.combined": "Kombinuoti",
|
"metrics.filter.combined": "Kombinuoti",
|
||||||
"metrics.filter.domain-only": "Tik domenas",
|
|
||||||
"metrics.filter.raw": "Neapdoroti",
|
"metrics.filter.raw": "Neapdoroti",
|
||||||
"metrics.languages": "Kalbos",
|
"metrics.languages": "Kalbos",
|
||||||
"metrics.operating-systems": "Operacinės sistemos",
|
"metrics.operating-systems": "Operacinės sistemos",
|
||||||
"metrics.page-views": "Puslapių peržiūros",
|
"metrics.page-views": "Puslapių peržiūros",
|
||||||
"metrics.pages": "Puslapiai",
|
"metrics.pages": "Puslapiai",
|
||||||
"metrics.referrers": "Referrers",
|
"metrics.referrers": "Referrers",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unikalūs lankytojai",
|
"metrics.unique-visitors": "Unikalūs lankytojai",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Peržiūros",
|
"metrics.views": "Peržiūros",
|
||||||
"metrics.visitors": "Lankytojai"
|
"metrics.visitors": "Lankytojai"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Цааш",
|
"label.more": "Цааш",
|
||||||
"label.name": "Нэр",
|
"label.name": "Нэр",
|
||||||
"label.new-password": "Шинэ нууц үг",
|
"label.new-password": "Шинэ нууц үг",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Эзэмшигч",
|
"label.owner": "Эзэмшигч",
|
||||||
"label.password": "Нууц үг",
|
"label.password": "Нууц үг",
|
||||||
"label.passwords-dont-match": "Нууц үг тохирохгүй байна",
|
"label.passwords-dont-match": "Нууц үг тохирохгүй байна",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Төхөөрөмж",
|
"metrics.devices": "Төхөөрөмж",
|
||||||
"metrics.events": "Үйлдэл",
|
"metrics.events": "Үйлдэл",
|
||||||
"metrics.filter.combined": "Нэгтгэсэн",
|
"metrics.filter.combined": "Нэгтгэсэн",
|
||||||
"metrics.filter.domain-only": "Зөвхөн домэйн",
|
|
||||||
"metrics.filter.raw": "Түүхий",
|
"metrics.filter.raw": "Түүхий",
|
||||||
"metrics.languages": "Хэл",
|
"metrics.languages": "Хэл",
|
||||||
"metrics.operating-systems": "Үйлдлийн систем",
|
"metrics.operating-systems": "Үйлдлийн систем",
|
||||||
"metrics.page-views": "Хуудас үзсэн",
|
"metrics.page-views": "Хуудас үзсэн",
|
||||||
"metrics.pages": "Хуудас",
|
"metrics.pages": "Хуудас",
|
||||||
"metrics.referrers": "Чиглүүлэгч",
|
"metrics.referrers": "Чиглүүлэгч",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Зочин",
|
"metrics.unique-visitors": "Зочин",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Үзсэн",
|
"metrics.views": "Үзсэн",
|
||||||
"metrics.visitors": "Зочин"
|
"metrics.visitors": "Зочин"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Lebih banyak lagi",
|
"label.more": "Lebih banyak lagi",
|
||||||
"label.name": "Nama",
|
"label.name": "Nama",
|
||||||
"label.new-password": "Kata laluan baru",
|
"label.new-password": "Kata laluan baru",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Kata laluan",
|
"label.password": "Kata laluan",
|
||||||
"label.passwords-dont-match": "Kata laluan tidak sepadan",
|
"label.passwords-dont-match": "Kata laluan tidak sepadan",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Peranti",
|
"metrics.devices": "Peranti",
|
||||||
"metrics.events": "Peristiwa",
|
"metrics.events": "Peristiwa",
|
||||||
"metrics.filter.combined": "Digabungkan",
|
"metrics.filter.combined": "Digabungkan",
|
||||||
"metrics.filter.domain-only": "Domain sahaja",
|
|
||||||
"metrics.filter.raw": "Mentah",
|
"metrics.filter.raw": "Mentah",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Sistem operasi",
|
"metrics.operating-systems": "Sistem operasi",
|
||||||
"metrics.page-views": "Paparan halaman",
|
"metrics.page-views": "Paparan halaman",
|
||||||
"metrics.pages": "Halaman",
|
"metrics.pages": "Halaman",
|
||||||
"metrics.referrers": "Perujuk",
|
"metrics.referrers": "Perujuk",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Pelawat unik",
|
"metrics.unique-visitors": "Pelawat unik",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Lawatan",
|
"metrics.views": "Lawatan",
|
||||||
"metrics.visitors": "Pelawat"
|
"metrics.visitors": "Pelawat"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -4,8 +4,8 @@
|
||||||
"label.add-website": "Legg til nettsted",
|
"label.add-website": "Legg til nettsted",
|
||||||
"label.administrator": "Administrator",
|
"label.administrator": "Administrator",
|
||||||
"label.all": "Alle",
|
"label.all": "Alle",
|
||||||
"label.all-events": "All events",
|
"label.all-events": "Alle hendelser",
|
||||||
"label.all-time": "All time",
|
"label.all-time": "Noensinne",
|
||||||
"label.all-websites": "Alle nettsteder",
|
"label.all-websites": "Alle nettsteder",
|
||||||
"label.back": "Tilbake",
|
"label.back": "Tilbake",
|
||||||
"label.cancel": "Avvis",
|
"label.cancel": "Avvis",
|
||||||
|
|
@ -14,7 +14,7 @@
|
||||||
"label.copy-to-clipboard": "Kopier til utklippstavle",
|
"label.copy-to-clipboard": "Kopier til utklippstavle",
|
||||||
"label.current-password": "Nåværende passord",
|
"label.current-password": "Nåværende passord",
|
||||||
"label.custom-range": "Egendefinert utvalg",
|
"label.custom-range": "Egendefinert utvalg",
|
||||||
"label.dashboard": "Dashboard",
|
"label.dashboard": "Dashbord",
|
||||||
"label.date-range": "Datointervall",
|
"label.date-range": "Datointervall",
|
||||||
"label.default-date-range": "Standard datoperiode",
|
"label.default-date-range": "Standard datoperiode",
|
||||||
"label.delete": "Slett",
|
"label.delete": "Slett",
|
||||||
|
|
@ -28,16 +28,17 @@
|
||||||
"label.enable-share-url": "Aktiver delings-URL",
|
"label.enable-share-url": "Aktiver delings-URL",
|
||||||
"label.invalid": "Ugyldig",
|
"label.invalid": "Ugyldig",
|
||||||
"label.invalid-domain": "Ugyldig domene",
|
"label.invalid-domain": "Ugyldig domene",
|
||||||
"label.language": "Language",
|
"label.language": "Språk",
|
||||||
"label.last-days": "Siste {x} dager",
|
"label.last-days": "Siste {x} dager",
|
||||||
"label.last-hours": "Siste {x} timer",
|
"label.last-hours": "Siste {x} timer",
|
||||||
"label.logged-in-as": "Logget på som {brukernavn}",
|
"label.logged-in-as": "Logget på som {username}",
|
||||||
"label.login": "Logg inn",
|
"label.login": "Logg inn",
|
||||||
"label.logout": "Logg ut",
|
"label.logout": "Logg ut",
|
||||||
"label.more": "Mer",
|
"label.more": "Mer",
|
||||||
"label.name": "Navn",
|
"label.name": "Navn",
|
||||||
"label.new-password": "Nytt passord",
|
"label.new-password": "Nytt passord",
|
||||||
"label.owner": "Owner",
|
"label.none": "None",
|
||||||
|
"label.owner": "Eier",
|
||||||
"label.password": "Passord",
|
"label.password": "Passord",
|
||||||
"label.passwords-dont-match": "Passordene er ikke like",
|
"label.passwords-dont-match": "Passordene er ikke like",
|
||||||
"label.profile": "Profil",
|
"label.profile": "Profil",
|
||||||
|
|
@ -46,7 +47,7 @@
|
||||||
"label.refresh": "Oppdater",
|
"label.refresh": "Oppdater",
|
||||||
"label.required": "Påkrevd",
|
"label.required": "Påkrevd",
|
||||||
"label.reset": "Nullstill",
|
"label.reset": "Nullstill",
|
||||||
"label.reset-website": "Reset statistics",
|
"label.reset-website": "Nullstill statistikk",
|
||||||
"label.save": "Lagre",
|
"label.save": "Lagre",
|
||||||
"label.settings": "Innstillinger",
|
"label.settings": "Innstillinger",
|
||||||
"label.share-url": "Del URL",
|
"label.share-url": "Del URL",
|
||||||
|
|
@ -64,7 +65,7 @@
|
||||||
"label.websites": "Nettsteder",
|
"label.websites": "Nettsteder",
|
||||||
"message.active-users": "{x} {x, plural, one {besøkende} other {besøkende}} nå",
|
"message.active-users": "{x} {x, plural, one {besøkende} other {besøkende}} nå",
|
||||||
"message.confirm-delete": "Er du sikker på at du vil slette {target}?",
|
"message.confirm-delete": "Er du sikker på at du vil slette {target}?",
|
||||||
"message.confirm-reset": "Are your sure you want to reset {target}'s statistics?",
|
"message.confirm-reset": "Er du sikker på at du vil nullstille {target}'s statistikk?",
|
||||||
"message.copied": "Kopiert!",
|
"message.copied": "Kopiert!",
|
||||||
"message.delete-warning": "Alle tilknyttede data slettes også.",
|
"message.delete-warning": "Alle tilknyttede data slettes også.",
|
||||||
"message.failure": "Noe gikk galt.",
|
"message.failure": "Noe gikk galt.",
|
||||||
|
|
@ -78,15 +79,15 @@
|
||||||
"message.no-websites-configured": "Du har ikke satt opp noen nettsteder.",
|
"message.no-websites-configured": "Du har ikke satt opp noen nettsteder.",
|
||||||
"message.page-not-found": "Side ikke funnet.",
|
"message.page-not-found": "Side ikke funnet.",
|
||||||
"message.powered-by": "Drevet av {name}",
|
"message.powered-by": "Drevet av {name}",
|
||||||
"message.reset-warning": "All statistics for this website will be deleted, but your tracking code will remain intact.",
|
"message.reset-warning": "All statistikk for denne nettsiden vil bli slettet, men sporingskoden din vil forbli uberørt.",
|
||||||
"message.save-success": "Lagret!",
|
"message.save-success": "Lagret!",
|
||||||
"message.share-url": "Dette er den offentlige delings-URL-en for {target}.",
|
"message.share-url": "Dette er den offentlige delings-URL-en for {target}.",
|
||||||
"message.toggle-charts": "Toggle charts",
|
"message.toggle-charts": "Veksle grafer",
|
||||||
"message.track-stats": "For å spore statistikk for {target}, plasser følgende kode i {head}-delen av nettstedet ditt.",
|
"message.track-stats": "For å spore statistikk for {target}, plasser følgende kode i {head}-delen av nettstedet ditt.",
|
||||||
"message.type-delete": "Skriv inn {delete} i boksen nedenfor for å bekrefte.",
|
"message.type-delete": "Skriv inn {delete} i boksen nedenfor for å bekrefte.",
|
||||||
"message.type-reset": "Skriv inn {reset} i boksen nedenfor for å bekrefte.",
|
"message.type-reset": "Skriv inn {reset} i boksen nedenfor for å bekrefte.",
|
||||||
"metrics.actions": "Handlinger",
|
"metrics.actions": "Handlinger",
|
||||||
"metrics.average-visit-time": "Gjennomsnittlig besøkelsestid",
|
"metrics.average-visit-time": "Gjennomsnittlig besøkstid",
|
||||||
"metrics.bounce-rate": "Avvisningsfrekvens",
|
"metrics.bounce-rate": "Avvisningsfrekvens",
|
||||||
"metrics.browsers": "Nettlesere",
|
"metrics.browsers": "Nettlesere",
|
||||||
"metrics.countries": "Land",
|
"metrics.countries": "Land",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Enheter",
|
"metrics.devices": "Enheter",
|
||||||
"metrics.events": "Arrangementer",
|
"metrics.events": "Arrangementer",
|
||||||
"metrics.filter.combined": "Kombinert",
|
"metrics.filter.combined": "Kombinert",
|
||||||
"metrics.filter.domain-only": "Bare domene",
|
|
||||||
"metrics.filter.raw": "Rå",
|
"metrics.filter.raw": "Rå",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Språk",
|
||||||
"metrics.operating-systems": "Operativsystemer",
|
"metrics.operating-systems": "Operativsystemer",
|
||||||
"metrics.page-views": "Sidevisninger",
|
"metrics.page-views": "Sidevisninger",
|
||||||
"metrics.pages": "Sider",
|
"metrics.pages": "Sider",
|
||||||
"metrics.referrers": "Referanser",
|
"metrics.referrers": "Referanser",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unike besøkende",
|
"metrics.unique-visitors": "Unike besøkende",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visninger",
|
"metrics.views": "Visninger",
|
||||||
"metrics.visitors": "Besøkende"
|
"metrics.visitors": "Besøkende"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -5,7 +5,7 @@
|
||||||
"label.administrator": "Administrator",
|
"label.administrator": "Administrator",
|
||||||
"label.all": "Alles",
|
"label.all": "Alles",
|
||||||
"label.all-events": "Alle gebeurtenissen",
|
"label.all-events": "Alle gebeurtenissen",
|
||||||
"label.all-time": "All time",
|
"label.all-time": "Onbeperkt",
|
||||||
"label.all-websites": "Alle websites",
|
"label.all-websites": "Alle websites",
|
||||||
"label.back": "Terug",
|
"label.back": "Terug",
|
||||||
"label.cancel": "Annuleren",
|
"label.cancel": "Annuleren",
|
||||||
|
|
@ -28,7 +28,7 @@
|
||||||
"label.enable-share-url": "Sta delen via openbare URL toe",
|
"label.enable-share-url": "Sta delen via openbare URL toe",
|
||||||
"label.invalid": "Ongeldig",
|
"label.invalid": "Ongeldig",
|
||||||
"label.invalid-domain": "Ongeldig domein",
|
"label.invalid-domain": "Ongeldig domein",
|
||||||
"label.language": "Language",
|
"label.language": "Taal",
|
||||||
"label.last-days": "Laatste {x} dagen",
|
"label.last-days": "Laatste {x} dagen",
|
||||||
"label.last-hours": "Laatste {x} uur",
|
"label.last-hours": "Laatste {x} uur",
|
||||||
"label.logged-in-as": "Ingelogd als {username}",
|
"label.logged-in-as": "Ingelogd als {username}",
|
||||||
|
|
@ -37,7 +37,8 @@
|
||||||
"label.more": "Toon meer",
|
"label.more": "Toon meer",
|
||||||
"label.name": "Naam",
|
"label.name": "Naam",
|
||||||
"label.new-password": "Nieuw wachtwoord",
|
"label.new-password": "Nieuw wachtwoord",
|
||||||
"label.owner": "Owner",
|
"label.none": "Geen",
|
||||||
|
"label.owner": "Eigenaar",
|
||||||
"label.password": "Wachtwoord",
|
"label.password": "Wachtwoord",
|
||||||
"label.passwords-dont-match": "Wachtwoorden komen niet overeen",
|
"label.passwords-dont-match": "Wachtwoorden komen niet overeen",
|
||||||
"label.profile": "Profiel",
|
"label.profile": "Profiel",
|
||||||
|
|
@ -46,12 +47,12 @@
|
||||||
"label.refresh": "Vernieuwen",
|
"label.refresh": "Vernieuwen",
|
||||||
"label.required": "Verplicht",
|
"label.required": "Verplicht",
|
||||||
"label.reset": "Resetten",
|
"label.reset": "Resetten",
|
||||||
"label.reset-website": "Reset statistics",
|
"label.reset-website": "Statistieken opnieuw instellen",
|
||||||
"label.save": "Opslaan",
|
"label.save": "Opslaan",
|
||||||
"label.settings": "Instellingen",
|
"label.settings": "Instellingen",
|
||||||
"label.share-url": "URL delen",
|
"label.share-url": "URL delen",
|
||||||
"label.single-day": "Enkele dag",
|
"label.single-day": "Enkele dag",
|
||||||
"label.theme": "Theme",
|
"label.theme": "Thema",
|
||||||
"label.this-month": "Deze maand",
|
"label.this-month": "Deze maand",
|
||||||
"label.this-week": "Deze week",
|
"label.this-week": "Deze week",
|
||||||
"label.this-year": "Dit jaar",
|
"label.this-year": "Dit jaar",
|
||||||
|
|
@ -64,7 +65,7 @@
|
||||||
"label.websites": "Websites",
|
"label.websites": "Websites",
|
||||||
"message.active-users": "{x} actieve {x, plural, one {bezoeker} other {bezoekers}}",
|
"message.active-users": "{x} actieve {x, plural, one {bezoeker} other {bezoekers}}",
|
||||||
"message.confirm-delete": "Weet je zeker dat je {target} wilt verwijderen?",
|
"message.confirm-delete": "Weet je zeker dat je {target} wilt verwijderen?",
|
||||||
"message.confirm-reset": "Are your sure you want to reset {target}'s statistics?",
|
"message.confirm-reset": "Weet je zeker dat je de statistieken van {target} opnieuw wilt instellen?",
|
||||||
"message.copied": "Gekopiëerd!",
|
"message.copied": "Gekopiëerd!",
|
||||||
"message.delete-warning": "Alle verwante gegezens zullen ook verwijderd worden.",
|
"message.delete-warning": "Alle verwante gegezens zullen ook verwijderd worden.",
|
||||||
"message.failure": "Er is iets misgegaan.",
|
"message.failure": "Er is iets misgegaan.",
|
||||||
|
|
@ -78,10 +79,10 @@
|
||||||
"message.no-websites-configured": "Je hebt geen websites ingesteld.",
|
"message.no-websites-configured": "Je hebt geen websites ingesteld.",
|
||||||
"message.page-not-found": "Pagina niet gevonden.",
|
"message.page-not-found": "Pagina niet gevonden.",
|
||||||
"message.powered-by": "mogelijk gemaakt door {name}",
|
"message.powered-by": "mogelijk gemaakt door {name}",
|
||||||
"message.reset-warning": "All statistics for this website will be deleted, but your tracking code will remain intact.",
|
"message.reset-warning": "Alle bijhorende statistieken van deze website worden verwijderd, maar jouw volgcode blijft gelden.",
|
||||||
"message.save-success": "Opslaan succesvol.",
|
"message.save-success": "Opslaan succesvol.",
|
||||||
"message.share-url": "Met deze URL kan {target} openbaar gedeeld worden.",
|
"message.share-url": "Met deze URL kan {target} openbaar gedeeld worden.",
|
||||||
"message.toggle-charts": "Toggle charts",
|
"message.toggle-charts": "Grafieken tonen/verbergen",
|
||||||
"message.track-stats": "Om statistieken voor {target} bij te houden, plaats je de volgende code in het {head} gedeelte van je website.",
|
"message.track-stats": "Om statistieken voor {target} bij te houden, plaats je de volgende code in het {head} gedeelte van je website.",
|
||||||
"message.type-delete": "Type {delete} in onderstaande veld om dit te bevestigen.",
|
"message.type-delete": "Type {delete} in onderstaande veld om dit te bevestigen.",
|
||||||
"message.type-reset": "Type {reset} in onderstaande veld om dit te bevestigen.",
|
"message.type-reset": "Type {reset} in onderstaande veld om dit te bevestigen.",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Apparaten",
|
"metrics.devices": "Apparaten",
|
||||||
"metrics.events": "Gebeurtenissen",
|
"metrics.events": "Gebeurtenissen",
|
||||||
"metrics.filter.combined": "Gecombineerd",
|
"metrics.filter.combined": "Gecombineerd",
|
||||||
"metrics.filter.domain-only": "Alleen domein",
|
|
||||||
"metrics.filter.raw": "Ruw",
|
"metrics.filter.raw": "Ruw",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Besturingssysteem",
|
"metrics.operating-systems": "Besturingssystemen",
|
||||||
"metrics.page-views": "Paginaweergaven",
|
"metrics.page-views": "Paginaweergaven",
|
||||||
"metrics.pages": "Pagina's",
|
"metrics.pages": "Pagina's",
|
||||||
"metrics.referrers": "Verwijzers",
|
"metrics.referrers": "Verwijzers",
|
||||||
|
"metrics.screens": "Schermen",
|
||||||
"metrics.unique-visitors": "Unieke bezoekers",
|
"metrics.unique-visitors": "Unieke bezoekers",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Weergaven",
|
"metrics.views": "Weergaven",
|
||||||
"metrics.visitors": "Bezoekers"
|
"metrics.visitors": "Bezoekers"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Więcej",
|
"label.more": "Więcej",
|
||||||
"label.name": "Nazwa",
|
"label.name": "Nazwa",
|
||||||
"label.new-password": "Nowe hasło",
|
"label.new-password": "Nowe hasło",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Właściciel",
|
"label.owner": "Właściciel",
|
||||||
"label.password": "Hasło",
|
"label.password": "Hasło",
|
||||||
"label.passwords-dont-match": "Hasła się nie zgadzają",
|
"label.passwords-dont-match": "Hasła się nie zgadzają",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Urządzenia",
|
"metrics.devices": "Urządzenia",
|
||||||
"metrics.events": "Zdarzenia",
|
"metrics.events": "Zdarzenia",
|
||||||
"metrics.filter.combined": "Połączone",
|
"metrics.filter.combined": "Połączone",
|
||||||
"metrics.filter.domain-only": "Tylko domena",
|
|
||||||
"metrics.filter.raw": "Surowe dane",
|
"metrics.filter.raw": "Surowe dane",
|
||||||
"metrics.languages": "Języki",
|
"metrics.languages": "Języki",
|
||||||
"metrics.operating-systems": "System operacyjny",
|
"metrics.operating-systems": "System operacyjny",
|
||||||
"metrics.page-views": "Wyświetlenia strony",
|
"metrics.page-views": "Wyświetlenia strony",
|
||||||
"metrics.pages": "Strony",
|
"metrics.pages": "Strony",
|
||||||
"metrics.referrers": "Źródła odsyłające",
|
"metrics.referrers": "Źródła odsyłające",
|
||||||
|
"metrics.screens": "Ekrany",
|
||||||
"metrics.unique-visitors": "Unikalni odwiedzający",
|
"metrics.unique-visitors": "Unikalni odwiedzający",
|
||||||
|
"metrics.utm": "Kampanie UTM",
|
||||||
|
"metrics.utm_campaign": "Kampania",
|
||||||
|
"metrics.utm_content": "Treść",
|
||||||
|
"metrics.utm_medium": "Medium",
|
||||||
|
"metrics.utm_source": "Źródło",
|
||||||
|
"metrics.utm_term": "Słowa kluczowe",
|
||||||
"metrics.views": "Wyświetlenia",
|
"metrics.views": "Wyświetlenia",
|
||||||
"metrics.visitors": "Odwiedzający"
|
"metrics.visitors": "Odwiedzający"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Mais",
|
"label.more": "Mais",
|
||||||
"label.name": "Nome",
|
"label.name": "Nome",
|
||||||
"label.new-password": "Nova senha",
|
"label.new-password": "Nova senha",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Proprietário",
|
"label.owner": "Proprietário",
|
||||||
"label.password": "Senha",
|
"label.password": "Senha",
|
||||||
"label.passwords-dont-match": "As senhas não correspondem",
|
"label.passwords-dont-match": "As senhas não correspondem",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispositivos",
|
"metrics.devices": "Dispositivos",
|
||||||
"metrics.events": "Eventos",
|
"metrics.events": "Eventos",
|
||||||
"metrics.filter.combined": "Combinado",
|
"metrics.filter.combined": "Combinado",
|
||||||
"metrics.filter.domain-only": "Apenas domínio",
|
|
||||||
"metrics.filter.raw": "Dados brutos",
|
"metrics.filter.raw": "Dados brutos",
|
||||||
"metrics.languages": "Idiomas",
|
"metrics.languages": "Idiomas",
|
||||||
"metrics.operating-systems": "Sistemas operacionais",
|
"metrics.operating-systems": "Sistemas operacionais",
|
||||||
"metrics.page-views": "Visualizações de página",
|
"metrics.page-views": "Visualizações de página",
|
||||||
"metrics.pages": "Páginas",
|
"metrics.pages": "Páginas",
|
||||||
"metrics.referrers": "Referências",
|
"metrics.referrers": "Referências",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Visitantes únicos",
|
"metrics.unique-visitors": "Visitantes únicos",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visualizações",
|
"metrics.views": "Visualizações",
|
||||||
"metrics.visitors": "Visitantes"
|
"metrics.visitors": "Visitantes"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Mais",
|
"label.more": "Mais",
|
||||||
"label.name": "Nome",
|
"label.name": "Nome",
|
||||||
"label.new-password": "Nova senha",
|
"label.new-password": "Nova senha",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Proprietário",
|
"label.owner": "Proprietário",
|
||||||
"label.password": "Senha",
|
"label.password": "Senha",
|
||||||
"label.passwords-dont-match": "As senhas não coincidem",
|
"label.passwords-dont-match": "As senhas não coincidem",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispositivos",
|
"metrics.devices": "Dispositivos",
|
||||||
"metrics.events": "Eventos",
|
"metrics.events": "Eventos",
|
||||||
"metrics.filter.combined": "Combinado",
|
"metrics.filter.combined": "Combinado",
|
||||||
"metrics.filter.domain-only": "Apenas domínio",
|
|
||||||
"metrics.filter.raw": "Dados brutos",
|
"metrics.filter.raw": "Dados brutos",
|
||||||
"metrics.languages": "Línguas",
|
"metrics.languages": "Línguas",
|
||||||
"metrics.operating-systems": "Sistemas operativos",
|
"metrics.operating-systems": "Sistemas operativos",
|
||||||
"metrics.page-views": "Visualizações da página",
|
"metrics.page-views": "Visualizações da página",
|
||||||
"metrics.pages": "Páginas",
|
"metrics.pages": "Páginas",
|
||||||
"metrics.referrers": "Referenciadores",
|
"metrics.referrers": "Referenciadores",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Visitantes únicos",
|
"metrics.unique-visitors": "Visitantes únicos",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visualizações",
|
"metrics.views": "Visualizações",
|
||||||
"metrics.visitors": "Visitantes"
|
"metrics.visitors": "Visitantes"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Mai mult",
|
"label.more": "Mai mult",
|
||||||
"label.name": "Nume",
|
"label.name": "Nume",
|
||||||
"label.new-password": "Parola nouă",
|
"label.new-password": "Parola nouă",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Parolă",
|
"label.password": "Parolă",
|
||||||
"label.passwords-dont-match": "Parolele nu se potrivesc",
|
"label.passwords-dont-match": "Parolele nu se potrivesc",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Dispozitive",
|
"metrics.devices": "Dispozitive",
|
||||||
"metrics.events": "Evenimente",
|
"metrics.events": "Evenimente",
|
||||||
"metrics.filter.combined": "Combinat",
|
"metrics.filter.combined": "Combinat",
|
||||||
"metrics.filter.domain-only": "Numai domeniu",
|
|
||||||
"metrics.filter.raw": "Brut",
|
"metrics.filter.raw": "Brut",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Sisteme de operare",
|
"metrics.operating-systems": "Sisteme de operare",
|
||||||
"metrics.page-views": "Vizualizări de pagină",
|
"metrics.page-views": "Vizualizări de pagină",
|
||||||
"metrics.pages": "Pagini",
|
"metrics.pages": "Pagini",
|
||||||
"metrics.referrers": "Site-uri de proveniență",
|
"metrics.referrers": "Site-uri de proveniență",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Vizitatori unici",
|
"metrics.unique-visitors": "Vizitatori unici",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Vizualizări",
|
"metrics.views": "Vizualizări",
|
||||||
"metrics.visitors": "Vizitatori"
|
"metrics.visitors": "Vizitatori"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Больше",
|
"label.more": "Больше",
|
||||||
"label.name": "Имя",
|
"label.name": "Имя",
|
||||||
"label.new-password": "Новый пароль",
|
"label.new-password": "Новый пароль",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Владелец",
|
"label.owner": "Владелец",
|
||||||
"label.password": "Пароль",
|
"label.password": "Пароль",
|
||||||
"label.passwords-dont-match": "Пароли не совпадают",
|
"label.passwords-dont-match": "Пароли не совпадают",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Устройства",
|
"metrics.devices": "Устройства",
|
||||||
"metrics.events": "События",
|
"metrics.events": "События",
|
||||||
"metrics.filter.combined": "Объединенные",
|
"metrics.filter.combined": "Объединенные",
|
||||||
"metrics.filter.domain-only": "Только домен",
|
|
||||||
"metrics.filter.raw": "Сырые данные",
|
"metrics.filter.raw": "Сырые данные",
|
||||||
"metrics.languages": "Языки",
|
"metrics.languages": "Языки",
|
||||||
"metrics.operating-systems": "Операционные системы",
|
"metrics.operating-systems": "Операционные системы",
|
||||||
"metrics.page-views": "Просмотры страниц",
|
"metrics.page-views": "Просмотры страниц",
|
||||||
"metrics.pages": "Страницы",
|
"metrics.pages": "Страницы",
|
||||||
"metrics.referrers": "Источники",
|
"metrics.referrers": "Источники",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Уникальные посетители",
|
"metrics.unique-visitors": "Уникальные посетители",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Просмотры",
|
"metrics.views": "Просмотры",
|
||||||
"metrics.visitors": "Посетители"
|
"metrics.visitors": "Посетители"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Viac",
|
"label.more": "Viac",
|
||||||
"label.name": "Meno",
|
"label.name": "Meno",
|
||||||
"label.new-password": "Nové heslo",
|
"label.new-password": "Nové heslo",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Heslo",
|
"label.password": "Heslo",
|
||||||
"label.passwords-dont-match": "Hesla se nezhodujú",
|
"label.passwords-dont-match": "Hesla se nezhodujú",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Zariadenie",
|
"metrics.devices": "Zariadenie",
|
||||||
"metrics.events": "Udalosti",
|
"metrics.events": "Udalosti",
|
||||||
"metrics.filter.combined": "Kombinácie",
|
"metrics.filter.combined": "Kombinácie",
|
||||||
"metrics.filter.domain-only": "Domény",
|
|
||||||
"metrics.filter.raw": "Nezpracované",
|
"metrics.filter.raw": "Nezpracované",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Operačný systém",
|
"metrics.operating-systems": "Operačný systém",
|
||||||
"metrics.page-views": "Zobrazenie stánok",
|
"metrics.page-views": "Zobrazenie stánok",
|
||||||
"metrics.pages": "Stránky",
|
"metrics.pages": "Stránky",
|
||||||
"metrics.referrers": "Odkazy",
|
"metrics.referrers": "Odkazy",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Jedinečné návštevy",
|
"metrics.unique-visitors": "Jedinečné návštevy",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Zobrazení",
|
"metrics.views": "Zobrazení",
|
||||||
"metrics.visitors": "Návštevy"
|
"metrics.visitors": "Návštevy"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Več",
|
"label.more": "Več",
|
||||||
"label.name": "Ime",
|
"label.name": "Ime",
|
||||||
"label.new-password": "Novo geslo",
|
"label.new-password": "Novo geslo",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Geslo",
|
"label.password": "Geslo",
|
||||||
"label.passwords-dont-match": "Gesli se ne ujemata",
|
"label.passwords-dont-match": "Gesli se ne ujemata",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Naprave",
|
"metrics.devices": "Naprave",
|
||||||
"metrics.events": "Dogodki",
|
"metrics.events": "Dogodki",
|
||||||
"metrics.filter.combined": "Skupno",
|
"metrics.filter.combined": "Skupno",
|
||||||
"metrics.filter.domain-only": "Samo domena",
|
|
||||||
"metrics.filter.raw": "Neobdelane meritve",
|
"metrics.filter.raw": "Neobdelane meritve",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "Operacijski sistemi",
|
"metrics.operating-systems": "Operacijski sistemi",
|
||||||
"metrics.page-views": "Ogledi strani",
|
"metrics.page-views": "Ogledi strani",
|
||||||
"metrics.pages": "Strani",
|
"metrics.pages": "Strani",
|
||||||
"metrics.referrers": "Viri",
|
"metrics.referrers": "Viri",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unikatni obiskovalci",
|
"metrics.unique-visitors": "Unikatni obiskovalci",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Ogledi",
|
"metrics.views": "Ogledi",
|
||||||
"metrics.visitors": "Obiskovalci"
|
"metrics.visitors": "Obiskovalci"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Mer",
|
"label.more": "Mer",
|
||||||
"label.name": "Namn",
|
"label.name": "Namn",
|
||||||
"label.new-password": "Nytt lösenord",
|
"label.new-password": "Nytt lösenord",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Ägare",
|
"label.owner": "Ägare",
|
||||||
"label.password": "Lösenord",
|
"label.password": "Lösenord",
|
||||||
"label.passwords-dont-match": "Lösenorden är inte samma",
|
"label.passwords-dont-match": "Lösenorden är inte samma",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Enheter",
|
"metrics.devices": "Enheter",
|
||||||
"metrics.events": "Händelser",
|
"metrics.events": "Händelser",
|
||||||
"metrics.filter.combined": "Kombinerade",
|
"metrics.filter.combined": "Kombinerade",
|
||||||
"metrics.filter.domain-only": "Endast domän",
|
|
||||||
"metrics.filter.raw": "Rådata",
|
"metrics.filter.raw": "Rådata",
|
||||||
"metrics.languages": "Språk",
|
"metrics.languages": "Språk",
|
||||||
"metrics.operating-systems": "Operativsystem",
|
"metrics.operating-systems": "Operativsystem",
|
||||||
"metrics.page-views": "Sidvisningar",
|
"metrics.page-views": "Sidvisningar",
|
||||||
"metrics.pages": "Sidor",
|
"metrics.pages": "Sidor",
|
||||||
"metrics.referrers": "Hänvisare",
|
"metrics.referrers": "Hänvisare",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Unika besökare",
|
"metrics.unique-visitors": "Unika besökare",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Visningar",
|
"metrics.views": "Visningar",
|
||||||
"metrics.visitors": "Besökare"
|
"metrics.visitors": "Besökare"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "மேலும்",
|
"label.more": "மேலும்",
|
||||||
"label.name": "பெயர்",
|
"label.name": "பெயர்",
|
||||||
"label.new-password": "புதிய கடவுச்சொல்",
|
"label.new-password": "புதிய கடவுச்சொல்",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "கடவுச்சொல்",
|
"label.password": "கடவுச்சொல்",
|
||||||
"label.passwords-dont-match": "இருக்கடவுச்சொல் பொருந்தவில்லை",
|
"label.passwords-dont-match": "இருக்கடவுச்சொல் பொருந்தவில்லை",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "சாதனங்கள்",
|
"metrics.devices": "சாதனங்கள்",
|
||||||
"metrics.events": "நிகழ்வுகள்",
|
"metrics.events": "நிகழ்வுகள்",
|
||||||
"metrics.filter.combined": "ஒருங்கிணைந்த",
|
"metrics.filter.combined": "ஒருங்கிணைந்த",
|
||||||
"metrics.filter.domain-only": "கள முகவரி மட்டும்",
|
|
||||||
"metrics.filter.raw": "மூல",
|
"metrics.filter.raw": "மூல",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "இயக்க முறைமைகள்",
|
"metrics.operating-systems": "இயக்க முறைமைகள்",
|
||||||
"metrics.page-views": "பக்க காட்சிகள்",
|
"metrics.page-views": "பக்க காட்சிகள்",
|
||||||
"metrics.pages": "பக்கங்கள்",
|
"metrics.pages": "பக்கங்கள்",
|
||||||
"metrics.referrers": "குறிப்பிடுவோர்",
|
"metrics.referrers": "குறிப்பிடுவோர்",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "தனிப்பட்ட பார்வையாளர்கள்",
|
"metrics.unique-visitors": "தனிப்பட்ட பார்வையாளர்கள்",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "பார்வைகள்",
|
"metrics.views": "பார்வைகள்",
|
||||||
"metrics.visitors": "பார்வையாளர்கள்"
|
"metrics.visitors": "பார்வையாளர்கள்"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Detaylı göster",
|
"label.more": "Detaylı göster",
|
||||||
"label.name": "İsim",
|
"label.name": "İsim",
|
||||||
"label.new-password": "Yeni parola",
|
"label.new-password": "Yeni parola",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "Parola",
|
"label.password": "Parola",
|
||||||
"label.passwords-dont-match": "Parolalar uyuşmuyor",
|
"label.passwords-dont-match": "Parolalar uyuşmuyor",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Cihazlar",
|
"metrics.devices": "Cihazlar",
|
||||||
"metrics.events": "Olaylar",
|
"metrics.events": "Olaylar",
|
||||||
"metrics.filter.combined": "Birleşik",
|
"metrics.filter.combined": "Birleşik",
|
||||||
"metrics.filter.domain-only": "Yalnızca alan adı",
|
|
||||||
"metrics.filter.raw": "Ham",
|
"metrics.filter.raw": "Ham",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "İşletim sistemi",
|
"metrics.operating-systems": "İşletim sistemi",
|
||||||
"metrics.page-views": "Sayfa görünümü",
|
"metrics.page-views": "Sayfa görünümü",
|
||||||
"metrics.pages": "Sayfalar",
|
"metrics.pages": "Sayfalar",
|
||||||
"metrics.referrers": "Yönlendirenler",
|
"metrics.referrers": "Yönlendirenler",
|
||||||
|
"metrics.screens": "Ekranlar",
|
||||||
"metrics.unique-visitors": "Tekil kullanıcı",
|
"metrics.unique-visitors": "Tekil kullanıcı",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Görüntüleme",
|
"metrics.views": "Görüntüleme",
|
||||||
"metrics.visitors": "Ziyaretçi"
|
"metrics.visitors": "Ziyaretçi"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Більше",
|
"label.more": "Більше",
|
||||||
"label.name": "Ім'я",
|
"label.name": "Ім'я",
|
||||||
"label.new-password": "Новий пароль",
|
"label.new-password": "Новий пароль",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Власник",
|
"label.owner": "Власник",
|
||||||
"label.password": "Пароль",
|
"label.password": "Пароль",
|
||||||
"label.passwords-dont-match": "Паролі не співпадають",
|
"label.passwords-dont-match": "Паролі не співпадають",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Пристрої",
|
"metrics.devices": "Пристрої",
|
||||||
"metrics.events": "Події",
|
"metrics.events": "Події",
|
||||||
"metrics.filter.combined": "Об'єднані",
|
"metrics.filter.combined": "Об'єднані",
|
||||||
"metrics.filter.domain-only": "Лише домен",
|
|
||||||
"metrics.filter.raw": "Сирі дані",
|
"metrics.filter.raw": "Сирі дані",
|
||||||
"metrics.languages": "Мови",
|
"metrics.languages": "Мови",
|
||||||
"metrics.operating-systems": "Операційні системи",
|
"metrics.operating-systems": "Операційні системи",
|
||||||
"metrics.page-views": "Перегляди сторінок",
|
"metrics.page-views": "Перегляди сторінок",
|
||||||
"metrics.pages": "Сторінки",
|
"metrics.pages": "Сторінки",
|
||||||
"metrics.referrers": "Джерела",
|
"metrics.referrers": "Джерела",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Унікальні відвідувачі",
|
"metrics.unique-visitors": "Унікальні відвідувачі",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Перегляди",
|
"metrics.views": "Перегляди",
|
||||||
"metrics.visitors": "Відвідувачі"
|
"metrics.visitors": "Відвідувачі"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "مزید",
|
"label.more": "مزید",
|
||||||
"label.name": "نام",
|
"label.name": "نام",
|
||||||
"label.new-password": "نیا پاس ورڈ",
|
"label.new-password": "نیا پاس ورڈ",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "مالک",
|
"label.owner": "مالک",
|
||||||
"label.password": "پاس ورڈ",
|
"label.password": "پاس ورڈ",
|
||||||
"label.passwords-dont-match": "پاس ورڈز مماثل نہیں ہیں",
|
"label.passwords-dont-match": "پاس ورڈز مماثل نہیں ہیں",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "آلات",
|
"metrics.devices": "آلات",
|
||||||
"metrics.events": "واقعات",
|
"metrics.events": "واقعات",
|
||||||
"metrics.filter.combined": "مشترکہ",
|
"metrics.filter.combined": "مشترکہ",
|
||||||
"metrics.filter.domain-only": "صرف ڈومین",
|
|
||||||
"metrics.filter.raw": "خام",
|
"metrics.filter.raw": "خام",
|
||||||
"metrics.languages": "زبانیں",
|
"metrics.languages": "زبانیں",
|
||||||
"metrics.operating-systems": "آپریٹنگ سسٹمز",
|
"metrics.operating-systems": "آپریٹنگ سسٹمز",
|
||||||
"metrics.page-views": "صفحہ کے نظارے",
|
"metrics.page-views": "صفحہ کے نظارے",
|
||||||
"metrics.pages": "صفحات",
|
"metrics.pages": "صفحات",
|
||||||
"metrics.referrers": "بھیجنے والے",
|
"metrics.referrers": "بھیجنے والے",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "منفرد زائرین",
|
"metrics.unique-visitors": "منفرد زائرین",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "مناظر",
|
"metrics.views": "مناظر",
|
||||||
"metrics.visitors": "زائرین"
|
"metrics.visitors": "زائرین"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "Thêm",
|
"label.more": "Thêm",
|
||||||
"label.name": "Tên",
|
"label.name": "Tên",
|
||||||
"label.new-password": "Mật khẩu mới",
|
"label.new-password": "Mật khẩu mới",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Chủ sở hữu",
|
"label.owner": "Chủ sở hữu",
|
||||||
"label.password": "Mật khẩu",
|
"label.password": "Mật khẩu",
|
||||||
"label.passwords-dont-match": "Mật khẩu không đồng nhất",
|
"label.passwords-dont-match": "Mật khẩu không đồng nhất",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "Thiết bị",
|
"metrics.devices": "Thiết bị",
|
||||||
"metrics.events": "Sự kiện",
|
"metrics.events": "Sự kiện",
|
||||||
"metrics.filter.combined": "Kết hợp",
|
"metrics.filter.combined": "Kết hợp",
|
||||||
"metrics.filter.domain-only": "Chỉ tên miền",
|
|
||||||
"metrics.filter.raw": "Gốc",
|
"metrics.filter.raw": "Gốc",
|
||||||
"metrics.languages": "Ngôn ngữ",
|
"metrics.languages": "Ngôn ngữ",
|
||||||
"metrics.operating-systems": "Hệ điều hành",
|
"metrics.operating-systems": "Hệ điều hành",
|
||||||
"metrics.page-views": "Lượt xem",
|
"metrics.page-views": "Lượt xem",
|
||||||
"metrics.pages": "Trang",
|
"metrics.pages": "Trang",
|
||||||
"metrics.referrers": "Liên kết giới thiệu",
|
"metrics.referrers": "Liên kết giới thiệu",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "Khách truy cập một lần",
|
"metrics.unique-visitors": "Khách truy cập một lần",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "Xem",
|
"metrics.views": "Xem",
|
||||||
"metrics.visitors": "Khách"
|
"metrics.visitors": "Khách"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "更多",
|
"label.more": "更多",
|
||||||
"label.name": "名字",
|
"label.name": "名字",
|
||||||
"label.new-password": "新密码",
|
"label.new-password": "新密码",
|
||||||
|
"label.none": "无",
|
||||||
"label.owner": "所有者",
|
"label.owner": "所有者",
|
||||||
"label.password": "密码",
|
"label.password": "密码",
|
||||||
"label.passwords-dont-match": "密码不一致",
|
"label.passwords-dont-match": "密码不一致",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "设备",
|
"metrics.devices": "设备",
|
||||||
"metrics.events": "行为类别",
|
"metrics.events": "行为类别",
|
||||||
"metrics.filter.combined": "总和",
|
"metrics.filter.combined": "总和",
|
||||||
"metrics.filter.domain-only": "只看域名",
|
|
||||||
"metrics.filter.raw": "原始",
|
"metrics.filter.raw": "原始",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "语言",
|
||||||
"metrics.operating-systems": "操作系统",
|
"metrics.operating-systems": "操作系统",
|
||||||
"metrics.page-views": "页面浏览量",
|
"metrics.page-views": "页面浏览量",
|
||||||
"metrics.pages": "网页",
|
"metrics.pages": "网页",
|
||||||
"metrics.referrers": "来源域名",
|
"metrics.referrers": "来源域名",
|
||||||
|
"metrics.screens": "屏幕尺寸",
|
||||||
"metrics.unique-visitors": "独立访客",
|
"metrics.unique-visitors": "独立访客",
|
||||||
|
"metrics.utm": "流量渠道标记(UTM)",
|
||||||
|
"metrics.utm_campaign": "流量标识",
|
||||||
|
"metrics.utm_content": "搜索关键字",
|
||||||
|
"metrics.utm_medium": "流量来源类型",
|
||||||
|
"metrics.utm_source": "流量来源",
|
||||||
|
"metrics.utm_term": "流量搜索关键字",
|
||||||
"metrics.views": "浏览量",
|
"metrics.views": "浏览量",
|
||||||
"metrics.visitors": "访客"
|
"metrics.visitors": "访客"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@
|
||||||
"label.more": "更多",
|
"label.more": "更多",
|
||||||
"label.name": "名字",
|
"label.name": "名字",
|
||||||
"label.new-password": "新密碼",
|
"label.new-password": "新密碼",
|
||||||
|
"label.none": "None",
|
||||||
"label.owner": "Owner",
|
"label.owner": "Owner",
|
||||||
"label.password": "密碼",
|
"label.password": "密碼",
|
||||||
"label.passwords-dont-match": "密碼不一致",
|
"label.passwords-dont-match": "密碼不一致",
|
||||||
|
|
@ -97,14 +98,20 @@
|
||||||
"metrics.devices": "裝置",
|
"metrics.devices": "裝置",
|
||||||
"metrics.events": "行為類別",
|
"metrics.events": "行為類別",
|
||||||
"metrics.filter.combined": "總和",
|
"metrics.filter.combined": "總和",
|
||||||
"metrics.filter.domain-only": "僅域名",
|
|
||||||
"metrics.filter.raw": "原始",
|
"metrics.filter.raw": "原始",
|
||||||
"metrics.languages": "Languages",
|
"metrics.languages": "Languages",
|
||||||
"metrics.operating-systems": "操作系统",
|
"metrics.operating-systems": "操作系统",
|
||||||
"metrics.page-views": "網頁流量",
|
"metrics.page-views": "網頁流量",
|
||||||
"metrics.pages": "網頁",
|
"metrics.pages": "網頁",
|
||||||
"metrics.referrers": "指入域名",
|
"metrics.referrers": "指入域名",
|
||||||
|
"metrics.screens": "Screens",
|
||||||
"metrics.unique-visitors": "獨立訪客",
|
"metrics.unique-visitors": "獨立訪客",
|
||||||
|
"metrics.utm": "UTM",
|
||||||
|
"metrics.utm_campaign": "UTM Campaign",
|
||||||
|
"metrics.utm_content": "UTM Content",
|
||||||
|
"metrics.utm_medium": "UTM Medium",
|
||||||
|
"metrics.utm_source": "UTM Source",
|
||||||
|
"metrics.utm_term": "UTM Term",
|
||||||
"metrics.views": "頁面流量",
|
"metrics.views": "頁面流量",
|
||||||
"metrics.visitors": "獨立訪客"
|
"metrics.visitors": "獨立訪客"
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
import { parseSecureToken, parseToken } from './crypto';
|
import { parseSecureToken, parseToken } from './crypto';
|
||||||
import { SHARE_TOKEN_HEADER } from './constants';
|
import { SHARE_TOKEN_HEADER } from './constants';
|
||||||
import { getWebsiteById } from './queries';
|
import { getWebsiteById } from 'queries';
|
||||||
|
|
||||||
export async function getAuthToken(req) {
|
export async function getAuthToken(req) {
|
||||||
try {
|
try {
|
||||||
|
|
|
||||||
|
|
@ -7,7 +7,8 @@ export const DASHBOARD_CONFIG = 'umami.dashboard';
|
||||||
export const VERSION_CHECK = 'umami.version-check';
|
export const VERSION_CHECK = 'umami.version-check';
|
||||||
export const SHARE_TOKEN_HEADER = 'x-umami-share-token';
|
export const SHARE_TOKEN_HEADER = 'x-umami-share-token';
|
||||||
export const HOMEPAGE_URL = 'https://umami.is';
|
export const HOMEPAGE_URL = 'https://umami.is';
|
||||||
export const VERSION_URL = 'https://github.com/mikecao/umami/releases';
|
export const REPO_URL = 'https://github.com/umami-software/umami';
|
||||||
|
export const UPDATES_URL = 'https://api.umami.is/v1/updates';
|
||||||
|
|
||||||
export const DEFAULT_LOCALE = 'en-US';
|
export const DEFAULT_LOCALE = 'en-US';
|
||||||
export const DEFAULT_THEME = 'light';
|
export const DEFAULT_THEME = 'light';
|
||||||
|
|
@ -64,8 +65,10 @@ export const EVENT_COLORS = [
|
||||||
'#ffec16',
|
'#ffec16',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export const RELATIONAL = 'relational';
|
||||||
export const POSTGRESQL = 'postgresql';
|
export const POSTGRESQL = 'postgresql';
|
||||||
export const MYSQL = 'mysql';
|
export const MYSQL = 'mysql';
|
||||||
|
export const CLICKHOUSE = 'clickhouse';
|
||||||
|
|
||||||
export const MYSQL_DATE_FORMATS = {
|
export const MYSQL_DATE_FORMATS = {
|
||||||
minute: '%Y-%m-%d %H:%i:00',
|
minute: '%Y-%m-%d %H:%i:00',
|
||||||
|
|
@ -83,6 +86,16 @@ export const POSTGRESQL_DATE_FORMATS = {
|
||||||
year: 'YYYY-01-01',
|
year: 'YYYY-01-01',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const CLICKHOUSE_DATE_FORMATS = {
|
||||||
|
minute: '%Y-%m-%d %H:%M:00',
|
||||||
|
hour: '%Y-%m-%d %H:00:00',
|
||||||
|
day: '%Y-%m-%d',
|
||||||
|
month: '%Y-%m-01',
|
||||||
|
year: '%Y-01-01',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const FILTER_IGNORED = Symbol.for('filter-ignored');
|
||||||
|
|
||||||
export const DOMAIN_REGEX =
|
export const DOMAIN_REGEX =
|
||||||
/^(localhost(:[1-9]\d{0,4})?|((?=[a-z0-9-]{1,63}\.)(xn--)?[a-z0-9]+(-[a-z0-9]+)*\.)+[a-z]{2,63})$/;
|
/^(localhost(:[1-9]\d{0,4})?|((?=[a-z0-9-]{1,63}\.)(xn--)?[a-z0-9]+(-[a-z0-9]+)*\.)+[a-z]{2,63})$/;
|
||||||
|
|
||||||
|
|
|
||||||
277
lib/db.js
277
lib/db.js
|
|
@ -1,5 +1,21 @@
|
||||||
import { PrismaClient } from '@prisma/client';
|
import { PrismaClient } from '@prisma/client';
|
||||||
|
import { ClickHouse } from 'clickhouse';
|
||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
|
import {
|
||||||
|
MYSQL,
|
||||||
|
MYSQL_DATE_FORMATS,
|
||||||
|
POSTGRESQL,
|
||||||
|
POSTGRESQL_DATE_FORMATS,
|
||||||
|
CLICKHOUSE,
|
||||||
|
RELATIONAL,
|
||||||
|
FILTER_IGNORED,
|
||||||
|
} from 'lib/constants';
|
||||||
|
import moment from 'moment-timezone';
|
||||||
|
import { CLICKHOUSE_DATE_FORMATS } from './constants';
|
||||||
|
|
||||||
|
BigInt.prototype.toJSON = function () {
|
||||||
|
return Number(this);
|
||||||
|
};
|
||||||
|
|
||||||
const options = {
|
const options = {
|
||||||
log: [
|
log: [
|
||||||
|
|
@ -14,20 +30,261 @@ function logQuery(e) {
|
||||||
console.log(chalk.yellow(e.params), '->', e.query, chalk.greenBright(`${e.duration}ms`));
|
console.log(chalk.yellow(e.params), '->', e.query, chalk.greenBright(`${e.duration}ms`));
|
||||||
}
|
}
|
||||||
|
|
||||||
let prisma;
|
function getPrismaClient(options) {
|
||||||
|
const prisma = new PrismaClient(options);
|
||||||
|
|
||||||
if (process.env.NODE_ENV === 'production') {
|
if (process.env.LOG_QUERY) {
|
||||||
prisma = new PrismaClient(options);
|
prisma.$on('query', logQuery);
|
||||||
} else {
|
|
||||||
if (!global.prisma) {
|
|
||||||
global.prisma = new PrismaClient(options);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
prisma = global.prisma;
|
return prisma;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (process.env.LOG_QUERY) {
|
function getClickhouseClient() {
|
||||||
prisma.$on('query', logQuery);
|
if (!process.env.ANALYTICS_URL) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = new URL(process.env.ANALYTICS_URL);
|
||||||
|
const database = url.pathname.replace('/', '');
|
||||||
|
|
||||||
|
return new ClickHouse({
|
||||||
|
url: url.hostname,
|
||||||
|
port: Number(url.port),
|
||||||
|
basicAuth: url.password
|
||||||
|
? {
|
||||||
|
username: url.username || 'default',
|
||||||
|
password: url.password,
|
||||||
|
}
|
||||||
|
: null,
|
||||||
|
format: 'json',
|
||||||
|
config: {
|
||||||
|
database,
|
||||||
|
},
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export default prisma;
|
const prisma = global.prisma || getPrismaClient(options);
|
||||||
|
const clickhouse = global.clickhouse || getClickhouseClient();
|
||||||
|
|
||||||
|
if (process.env.NODE_ENV !== 'production') {
|
||||||
|
global.prisma = prisma;
|
||||||
|
global.clickhouse = clickhouse;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { prisma, clickhouse };
|
||||||
|
|
||||||
|
export function getDatabase() {
|
||||||
|
const type =
|
||||||
|
process.env.DATABASE_TYPE ||
|
||||||
|
(process.env.DATABASE_URL && process.env.DATABASE_URL.split(':')[0]);
|
||||||
|
|
||||||
|
if (type === 'postgres') {
|
||||||
|
return POSTGRESQL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getAnalyticsDatabase() {
|
||||||
|
const type =
|
||||||
|
process.env.ANALYTICS_TYPE ||
|
||||||
|
(process.env.ANALYTICS_URL && process.env.ANALYTICS_URL.split(':')[0]);
|
||||||
|
|
||||||
|
if (type === 'postgres') {
|
||||||
|
return POSTGRESQL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!type) {
|
||||||
|
return getDatabase();
|
||||||
|
}
|
||||||
|
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getDateStringQueryClickhouse(data, unit) {
|
||||||
|
return `formatDateTime(${data}, '${CLICKHOUSE_DATE_FORMATS[unit]}')`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getDateQuery(field, unit, timezone) {
|
||||||
|
const db = getDatabase();
|
||||||
|
|
||||||
|
if (db === POSTGRESQL) {
|
||||||
|
if (timezone) {
|
||||||
|
return `to_char(date_trunc('${unit}', ${field} at time zone '${timezone}'), '${POSTGRESQL_DATE_FORMATS[unit]}')`;
|
||||||
|
}
|
||||||
|
return `to_char(date_trunc('${unit}', ${field}), '${POSTGRESQL_DATE_FORMATS[unit]}')`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (db === MYSQL) {
|
||||||
|
if (timezone) {
|
||||||
|
const tz = moment.tz(timezone).format('Z');
|
||||||
|
|
||||||
|
return `date_format(convert_tz(${field},'+00:00','${tz}'), '${MYSQL_DATE_FORMATS[unit]}')`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return `date_format(${field}, '${MYSQL_DATE_FORMATS[unit]}')`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getDateQueryClickhouse(field, unit, timezone) {
|
||||||
|
if (timezone) {
|
||||||
|
return `date_trunc('${unit}', ${field},'${timezone}')`;
|
||||||
|
}
|
||||||
|
return `date_trunc('${unit}', ${field})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getDateFormatClickhouse(date) {
|
||||||
|
return `parseDateTimeBestEffort('${date.toUTCString()}')`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getBetweenDatesClickhouse(field, start_at, end_at) {
|
||||||
|
return `${field} between ${getDateFormatClickhouse(start_at)}
|
||||||
|
and ${getDateFormatClickhouse(end_at)}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getTimestampInterval(field) {
|
||||||
|
const db = getDatabase();
|
||||||
|
|
||||||
|
if (db === POSTGRESQL) {
|
||||||
|
return `floor(extract(epoch from max(${field}) - min(${field})))`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (db === MYSQL) {
|
||||||
|
return `floor(unix_timestamp(max(${field})) - unix_timestamp(min(${field})))`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getFilterQuery(table, column, filters = {}, params = []) {
|
||||||
|
const query = Object.keys(filters).reduce((arr, key) => {
|
||||||
|
const filter = filters[key];
|
||||||
|
|
||||||
|
if (filter === undefined || filter === FILTER_IGNORED) {
|
||||||
|
return arr;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (key) {
|
||||||
|
case 'url':
|
||||||
|
if (table === 'pageview' || table === 'event') {
|
||||||
|
arr.push(`and ${table}.${key}=$${params.length + 1}`);
|
||||||
|
params.push(decodeURIComponent(filter));
|
||||||
|
console.log(params);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'os':
|
||||||
|
case 'browser':
|
||||||
|
case 'device':
|
||||||
|
case 'country':
|
||||||
|
if (table === 'session') {
|
||||||
|
arr.push(`and ${table}.${key}=$${params.length + 1}`);
|
||||||
|
params.push(decodeURIComponent(filter));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'event_type':
|
||||||
|
if (table === 'event') {
|
||||||
|
arr.push(`and ${table}.${key}=$${params.length + 1}`);
|
||||||
|
params.push(decodeURIComponent(filter));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'referrer':
|
||||||
|
if (table === 'pageview' || table === 'event') {
|
||||||
|
arr.push(`and ${table}.referrer like $${params.length + 1}`);
|
||||||
|
params.push(`%${decodeURIComponent(filter)}%`);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'domain':
|
||||||
|
if (table === 'pageview') {
|
||||||
|
arr.push(`and ${table}.referrer not like $${params.length + 1}`);
|
||||||
|
arr.push(`and ${table}.referrer not like '/%'`);
|
||||||
|
params.push(`%://${filter}/%`);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return arr;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return query.join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseFilters(table, column, filters = {}, params = [], sessionKey = 'session_id') {
|
||||||
|
const { domain, url, event_url, referrer, os, browser, device, country, event_type } = filters;
|
||||||
|
|
||||||
|
const pageviewFilters = { domain, url, referrer };
|
||||||
|
const sessionFilters = { os, browser, device, country };
|
||||||
|
const eventFilters = { url: event_url, event_type };
|
||||||
|
|
||||||
|
return {
|
||||||
|
pageviewFilters,
|
||||||
|
sessionFilters,
|
||||||
|
eventFilters,
|
||||||
|
event: { event_type },
|
||||||
|
joinSession:
|
||||||
|
os || browser || device || country
|
||||||
|
? `inner join session on ${table}.${sessionKey} = session.${sessionKey}`
|
||||||
|
: '',
|
||||||
|
pageviewQuery: getFilterQuery('pageview', column, pageviewFilters, params),
|
||||||
|
sessionQuery: getFilterQuery('session', column, sessionFilters, params),
|
||||||
|
eventQuery: getFilterQuery('event', column, eventFilters, params),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function replaceQueryClickhouse(string, params = []) {
|
||||||
|
let formattedString = string;
|
||||||
|
|
||||||
|
params.forEach((a, i) => {
|
||||||
|
let replace = a;
|
||||||
|
|
||||||
|
if (typeof a === 'string' || a instanceof String) {
|
||||||
|
replace = `'${replace}'`;
|
||||||
|
}
|
||||||
|
|
||||||
|
formattedString = formattedString.replace(`$${i + 1}`, replace);
|
||||||
|
});
|
||||||
|
|
||||||
|
return formattedString;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function runQuery(query) {
|
||||||
|
return query.catch(e => {
|
||||||
|
throw e;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function rawQuery(query, params = []) {
|
||||||
|
const db = getDatabase();
|
||||||
|
|
||||||
|
if (db !== POSTGRESQL && db !== MYSQL) {
|
||||||
|
return Promise.reject(new Error('Unknown database.'));
|
||||||
|
}
|
||||||
|
|
||||||
|
const sql = db === MYSQL ? query.replace(/\$[0-9]+/g, '?') : query;
|
||||||
|
|
||||||
|
return runQuery(prisma.$queryRawUnsafe.apply(prisma, [sql, ...params]));
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function rawQueryClickhouse(query, params = [], debug = false) {
|
||||||
|
let formattedQuery = replaceQueryClickhouse(query, params);
|
||||||
|
|
||||||
|
if (debug || process.env.LOG_QUERY) {
|
||||||
|
console.log(formattedQuery);
|
||||||
|
}
|
||||||
|
|
||||||
|
return clickhouse.query(formattedQuery).toPromise();
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function runAnalyticsQuery(queries) {
|
||||||
|
const db = getAnalyticsDatabase();
|
||||||
|
|
||||||
|
if (db === POSTGRESQL || db === MYSQL) {
|
||||||
|
return queries[`${RELATIONAL}`]();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (db === CLICKHOUSE) {
|
||||||
|
return queries[`${CLICKHOUSE}`]();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,14 +1,10 @@
|
||||||
import { removeTrailingSlash, removeWWW, getDomainName } from './url';
|
import { removeWWW } from './url';
|
||||||
|
|
||||||
export const urlFilter = (data, { raw }) => {
|
export const urlFilter = data => {
|
||||||
const isValidUrl = url => {
|
const isValidUrl = url => {
|
||||||
return url !== '' && url !== null && !url.startsWith('#');
|
return url !== '' && url !== null && !url.startsWith('#');
|
||||||
};
|
};
|
||||||
|
|
||||||
if (raw) {
|
|
||||||
return data.filter(({ x }) => isValidUrl(x));
|
|
||||||
}
|
|
||||||
|
|
||||||
const cleanUrl = url => {
|
const cleanUrl = url => {
|
||||||
try {
|
try {
|
||||||
const { pathname, search } = new URL(url, location.origin);
|
const { pathname, search } = new URL(url, location.origin);
|
||||||
|
|
@ -44,66 +40,26 @@ export const urlFilter = (data, { raw }) => {
|
||||||
return Object.keys(map).map(key => ({ x: key, y: map[key] }));
|
return Object.keys(map).map(key => ({ x: key, y: map[key] }));
|
||||||
};
|
};
|
||||||
|
|
||||||
export const refFilter = (data, { domain, domainOnly, raw }) => {
|
export const refFilter = data => {
|
||||||
const domainName = getDomainName(domain);
|
|
||||||
const regex = new RegExp(`http[s]?://([a-z0-9-]+\\.)*${domainName}`);
|
|
||||||
const links = {};
|
const links = {};
|
||||||
|
|
||||||
const isValidRef = referrer => {
|
|
||||||
return (
|
|
||||||
referrer !== '' && referrer !== null && !referrer.startsWith('/') && !referrer.startsWith('#')
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const cleanUrl = url => {
|
|
||||||
try {
|
|
||||||
const { hostname, origin, pathname, searchParams, protocol } = new URL(url);
|
|
||||||
|
|
||||||
if (regex.test(url)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (domainOnly && hostname) {
|
|
||||||
return removeWWW(hostname);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!origin || origin === 'null') {
|
|
||||||
return `${protocol}${removeTrailingSlash(pathname)}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (protocol.startsWith('http')) {
|
|
||||||
const path = removeTrailingSlash(pathname);
|
|
||||||
const referrer = searchParams.get('referrer');
|
|
||||||
const query = referrer ? `?referrer=${referrer}` : '';
|
|
||||||
|
|
||||||
return removeTrailingSlash(`${removeWWW(hostname)}${path}`) + query;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
} catch {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (raw) {
|
|
||||||
return data.filter(({ x }) => isValidRef(x) && !regex.test(x));
|
|
||||||
}
|
|
||||||
|
|
||||||
const map = data.reduce((obj, { x, y }) => {
|
const map = data.reduce((obj, { x, y }) => {
|
||||||
if (!isValidRef(x)) {
|
let id;
|
||||||
return obj;
|
|
||||||
|
try {
|
||||||
|
const url = new URL(x);
|
||||||
|
|
||||||
|
id = removeWWW(url.hostname) || url.href;
|
||||||
|
} catch {
|
||||||
|
id = '';
|
||||||
}
|
}
|
||||||
|
|
||||||
const url = cleanUrl(x);
|
links[id] = x;
|
||||||
|
|
||||||
links[url] = x;
|
if (!obj[id]) {
|
||||||
|
obj[id] = y;
|
||||||
if (url) {
|
} else {
|
||||||
if (!obj[url]) {
|
obj[id] += y;
|
||||||
obj[url] = y;
|
|
||||||
} else {
|
|
||||||
obj[url] += y;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return obj;
|
return obj;
|
||||||
|
|
|
||||||
602
lib/queries.js
602
lib/queries.js
|
|
@ -1,602 +0,0 @@
|
||||||
import moment from 'moment-timezone';
|
|
||||||
import prisma from 'lib/db';
|
|
||||||
import { subMinutes } from 'date-fns';
|
|
||||||
import {
|
|
||||||
MYSQL,
|
|
||||||
POSTGRESQL,
|
|
||||||
MYSQL_DATE_FORMATS,
|
|
||||||
POSTGRESQL_DATE_FORMATS,
|
|
||||||
URL_LENGTH,
|
|
||||||
} from 'lib/constants';
|
|
||||||
|
|
||||||
export function getDatabase() {
|
|
||||||
const type =
|
|
||||||
process.env.DATABASE_TYPE ||
|
|
||||||
(process.env.DATABASE_URL && process.env.DATABASE_URL.split(':')[0]);
|
|
||||||
|
|
||||||
if (type === 'postgres') {
|
|
||||||
return 'postgresql';
|
|
||||||
}
|
|
||||||
|
|
||||||
return type;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getDateQuery(field, unit, timezone) {
|
|
||||||
const db = getDatabase();
|
|
||||||
|
|
||||||
if (db === POSTGRESQL) {
|
|
||||||
if (timezone) {
|
|
||||||
return `to_char(date_trunc('${unit}', ${field} at time zone '${timezone}'), '${POSTGRESQL_DATE_FORMATS[unit]}')`;
|
|
||||||
}
|
|
||||||
return `to_char(date_trunc('${unit}', ${field}), '${POSTGRESQL_DATE_FORMATS[unit]}')`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (db === MYSQL) {
|
|
||||||
if (timezone) {
|
|
||||||
const tz = moment.tz(timezone).format('Z');
|
|
||||||
|
|
||||||
return `DATE_FORMAT(convert_tz(${field},'+00:00','${tz}'), '${MYSQL_DATE_FORMATS[unit]}')`;
|
|
||||||
}
|
|
||||||
|
|
||||||
return `DATE_FORMAT(${field}, '${MYSQL_DATE_FORMATS[unit]}')`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getTimestampInterval(field) {
|
|
||||||
const db = getDatabase();
|
|
||||||
|
|
||||||
if (db === POSTGRESQL) {
|
|
||||||
return `floor(extract(epoch from max(${field}) - min(${field})))`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (db === MYSQL) {
|
|
||||||
return `floor(unix_timestamp(max(${field})) - unix_timestamp(min(${field})))`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getFilterQuery(table, filters = {}, params = []) {
|
|
||||||
const query = Object.keys(filters).reduce((arr, key) => {
|
|
||||||
const value = filters[key];
|
|
||||||
|
|
||||||
if (value === undefined) {
|
|
||||||
return arr;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (key) {
|
|
||||||
case 'url':
|
|
||||||
if (table === 'pageview' || table === 'event') {
|
|
||||||
arr.push(`and ${table}.${key}=$${params.length + 1}`);
|
|
||||||
params.push(decodeURIComponent(value));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'os':
|
|
||||||
case 'browser':
|
|
||||||
case 'device':
|
|
||||||
case 'country':
|
|
||||||
if (table === 'session') {
|
|
||||||
arr.push(`and ${table}.${key}=$${params.length + 1}`);
|
|
||||||
params.push(decodeURIComponent(value));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'event_type':
|
|
||||||
if (table === 'event') {
|
|
||||||
arr.push(`and ${table}.${key}=$${params.length + 1}`);
|
|
||||||
params.push(decodeURIComponent(value));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'referrer':
|
|
||||||
if (table === 'pageview') {
|
|
||||||
arr.push(`and ${table}.referrer like $${params.length + 1}`);
|
|
||||||
params.push(`%${decodeURIComponent(value)}%`);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'domain':
|
|
||||||
if (table === 'pageview') {
|
|
||||||
arr.push(`and ${table}.referrer not like $${params.length + 1}`);
|
|
||||||
arr.push(`and ${table}.referrer not like '/%'`);
|
|
||||||
params.push(`%://${value}/%`);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return arr;
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return query.join('\n');
|
|
||||||
}
|
|
||||||
|
|
||||||
export function parseFilters(table, filters = {}, params = []) {
|
|
||||||
const { domain, url, event_url, referrer, os, browser, device, country, event_type } = filters;
|
|
||||||
|
|
||||||
const pageviewFilters = { domain, url, referrer };
|
|
||||||
const sessionFilters = { os, browser, device, country };
|
|
||||||
const eventFilters = { url: event_url, event_type };
|
|
||||||
|
|
||||||
return {
|
|
||||||
pageviewFilters,
|
|
||||||
sessionFilters,
|
|
||||||
eventFilters,
|
|
||||||
event: { event_type },
|
|
||||||
joinSession:
|
|
||||||
os || browser || device || country
|
|
||||||
? `inner join session on ${table}.session_id = session.session_id`
|
|
||||||
: '',
|
|
||||||
pageviewQuery: getFilterQuery('pageview', pageviewFilters, params),
|
|
||||||
sessionQuery: getFilterQuery('session', sessionFilters, params),
|
|
||||||
eventQuery: getFilterQuery('event', eventFilters, params),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function runQuery(query) {
|
|
||||||
return query.catch(e => {
|
|
||||||
throw e;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function rawQuery(query, params = []) {
|
|
||||||
const db = getDatabase();
|
|
||||||
|
|
||||||
if (db !== POSTGRESQL && db !== MYSQL) {
|
|
||||||
return Promise.reject(new Error('Unknown database.'));
|
|
||||||
}
|
|
||||||
|
|
||||||
const sql = db === MYSQL ? query.replace(/\$[0-9]+/g, '?') : query;
|
|
||||||
|
|
||||||
return runQuery(prisma.$queryRawUnsafe.apply(prisma, [sql, ...params]));
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getWebsiteById(website_id) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.findUnique({
|
|
||||||
where: {
|
|
||||||
website_id,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getWebsiteByUuid(website_uuid) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.findUnique({
|
|
||||||
where: {
|
|
||||||
website_uuid,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getWebsiteByShareId(share_id) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.findUnique({
|
|
||||||
where: {
|
|
||||||
share_id,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getUserWebsites(user_id) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.findMany({
|
|
||||||
where: {
|
|
||||||
user_id,
|
|
||||||
},
|
|
||||||
orderBy: {
|
|
||||||
name: 'asc',
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getAllWebsites() {
|
|
||||||
let data = await runQuery(
|
|
||||||
prisma.website.findMany({
|
|
||||||
orderBy: [
|
|
||||||
{
|
|
||||||
user_id: 'asc',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'asc',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
include: {
|
|
||||||
account: {
|
|
||||||
select: {
|
|
||||||
username: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
return data.map(i => ({ ...i, account: i.account.username }));
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function createWebsite(user_id, data) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.create({
|
|
||||||
data: {
|
|
||||||
account: {
|
|
||||||
connect: {
|
|
||||||
user_id,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
...data,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function updateWebsite(website_id, data) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.update({
|
|
||||||
where: {
|
|
||||||
website_id,
|
|
||||||
},
|
|
||||||
data,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function resetWebsite(website_id) {
|
|
||||||
return runQuery(prisma.$queryRaw`delete from session where website_id=${website_id}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function deleteWebsite(website_id) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.website.delete({
|
|
||||||
where: {
|
|
||||||
website_id,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function createSession(website_id, data) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.session.create({
|
|
||||||
data: {
|
|
||||||
website_id,
|
|
||||||
...data,
|
|
||||||
},
|
|
||||||
select: {
|
|
||||||
session_id: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getSessionByUuid(session_uuid) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.session.findUnique({
|
|
||||||
where: {
|
|
||||||
session_uuid,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function savePageView(website_id, session_id, url, referrer) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.pageview.create({
|
|
||||||
data: {
|
|
||||||
website_id,
|
|
||||||
session_id,
|
|
||||||
url: url?.substr(0, URL_LENGTH),
|
|
||||||
referrer: referrer?.substr(0, URL_LENGTH),
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function saveEvent(website_id, session_id, url, event_type, event_value) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.event.create({
|
|
||||||
data: {
|
|
||||||
website_id,
|
|
||||||
session_id,
|
|
||||||
url: url?.substr(0, URL_LENGTH),
|
|
||||||
event_type: event_type?.substr(0, 50),
|
|
||||||
event_value: event_value?.substr(0, 50),
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getAccounts() {
|
|
||||||
return runQuery(
|
|
||||||
prisma.account.findMany({
|
|
||||||
orderBy: [
|
|
||||||
{ is_admin: 'desc' },
|
|
||||||
{
|
|
||||||
username: 'asc',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
select: {
|
|
||||||
user_id: true,
|
|
||||||
username: true,
|
|
||||||
is_admin: true,
|
|
||||||
created_at: true,
|
|
||||||
updated_at: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getAccountById(user_id) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.account.findUnique({
|
|
||||||
where: {
|
|
||||||
user_id,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getAccountByUsername(username) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.account.findUnique({
|
|
||||||
where: {
|
|
||||||
username,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function updateAccount(user_id, data) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.account.update({
|
|
||||||
where: {
|
|
||||||
user_id,
|
|
||||||
},
|
|
||||||
data,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function deleteAccount(user_id) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.account.delete({
|
|
||||||
where: {
|
|
||||||
user_id,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function createAccount(data) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.account.create({
|
|
||||||
data,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getSessions(websites, start_at) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.session.findMany({
|
|
||||||
where: {
|
|
||||||
website: {
|
|
||||||
website_id: {
|
|
||||||
in: websites,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
created_at: {
|
|
||||||
gte: start_at,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getPageviews(websites, start_at) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.pageview.findMany({
|
|
||||||
where: {
|
|
||||||
website: {
|
|
||||||
website_id: {
|
|
||||||
in: websites,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
created_at: {
|
|
||||||
gte: start_at,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getEvents(websites, start_at) {
|
|
||||||
return runQuery(
|
|
||||||
prisma.event.findMany({
|
|
||||||
where: {
|
|
||||||
website: {
|
|
||||||
website_id: {
|
|
||||||
in: websites,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
created_at: {
|
|
||||||
gte: start_at,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getWebsiteStats(website_id, start_at, end_at, filters = {}) {
|
|
||||||
const params = [website_id, start_at, end_at];
|
|
||||||
const { pageviewQuery, sessionQuery, joinSession } = parseFilters('pageview', filters, params);
|
|
||||||
|
|
||||||
return rawQuery(
|
|
||||||
`
|
|
||||||
select sum(t.c) as "pageviews",
|
|
||||||
count(distinct t.session_id) as "uniques",
|
|
||||||
sum(case when t.c = 1 then 1 else 0 end) as "bounces",
|
|
||||||
sum(t.time) as "totaltime"
|
|
||||||
from (
|
|
||||||
select pageview.session_id,
|
|
||||||
${getDateQuery('pageview.created_at', 'hour')},
|
|
||||||
count(*) c,
|
|
||||||
${getTimestampInterval('pageview.created_at')} as "time"
|
|
||||||
from pageview
|
|
||||||
${joinSession}
|
|
||||||
where pageview.website_id=$1
|
|
||||||
and pageview.created_at between $2 and $3
|
|
||||||
${pageviewQuery}
|
|
||||||
${sessionQuery}
|
|
||||||
group by 1, 2
|
|
||||||
) t
|
|
||||||
`,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getPageviewStats(
|
|
||||||
website_id,
|
|
||||||
start_at,
|
|
||||||
end_at,
|
|
||||||
timezone = 'utc',
|
|
||||||
unit = 'day',
|
|
||||||
count = '*',
|
|
||||||
filters = {},
|
|
||||||
) {
|
|
||||||
const params = [website_id, start_at, end_at];
|
|
||||||
const { pageviewQuery, sessionQuery, joinSession } = parseFilters('pageview', filters, params);
|
|
||||||
|
|
||||||
return rawQuery(
|
|
||||||
`
|
|
||||||
select ${getDateQuery('pageview.created_at', unit, timezone)} t,
|
|
||||||
count(${count}) y
|
|
||||||
from pageview
|
|
||||||
${joinSession}
|
|
||||||
where pageview.website_id=$1
|
|
||||||
and pageview.created_at between $2 and $3
|
|
||||||
${pageviewQuery}
|
|
||||||
${sessionQuery}
|
|
||||||
group by 1
|
|
||||||
order by 1
|
|
||||||
`,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getSessionMetrics(website_id, start_at, end_at, field, filters = {}) {
|
|
||||||
const params = [website_id, start_at, end_at];
|
|
||||||
const { pageviewQuery, sessionQuery, joinSession } = parseFilters('pageview', filters, params);
|
|
||||||
|
|
||||||
return rawQuery(
|
|
||||||
`
|
|
||||||
select ${field} x, count(*) y
|
|
||||||
from session as x
|
|
||||||
where x.session_id in (
|
|
||||||
select pageview.session_id
|
|
||||||
from pageview
|
|
||||||
${joinSession}
|
|
||||||
where pageview.website_id=$1
|
|
||||||
and pageview.created_at between $2 and $3
|
|
||||||
${pageviewQuery}
|
|
||||||
${sessionQuery}
|
|
||||||
)
|
|
||||||
group by 1
|
|
||||||
order by 2 desc
|
|
||||||
`,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getPageviewMetrics(website_id, start_at, end_at, field, table, filters = {}) {
|
|
||||||
const params = [website_id, start_at, end_at];
|
|
||||||
const { pageviewQuery, sessionQuery, eventQuery, joinSession } = parseFilters(
|
|
||||||
table,
|
|
||||||
filters,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
|
|
||||||
return rawQuery(
|
|
||||||
`
|
|
||||||
select ${field} x, count(*) y
|
|
||||||
from ${table}
|
|
||||||
${joinSession}
|
|
||||||
where ${table}.website_id=$1
|
|
||||||
and ${table}.created_at between $2 and $3
|
|
||||||
${pageviewQuery}
|
|
||||||
${joinSession && sessionQuery}
|
|
||||||
${eventQuery}
|
|
||||||
group by 1
|
|
||||||
order by 2 desc
|
|
||||||
`,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getActiveVisitors(website_id) {
|
|
||||||
const date = subMinutes(new Date(), 5);
|
|
||||||
const params = [website_id, date];
|
|
||||||
|
|
||||||
return rawQuery(
|
|
||||||
`
|
|
||||||
select count(distinct session_id) x
|
|
||||||
from pageview
|
|
||||||
where website_id=$1
|
|
||||||
and created_at >= $2
|
|
||||||
`,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getEventMetrics(
|
|
||||||
website_id,
|
|
||||||
start_at,
|
|
||||||
end_at,
|
|
||||||
timezone = 'utc',
|
|
||||||
unit = 'day',
|
|
||||||
filters = {},
|
|
||||||
) {
|
|
||||||
const params = [website_id, start_at, end_at];
|
|
||||||
|
|
||||||
return rawQuery(
|
|
||||||
`
|
|
||||||
select
|
|
||||||
event_value x,
|
|
||||||
${getDateQuery('created_at', unit, timezone)} t,
|
|
||||||
count(*) y
|
|
||||||
from event
|
|
||||||
where website_id=$1
|
|
||||||
and created_at between $2 and $3
|
|
||||||
${getFilterQuery('event', filters, params)}
|
|
||||||
group by 1, 2
|
|
||||||
order by 2
|
|
||||||
`,
|
|
||||||
params,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getRealtimeData(websites, time) {
|
|
||||||
const [pageviews, sessions, events] = await Promise.all([
|
|
||||||
getPageviews(websites, time),
|
|
||||||
getSessions(websites, time),
|
|
||||||
getEvents(websites, time),
|
|
||||||
]);
|
|
||||||
|
|
||||||
return {
|
|
||||||
pageviews: pageviews.map(({ view_id, ...props }) => ({
|
|
||||||
__id: `p${view_id}`,
|
|
||||||
view_id,
|
|
||||||
...props,
|
|
||||||
})),
|
|
||||||
sessions: sessions.map(({ session_id, ...props }) => ({
|
|
||||||
__id: `s${session_id}`,
|
|
||||||
session_id,
|
|
||||||
...props,
|
|
||||||
})),
|
|
||||||
events: events.map(({ event_id, ...props }) => ({
|
|
||||||
__id: `e${event_id}`,
|
|
||||||
event_id,
|
|
||||||
...props,
|
|
||||||
})),
|
|
||||||
timestamp: Date.now(),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getWebsiteByUuid, getSessionByUuid, createSession } from 'lib/queries';
|
import { getWebsiteByUuid, getSessionByUuid, createSession } from 'queries';
|
||||||
import { getJsonBody, getClientInfo } from 'lib/request';
|
import { getJsonBody, getClientInfo } from 'lib/request';
|
||||||
import { uuid, isValidUuid, parseToken } from 'lib/crypto';
|
import { uuid, isValidUuid, parseToken } from 'lib/crypto';
|
||||||
|
|
||||||
|
|
@ -37,6 +37,8 @@ export async function getSession(req) {
|
||||||
|
|
||||||
let session = await getSessionByUuid(session_uuid);
|
let session = await getSessionByUuid(session_uuid);
|
||||||
|
|
||||||
|
session = Array.isArray(session) && session[0] ? session[0] : session;
|
||||||
|
|
||||||
if (!session) {
|
if (!session) {
|
||||||
try {
|
try {
|
||||||
session = await createSession(website_id, {
|
session = await createSession(website_id, {
|
||||||
|
|
@ -65,5 +67,6 @@ export async function getSession(req) {
|
||||||
return {
|
return {
|
||||||
website_id,
|
website_id,
|
||||||
session_id,
|
session_id,
|
||||||
|
session_uuid,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -6,14 +6,6 @@ export function removeWWW(url) {
|
||||||
return url && url.length > 1 && url.startsWith('www.') ? url.slice(4) : url;
|
return url && url.length > 1 && url.startsWith('www.') ? url.slice(4) : url;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getDomainName(str) {
|
|
||||||
try {
|
|
||||||
return new URL(str).hostname;
|
|
||||||
} catch (e) {
|
|
||||||
return str;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getQueryString(params = {}) {
|
export function getQueryString(params = {}) {
|
||||||
const map = Object.keys(params).reduce((arr, key) => {
|
const map = Object.keys(params).reduce((arr, key) => {
|
||||||
if (params[key] !== undefined) {
|
if (params[key] !== undefined) {
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,23 @@
|
||||||
import { NextResponse } from 'next/server';
|
import { NextResponse } from 'next/server';
|
||||||
|
|
||||||
|
export const config = {
|
||||||
|
matcher: '/:path*',
|
||||||
|
};
|
||||||
|
|
||||||
|
function customCollectEndpoint(req) {
|
||||||
|
const collectEndpoint = process.env.COLLECT_API_ENDPOINT;
|
||||||
|
|
||||||
|
if (collectEndpoint) {
|
||||||
|
const url = req.nextUrl.clone();
|
||||||
|
const { pathname } = url;
|
||||||
|
|
||||||
|
if (pathname.endsWith(collectEndpoint)) {
|
||||||
|
url.pathname = '/api/collect';
|
||||||
|
return NextResponse.rewrite(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function customScriptName(req) {
|
function customScriptName(req) {
|
||||||
const scriptName = process.env.TRACKER_SCRIPT_NAME;
|
const scriptName = process.env.TRACKER_SCRIPT_NAME;
|
||||||
|
|
||||||
|
|
@ -23,8 +41,8 @@ function forceSSL(req, res) {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function middleware(req) {
|
export default function middleware(req) {
|
||||||
const fns = [customScriptName];
|
const fns = [customCollectEndpoint, customScriptName];
|
||||||
|
|
||||||
for (const fn of fns) {
|
for (const fn of fns) {
|
||||||
const res = fn(req);
|
const res = fn(req);
|
||||||
|
|
@ -6,11 +6,10 @@ module.exports = {
|
||||||
currentVersion: pkg.version,
|
currentVersion: pkg.version,
|
||||||
loginDisabled: process.env.DISABLE_LOGIN,
|
loginDisabled: process.env.DISABLE_LOGIN,
|
||||||
updatesDisabled: process.env.DISABLE_UPDATES,
|
updatesDisabled: process.env.DISABLE_UPDATES,
|
||||||
|
telemetryDisabled: process.env.DISABLE_TELEMETRY,
|
||||||
},
|
},
|
||||||
basePath: process.env.BASE_PATH,
|
basePath: process.env.BASE_PATH,
|
||||||
experimental: {
|
output: 'standalone',
|
||||||
outputStandalone: true,
|
|
||||||
},
|
|
||||||
eslint: {
|
eslint: {
|
||||||
ignoreDuringBuilds: true,
|
ignoreDuringBuilds: true,
|
||||||
},
|
},
|
||||||
|
|
|
||||||
16
package.json
16
package.json
|
|
@ -1,13 +1,13 @@
|
||||||
{
|
{
|
||||||
"name": "umami",
|
"name": "umami",
|
||||||
"version": "1.33.1",
|
"version": "1.36.1",
|
||||||
"description": "A simple, fast, privacy-focused alternative to Google Analytics.",
|
"description": "A simple, fast, privacy-focused alternative to Google Analytics.",
|
||||||
"author": "Mike Cao <mike@mikecao.com>",
|
"author": "Mike Cao <mike@mikecao.com>",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"homepage": "https://umami.is",
|
"homepage": "https://umami.is",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/mikecao/umami.git"
|
"url": "https://github.com/umami-software/umami.git"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev",
|
"dev": "next dev",
|
||||||
|
|
@ -55,11 +55,12 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fontsource/inter": "4.5.7",
|
"@fontsource/inter": "4.5.7",
|
||||||
"@prisma/client": "3.15.2",
|
"@prisma/client": "4.1.0",
|
||||||
"bcryptjs": "^2.4.3",
|
"bcryptjs": "^2.4.3",
|
||||||
"chalk": "^4.1.1",
|
"chalk": "^4.1.1",
|
||||||
"chart.js": "^2.9.4",
|
"chart.js": "^2.9.4",
|
||||||
"classnames": "^2.3.1",
|
"classnames": "^2.3.1",
|
||||||
|
"clickhouse": "^2.5.0",
|
||||||
"colord": "^2.9.2",
|
"colord": "^2.9.2",
|
||||||
"cors": "^2.8.5",
|
"cors": "^2.8.5",
|
||||||
"cross-spawn": "^7.0.3",
|
"cross-spawn": "^7.0.3",
|
||||||
|
|
@ -79,8 +80,8 @@
|
||||||
"jose": "2.0.5",
|
"jose": "2.0.5",
|
||||||
"maxmind": "^4.3.6",
|
"maxmind": "^4.3.6",
|
||||||
"moment-timezone": "^0.5.33",
|
"moment-timezone": "^0.5.33",
|
||||||
"next": "12.1.0",
|
"next": "^12.2.0",
|
||||||
"node-fetch": "^3.2.3",
|
"node-fetch": "^3.2.8",
|
||||||
"npm-run-all": "^4.1.5",
|
"npm-run-all": "^4.1.5",
|
||||||
"prop-types": "^15.7.2",
|
"prop-types": "^15.7.2",
|
||||||
"react": "^17.0.0",
|
"react": "^17.0.0",
|
||||||
|
|
@ -101,10 +102,11 @@
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@formatjs/cli": "^4.2.29",
|
"@formatjs/cli": "^4.2.29",
|
||||||
"@rollup/plugin-buble": "^0.21.3",
|
"@rollup/plugin-buble": "^0.21.3",
|
||||||
|
"@rollup/plugin-replace": "^4.0.0",
|
||||||
"@svgr/webpack": "^6.2.1",
|
"@svgr/webpack": "^6.2.1",
|
||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
"eslint": "^7.32.0",
|
"eslint": "^7.32.0",
|
||||||
"eslint-config-next": "^12.0.1",
|
"eslint-config-next": "^12.2.0",
|
||||||
"eslint-config-prettier": "^8.5.0",
|
"eslint-config-prettier": "^8.5.0",
|
||||||
"eslint-plugin-prettier": "^4.0.0",
|
"eslint-plugin-prettier": "^4.0.0",
|
||||||
"extract-react-intl-messages": "^4.1.1",
|
"extract-react-intl-messages": "^4.1.1",
|
||||||
|
|
@ -116,7 +118,7 @@
|
||||||
"postcss-preset-env": "7.4.3",
|
"postcss-preset-env": "7.4.3",
|
||||||
"postcss-rtlcss": "^3.6.1",
|
"postcss-rtlcss": "^3.6.1",
|
||||||
"prettier": "^2.6.2",
|
"prettier": "^2.6.2",
|
||||||
"prisma": "3.15.2",
|
"prisma": "4.1.0",
|
||||||
"prompts": "2.4.2",
|
"prompts": "2.4.2",
|
||||||
"rollup": "^2.70.1",
|
"rollup": "^2.70.1",
|
||||||
"rollup-plugin-terser": "^7.0.2",
|
"rollup-plugin-terser": "^7.0.2",
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getAccountById, deleteAccount } from 'lib/queries';
|
import { getAccountById, deleteAccount } from 'queries';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getAccountById, getAccountByUsername, updateAccount, createAccount } from 'lib/queries';
|
import { getAccountById, getAccountByUsername, updateAccount, createAccount } from 'queries';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { hashPassword } from 'lib/crypto';
|
import { hashPassword } from 'lib/crypto';
|
||||||
import { ok, unauthorized, methodNotAllowed, badRequest } from 'lib/response';
|
import { ok, unauthorized, methodNotAllowed, badRequest } from 'lib/response';
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getAccountById, updateAccount } from 'lib/queries';
|
import { getAccountById, updateAccount } from 'queries';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { badRequest, methodNotAllowed, ok, unauthorized } from 'lib/response';
|
import { badRequest, methodNotAllowed, ok, unauthorized } from 'lib/response';
|
||||||
import { checkPassword, hashPassword } from 'lib/crypto';
|
import { checkPassword, hashPassword } from 'lib/crypto';
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getAccounts } from 'lib/queries';
|
import { getAccounts } from 'queries';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { ok, unauthorized, methodNotAllowed } from 'lib/response';
|
import { ok, unauthorized, methodNotAllowed } from 'lib/response';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
import { checkPassword, createSecureToken } from 'lib/crypto';
|
import { checkPassword, createSecureToken } from 'lib/crypto';
|
||||||
import { getAccountByUsername } from 'lib/queries';
|
import { getAccountByUsername } from 'queries/admin/account/getAccountByUsername';
|
||||||
import { ok, unauthorized, badRequest } from 'lib/response';
|
import { ok, unauthorized, badRequest } from 'lib/response';
|
||||||
|
|
||||||
export default async (req, res) => {
|
export default async (req, res) => {
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
const { Resolver } = require('dns').promises;
|
const { Resolver } = require('dns').promises;
|
||||||
import isbot from 'isbot';
|
import isbot from 'isbot';
|
||||||
import ipaddr from 'ipaddr.js';
|
import ipaddr from 'ipaddr.js';
|
||||||
import { savePageView, saveEvent } from 'lib/queries';
|
import { savePageView, saveEvent } from 'queries';
|
||||||
import { useCors, useSession } from 'lib/middleware';
|
import { useCors, useSession } from 'lib/middleware';
|
||||||
import { getJsonBody, getIpAddress } from 'lib/request';
|
import { getJsonBody, getIpAddress } from 'lib/request';
|
||||||
import { ok, send, badRequest, forbidden } from 'lib/response';
|
import { ok, send, badRequest, forbidden } from 'lib/response';
|
||||||
|
|
@ -60,7 +60,7 @@ export default async (req, res) => {
|
||||||
await useSession(req, res);
|
await useSession(req, res);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
session: { website_id, session_id },
|
session: { website_id, session_id, session_uuid },
|
||||||
} = req;
|
} = req;
|
||||||
|
|
||||||
const { type, payload } = getJsonBody(req);
|
const { type, payload } = getJsonBody(req);
|
||||||
|
|
@ -72,9 +72,9 @@ export default async (req, res) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (type === 'pageview') {
|
if (type === 'pageview') {
|
||||||
await savePageView(website_id, session_id, url, referrer);
|
await savePageView(website_id, { session_id, session_uuid, url, referrer });
|
||||||
} else if (type === 'event') {
|
} else if (type === 'event') {
|
||||||
await saveEvent(website_id, session_id, url, event_type, event_value);
|
await saveEvent(website_id, { session_id, session_uuid, url, event_type, event_value });
|
||||||
} else {
|
} else {
|
||||||
return badRequest(res);
|
return badRequest(res);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
import { subMinutes } from 'date-fns';
|
import { subMinutes } from 'date-fns';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { ok, methodNotAllowed } from 'lib/response';
|
import { ok, methodNotAllowed } from 'lib/response';
|
||||||
import { getUserWebsites, getRealtimeData } from 'lib/queries';
|
import { getUserWebsites, getRealtimeData } from 'queries';
|
||||||
import { createToken } from 'lib/crypto';
|
import { createToken } from 'lib/crypto';
|
||||||
|
|
||||||
export default async (req, res) => {
|
export default async (req, res) => {
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { ok, methodNotAllowed, badRequest } from 'lib/response';
|
import { ok, methodNotAllowed, badRequest } from 'lib/response';
|
||||||
import { getRealtimeData } from 'lib/queries';
|
import { getRealtimeData } from 'queries';
|
||||||
import { parseToken } from 'lib/crypto';
|
import { parseToken } from 'lib/crypto';
|
||||||
import { SHARE_TOKEN_HEADER } from 'lib/constants';
|
import { SHARE_TOKEN_HEADER } from 'lib/constants';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getWebsiteByShareId } from 'lib/queries';
|
import { getWebsiteByShareId } from 'queries';
|
||||||
import { ok, notFound, methodNotAllowed } from 'lib/response';
|
import { ok, notFound, methodNotAllowed } from 'lib/response';
|
||||||
import { createToken } from 'lib/crypto';
|
import { createToken } from 'lib/crypto';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
import { getActiveVisitors } from 'lib/queries';
|
|
||||||
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
import { useCors } from 'lib/middleware';
|
import { useCors } from 'lib/middleware';
|
||||||
|
import { getActiveVisitors } from 'queries';
|
||||||
|
|
||||||
export default async (req, res) => {
|
export default async (req, res) => {
|
||||||
if (req.method === 'GET') {
|
if (req.method === 'GET') {
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
import moment from 'moment-timezone';
|
import moment from 'moment-timezone';
|
||||||
import { getEventMetrics } from 'lib/queries';
|
import { getEventMetrics } from 'queries';
|
||||||
import { ok, badRequest, methodNotAllowed, unauthorized } from 'lib/response';
|
import { ok, badRequest, methodNotAllowed, unauthorized } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
import { useCors } from 'lib/middleware';
|
import { useCors } from 'lib/middleware';
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { deleteWebsite, getWebsiteById } from 'lib/queries';
|
import { deleteWebsite, getWebsiteById } from 'queries';
|
||||||
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
import { useCors } from 'lib/middleware';
|
import { useCors } from 'lib/middleware';
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,12 @@
|
||||||
import { getPageviewMetrics, getSessionMetrics, getWebsiteById } from 'lib/queries';
|
import { getPageviewMetrics, getSessionMetrics, getWebsiteById, getPageviewParams } from 'queries';
|
||||||
import { ok, methodNotAllowed, unauthorized, badRequest } from 'lib/response';
|
import { ok, methodNotAllowed, unauthorized, badRequest } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
import { useCors } from 'lib/middleware';
|
import { useCors } from 'lib/middleware';
|
||||||
|
import { FILTER_IGNORED } from 'lib/constants';
|
||||||
|
|
||||||
const sessionColumns = ['browser', 'os', 'device', 'country', 'language'];
|
const sessionColumns = ['browser', 'os', 'device', 'screen', 'country', 'language'];
|
||||||
const pageviewColumns = ['url', 'referrer'];
|
const pageviewColumns = ['url', 'referrer'];
|
||||||
|
const paramTypes = ['utm_source', 'utm_medium', 'utm_campaign', 'utm_content', 'utm_term', 'ref'];
|
||||||
|
|
||||||
function getTable(type) {
|
function getTable(type) {
|
||||||
if (type === 'event') {
|
if (type === 'event') {
|
||||||
|
|
@ -39,6 +41,52 @@ export default async (req, res) => {
|
||||||
const startDate = new Date(+start_at);
|
const startDate = new Date(+start_at);
|
||||||
const endDate = new Date(+end_at);
|
const endDate = new Date(+end_at);
|
||||||
|
|
||||||
|
if (paramTypes.includes(type)) {
|
||||||
|
const column = 'url';
|
||||||
|
const table = getTable(type);
|
||||||
|
|
||||||
|
let domain;
|
||||||
|
if (type === 'referrer') {
|
||||||
|
const website = await getWebsiteById(websiteId);
|
||||||
|
|
||||||
|
if (!website) {
|
||||||
|
return badRequest(res);
|
||||||
|
}
|
||||||
|
|
||||||
|
domain = website.domain;
|
||||||
|
}
|
||||||
|
|
||||||
|
const filters = {
|
||||||
|
domain,
|
||||||
|
url: type !== 'url' && table !== 'event' ? url : undefined,
|
||||||
|
referrer: type !== 'referrer' ? referrer : true,
|
||||||
|
os: type !== 'os' ? os : undefined,
|
||||||
|
browser: type !== 'browser' ? browser : undefined,
|
||||||
|
device: type !== 'device' ? device : undefined,
|
||||||
|
country: type !== 'country' ? country : undefined,
|
||||||
|
event_url: type !== 'url' && table === 'event' ? url : undefined,
|
||||||
|
};
|
||||||
|
|
||||||
|
let data = await getPageviewParams(
|
||||||
|
type,
|
||||||
|
websiteId,
|
||||||
|
startDate,
|
||||||
|
endDate,
|
||||||
|
column,
|
||||||
|
table,
|
||||||
|
filters,
|
||||||
|
);
|
||||||
|
|
||||||
|
let terms = {};
|
||||||
|
new Set(data.map(i => i.param)).forEach(term => (terms[term] = null));
|
||||||
|
for (let { param } of data) terms[param] += 1;
|
||||||
|
|
||||||
|
return ok(
|
||||||
|
res,
|
||||||
|
Object.keys(terms).map(i => ({ x: i, y: terms[i] })),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (sessionColumns.includes(type)) {
|
if (sessionColumns.includes(type)) {
|
||||||
let data = await getSessionMetrics(websiteId, startDate, endDate, type, {
|
let data = await getSessionMetrics(websiteId, startDate, endDate, type, {
|
||||||
os,
|
os,
|
||||||
|
|
@ -69,7 +117,7 @@ export default async (req, res) => {
|
||||||
if (pageviewColumns.includes(type) || type === 'event') {
|
if (pageviewColumns.includes(type) || type === 'event') {
|
||||||
let domain;
|
let domain;
|
||||||
if (type === 'referrer') {
|
if (type === 'referrer') {
|
||||||
const website = getWebsiteById(websiteId);
|
const website = await getWebsiteById(websiteId);
|
||||||
|
|
||||||
if (!website) {
|
if (!website) {
|
||||||
return badRequest(res);
|
return badRequest(res);
|
||||||
|
|
@ -80,17 +128,18 @@ export default async (req, res) => {
|
||||||
|
|
||||||
const column = getColumn(type);
|
const column = getColumn(type);
|
||||||
const table = getTable(type);
|
const table = getTable(type);
|
||||||
|
const filters = {
|
||||||
const data = await getPageviewMetrics(websiteId, startDate, endDate, column, table, {
|
|
||||||
domain,
|
domain,
|
||||||
url: type !== 'url' && table !== 'event' ? url : undefined,
|
url: type !== 'url' && table !== 'event' ? url : undefined,
|
||||||
referrer: type !== 'referrer' ? referrer : undefined,
|
referrer: type !== 'referrer' && table !== 'event' ? referrer : FILTER_IGNORED,
|
||||||
os: type !== 'os' ? os : undefined,
|
os: type !== 'os' ? os : undefined,
|
||||||
browser: type !== 'browser' ? browser : undefined,
|
browser: type !== 'browser' ? browser : undefined,
|
||||||
device: type !== 'device' ? device : undefined,
|
device: type !== 'device' ? device : undefined,
|
||||||
country: type !== 'country' ? country : undefined,
|
country: type !== 'country' ? country : undefined,
|
||||||
event_url: type !== 'url' && table === 'event' ? url : undefined,
|
event_url: type !== 'url' && table === 'event' ? url : undefined,
|
||||||
});
|
};
|
||||||
|
|
||||||
|
const data = await getPageviewMetrics(websiteId, startDate, endDate, column, table, filters);
|
||||||
|
|
||||||
return ok(res, data);
|
return ok(res, data);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
import moment from 'moment-timezone';
|
import moment from 'moment-timezone';
|
||||||
import { getPageviewStats } from 'lib/queries';
|
import { getPageviewStats } from 'queries';
|
||||||
import { ok, badRequest, methodNotAllowed, unauthorized } from 'lib/response';
|
import { ok, badRequest, methodNotAllowed, unauthorized } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
import { useCors } from 'lib/middleware';
|
import { useCors } from 'lib/middleware';
|
||||||
|
|
@ -34,7 +34,7 @@ export default async (req, res) => {
|
||||||
device,
|
device,
|
||||||
country,
|
country,
|
||||||
}),
|
}),
|
||||||
getPageviewStats(websiteId, startDate, endDate, tz, unit, 'distinct pageview.session_id', {
|
getPageviewStats(websiteId, startDate, endDate, tz, unit, 'distinct pageview.', {
|
||||||
url,
|
url,
|
||||||
os,
|
os,
|
||||||
browser,
|
browser,
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { resetWebsite } from 'lib/queries';
|
import { resetWebsite } from 'queries';
|
||||||
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getWebsiteStats } from 'lib/queries';
|
import { getWebsiteStats } from 'queries';
|
||||||
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
import { methodNotAllowed, ok, unauthorized } from 'lib/response';
|
||||||
import { allowQuery } from 'lib/auth';
|
import { allowQuery } from 'lib/auth';
|
||||||
import { useCors } from 'lib/middleware';
|
import { useCors } from 'lib/middleware';
|
||||||
|
|
@ -41,7 +41,7 @@ export default async (req, res) => {
|
||||||
const stats = Object.keys(metrics[0]).reduce((obj, key) => {
|
const stats = Object.keys(metrics[0]).reduce((obj, key) => {
|
||||||
obj[key] = {
|
obj[key] = {
|
||||||
value: Number(metrics[0][key]) || 0,
|
value: Number(metrics[0][key]) || 0,
|
||||||
change: Number(metrics[0][key] - prevPeriod[0][key]) || 0,
|
change: Number(metrics[0][key]) - Number(prevPeriod[0][key]) || 0,
|
||||||
};
|
};
|
||||||
return obj;
|
return obj;
|
||||||
}, {});
|
}, {});
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { updateWebsite, createWebsite, getWebsiteById } from 'lib/queries';
|
import { updateWebsite, createWebsite, getWebsiteById } from 'queries';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { uuid, getRandomChars } from 'lib/crypto';
|
import { uuid, getRandomChars } from 'lib/crypto';
|
||||||
import { ok, unauthorized, methodNotAllowed } from 'lib/response';
|
import { ok, unauthorized, methodNotAllowed } from 'lib/response';
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { getAllWebsites, getUserWebsites } from 'lib/queries';
|
import { getAllWebsites, getUserWebsites } from 'queries';
|
||||||
import { useAuth } from 'lib/middleware';
|
import { useAuth } from 'lib/middleware';
|
||||||
import { ok, methodNotAllowed, unauthorized } from 'lib/response';
|
import { ok, methodNotAllowed, unauthorized } from 'lib/response';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "كلمة مرور جديدة"
|
"value": "كلمة مرور جديدة"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "مجمعة"
|
"value": "مجمعة"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "نطاق فقط"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "التحويلات"
|
"value": "التحويلات"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "زائرون فريدون"
|
"value": "زائرون فريدون"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Contrasenya nova"
|
"value": "Contrasenya nova"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Combinat"
|
"value": "Combinat"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Només domini"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Referents"
|
"value": "Referents"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Visitants únics"
|
"value": "Visitants únics"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Nové heslo"
|
"value": "Nové heslo"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Kombinace"
|
"value": "Kombinace"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Domény"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Odkazy"
|
"value": "Odkazy"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Jedinečné návštěvy"
|
"value": "Jedinečné návštěvy"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -32,19 +32,19 @@
|
||||||
"label.all-events": [
|
"label.all-events": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "All events"
|
"value": "Alle hændelser"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.all-time": [
|
"label.all-time": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "All time"
|
"value": "Altid"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.all-websites": [
|
"label.all-websites": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Alle websites"
|
"value": "Alle hjemmesider"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.back": [
|
"label.back": [
|
||||||
|
|
@ -176,7 +176,7 @@
|
||||||
"label.language": [
|
"label.language": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Language"
|
"value": "Sprog"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.last-days": [
|
"label.last-days": [
|
||||||
|
|
@ -210,7 +210,7 @@
|
||||||
"label.logged-in-as": [
|
"label.logged-in-as": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Loggede ind som "
|
"value": "Logget ind som "
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": 1,
|
"type": 1,
|
||||||
|
|
@ -247,10 +247,16 @@
|
||||||
"value": "Ny adgangskode"
|
"value": "Ny adgangskode"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Owner"
|
"value": "Ejer"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.password": [
|
"label.password": [
|
||||||
|
|
@ -262,7 +268,7 @@
|
||||||
"label.passwords-dont-match": [
|
"label.passwords-dont-match": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Adgangskoder matcher ikke"
|
"value": "Adgangskoderne matcher ikke"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.profile": [
|
"label.profile": [
|
||||||
|
|
@ -298,13 +304,13 @@
|
||||||
"label.reset": [
|
"label.reset": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Reset"
|
"value": "Nulstil"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.reset-website": [
|
"label.reset-website": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Reset statistics"
|
"value": "Nulstil statistikker"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.save": [
|
"label.save": [
|
||||||
|
|
@ -334,7 +340,7 @@
|
||||||
"label.theme": [
|
"label.theme": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Theme"
|
"value": "Tema"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"label.this-month": [
|
"label.this-month": [
|
||||||
|
|
@ -448,7 +454,7 @@
|
||||||
"message.confirm-reset": [
|
"message.confirm-reset": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Are your sure you want to reset "
|
"value": "Er du sikker på at du ville nulstille "
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": 1,
|
"type": 1,
|
||||||
|
|
@ -456,7 +462,7 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "'s statistics?"
|
"value": "'s statistikker?"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"message.copied": [
|
"message.copied": [
|
||||||
|
|
@ -558,7 +564,7 @@
|
||||||
"message.no-websites-configured": [
|
"message.no-websites-configured": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Du har ikke konfigureret nogen websteder."
|
"value": "Du har ikke konfigureret nogen hjemmesider."
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"message.page-not-found": [
|
"message.page-not-found": [
|
||||||
|
|
@ -580,7 +586,7 @@
|
||||||
"message.reset-warning": [
|
"message.reset-warning": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "All statistics for this website will be deleted, but your tracking code will remain intact."
|
"value": "Alle statistikker for denne hjemmeside ville blive slettet, men sporingskode ville forblive intakt."
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"message.save-success": [
|
"message.save-success": [
|
||||||
|
|
@ -592,7 +598,7 @@
|
||||||
"message.share-url": [
|
"message.share-url": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Dette er den offentligt delings-URL til "
|
"value": "Dette er den offentlige delings-URL til "
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": 1,
|
"type": 1,
|
||||||
|
|
@ -606,7 +612,7 @@
|
||||||
"message.toggle-charts": [
|
"message.toggle-charts": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Toggle charts"
|
"value": "Ændre graf"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"message.track-stats": [
|
"message.track-stats": [
|
||||||
|
|
@ -628,7 +634,7 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": " sektionen på dit websted."
|
"value": " sektionen på din hjemmeside."
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"message.type-delete": [
|
"message.type-delete": [
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Kombineret"
|
"value": "Kombineret"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Kun domæne"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -746,7 +746,7 @@
|
||||||
"metrics.languages": [
|
"metrics.languages": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Languages"
|
"value": "Sprog"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.operating-systems": [
|
"metrics.operating-systems": [
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Henvisninger"
|
"value": "Henvisninger"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Unikke besøgende"
|
"value": "Unikke besøgende"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Neues Passwort"
|
"value": "Neues Passwort"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Kombiniert"
|
"value": "Kombiniert"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Nur diese Domain"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Referrer"
|
"value": "Referrer"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Bildschirmauflösungen"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Eindeutige Besucher"
|
"value": "Eindeutige Besucher"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Νέος κωδικός"
|
"value": "Νέος κωδικός"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Σε συνδυασμό"
|
"value": "Σε συνδυασμό"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Μόνο τομέας"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Παραπομπές"
|
"value": "Παραπομπές"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Μοναδικοί επισκέπτες"
|
"value": "Μοναδικοί επισκέπτες"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "New password"
|
"value": "New password"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Combined"
|
"value": "Combined"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Domain only"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Referrers"
|
"value": "Referrers"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Unique visitors"
|
"value": "Unique visitors"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "New password"
|
"value": "New password"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Combined"
|
"value": "Combined"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Domain only"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Referrers"
|
"value": "Referrers"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Unique visitors"
|
"value": "Unique visitors"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Nueva contraseña"
|
"value": "Nueva contraseña"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Combinado"
|
"value": "Combinado"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Únicamente dominio"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Referentes"
|
"value": "Referentes"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Visitantes únicos"
|
"value": "Visitantes únicos"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "رمز جدید"
|
"value": "رمز جدید"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -715,12 +721,6 @@
|
||||||
"value": "ترکیب شده"
|
"value": "ترکیب شده"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "فقط دامنه"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -757,12 +757,54 @@
|
||||||
"value": "ارجاع دهندگان"
|
"value": "ارجاع دهندگان"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "بازدیدکنندههای یکتا"
|
"value": "بازدیدکنندههای یکتا"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Uusi salasana"
|
"value": "Uusi salasana"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Yhdistetty"
|
"value": "Yhdistetty"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Vain verkkotunnus"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Viittaajat"
|
"value": "Viittaajat"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Yksittäiset kävijät"
|
"value": "Yksittäiset kävijät"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -247,6 +247,12 @@
|
||||||
"value": "Nýtt loyniorð"
|
"value": "Nýtt loyniorð"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "None"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -731,12 +737,6 @@
|
||||||
"value": "Samansett"
|
"value": "Samansett"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Bara økisnavn"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -773,12 +773,54 @@
|
||||||
"value": "Framsendingar"
|
"value": "Framsendingar"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Screens"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Einsýna vitjanir"
|
"value": "Einsýna vitjanir"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
|
|
@ -239,6 +239,12 @@
|
||||||
"value": "Nouveau mot de passe"
|
"value": "Nouveau mot de passe"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"label.none": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Aucun·e"
|
||||||
|
}
|
||||||
|
],
|
||||||
"label.owner": [
|
"label.owner": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -727,12 +733,6 @@
|
||||||
"value": "Combiné"
|
"value": "Combiné"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"metrics.filter.domain-only": [
|
|
||||||
{
|
|
||||||
"type": 0,
|
|
||||||
"value": "Domaine uniquement"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metrics.filter.raw": [
|
"metrics.filter.raw": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
@ -769,12 +769,54 @@
|
||||||
"value": "Sources"
|
"value": "Sources"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.screens": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "Tailles d'écran"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.unique-visitors": [
|
"metrics.unique-visitors": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
"value": "Visiteurs uniques"
|
"value": "Visiteurs uniques"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"metrics.utm": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_campaign": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Campaign"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_content": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Content"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_medium": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Medium"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_source": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Source"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"metrics.utm_term": [
|
||||||
|
{
|
||||||
|
"type": 0,
|
||||||
|
"value": "UTM Term"
|
||||||
|
}
|
||||||
|
],
|
||||||
"metrics.views": [
|
"metrics.views": [
|
||||||
{
|
{
|
||||||
"type": 0,
|
"type": 0,
|
||||||
|
|
|
||||||
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue