Compare commits
50 Commits
Author | SHA1 | Date |
---|---|---|
![]() |
732fc73fe6 | |
![]() |
586529a5ca | |
![]() |
22e2f79bff | |
![]() |
6f4cc9e84c | |
![]() |
e77e030f2c | |
![]() |
a43bbc51f6 | |
![]() |
bf26874965 | |
![]() |
6cdf9a68d1 | |
![]() |
402acf563e | |
![]() |
2d6077fe16 | |
![]() |
c9ec055996 | |
![]() |
3f0815855c | |
![]() |
46615fe7eb | |
![]() |
9345c9ac9e | |
![]() |
938047ce45 | |
![]() |
34757e7b2a | |
![]() |
01fe8b44a3 | |
![]() |
ae85bf696b | |
![]() |
f450fc35fb | |
![]() |
4c544361fd | |
![]() |
4ea3eec98c | |
![]() |
5864aee043 | |
![]() |
c49e2c6974 | |
![]() |
02f031bde1 | |
![]() |
3f1ed750f0 | |
![]() |
d73def80c7 | |
![]() |
4b72918a91 | |
![]() |
44a1c7fa10 | |
![]() |
c47d246a79 | |
![]() |
89d37e0045 | |
![]() |
4045cfd039 | |
![]() |
c3beb10e3b | |
![]() |
b6f3f57455 | |
![]() |
4876684461 | |
![]() |
9e11866ddd | |
![]() |
1e9a4ad08f | |
![]() |
92ab391ef8 | |
![]() |
c23373d164 | |
![]() |
1b8c2c4dde | |
![]() |
26a9cb67d8 | |
![]() |
4a51a5db3f | |
![]() |
8bc0fb92ba | |
![]() |
ff9a6335d1 | |
![]() |
c5bffb97cc | |
![]() |
2e3d2924e7 | |
![]() |
8666965930 | |
![]() |
5027d9d945 | |
![]() |
6cb2429ee1 | |
![]() |
ef59e93adc | |
![]() |
47f63d80c4 |
|
@ -0,0 +1,32 @@
|
|||
name: "🐛 Bug Report"
|
||||
description: Create a bug report for Umami.
|
||||
body:
|
||||
- type: textarea
|
||||
attributes:
|
||||
label: Describe the Bug
|
||||
description: A clear and concise description of what the bug is.
|
||||
validations:
|
||||
required: true
|
||||
- type: dropdown
|
||||
attributes:
|
||||
label: Database
|
||||
description: What database are you using?
|
||||
options:
|
||||
- PostgreSQL
|
||||
- MySQL
|
||||
- Umami Cloud
|
||||
validations:
|
||||
required: true
|
||||
- type: textarea
|
||||
attributes:
|
||||
label: Relevant log output
|
||||
description: Please copy and paste any relevant log output. This will be automatically formatted into code, so no need for backticks.
|
||||
render: shell
|
||||
- type: input
|
||||
attributes:
|
||||
label: Which browser are you using? (if relevant)
|
||||
description: 'For example: Chrome, Edge, Firefox, etc'
|
||||
- type: input
|
||||
attributes:
|
||||
label: How are you deploying your application? (if relevant)
|
||||
description: 'For example: Vercel, Railway, Docker, etc'
|
|
@ -0,0 +1,10 @@
|
|||
name: "✨ Feature Request"
|
||||
description: Create a feature or enhancement request for Umami.
|
||||
labels: ['enhancement']
|
||||
body:
|
||||
- type: textarea
|
||||
attributes:
|
||||
label: Describe the feature or enhancement
|
||||
description: A clear and concise description of what the feature or enhancement is.
|
||||
validations:
|
||||
required: true
|
|
@ -0,0 +1,6 @@
|
|||
|
||||
blank_issues_enabled: false
|
||||
contact_links:
|
||||
- name: "🤔 Ask a question"
|
||||
url: https://github.com/umami-software/umami/discussions
|
||||
about: Ask questions and discuss with other community members.
|
|
@ -1,6 +1,6 @@
|
|||
# umami
|
||||
|
||||
Umami is a simple, fast, privacy-focused alternative to Google Analytics.
|
||||
Umami MAMI is a simple, fast, privacy-focused alternative to Google Analytics.
|
||||
|
||||
## Getting started
|
||||
|
||||
|
|
|
@ -0,0 +1,33 @@
|
|||
/* eslint-disable no-console */
|
||||
import { ErrorBoundary as Boundary } from 'react-error-boundary';
|
||||
import { Button } from 'react-basics';
|
||||
import useMessages from 'hooks/useMessages';
|
||||
import styles from './ErrorBoundry.module.css';
|
||||
|
||||
const logError = (error, info) => {
|
||||
console.error(error, info.componentStack);
|
||||
};
|
||||
|
||||
export function ErrorBoundary({ children }) {
|
||||
const { formatMessage, messages } = useMessages();
|
||||
|
||||
const fallbackRender = ({ error, resetErrorBoundary }) => {
|
||||
console.log({ error });
|
||||
return (
|
||||
<div className={styles.error} role="alert">
|
||||
<h1>{formatMessage(messages.error)}</h1>
|
||||
<h3>{error.message}</h3>
|
||||
<pre>{error.stack}</pre>
|
||||
<Button onClick={resetErrorBoundary}>OK</Button>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
return (
|
||||
<Boundary fallbackRender={fallbackRender} onError={logError}>
|
||||
{children}
|
||||
</Boundary>
|
||||
);
|
||||
}
|
||||
|
||||
export default ErrorBoundary;
|
|
@ -0,0 +1,19 @@
|
|||
.error {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
margin: auto;
|
||||
z-index: var(--z-index-overlay);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
min-height: 600px;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.error button {
|
||||
align-self: center;
|
||||
}
|
|
@ -6,24 +6,23 @@ import usePageQuery from 'hooks/usePageQuery';
|
|||
import useMessages from 'hooks/useMessages';
|
||||
import styles from './FilterLink.module.css';
|
||||
|
||||
export function FilterLink({ id, value, label, externalUrl }) {
|
||||
export function FilterLink({ id, value, label, externalUrl, children, className }) {
|
||||
const { formatMessage, labels } = useMessages();
|
||||
const { resolveUrl, query } = usePageQuery();
|
||||
const active = query[id] !== undefined;
|
||||
const selected = query[id] === value;
|
||||
|
||||
return (
|
||||
<div className={styles.row}>
|
||||
{!value && `(${label || formatMessage(labels.unknown)})`}
|
||||
{value && (
|
||||
<Link
|
||||
href={resolveUrl({ [id]: value })}
|
||||
className={classNames(styles.label, {
|
||||
<div
|
||||
className={classNames(styles.row, className, {
|
||||
[styles.inactive]: active && !selected,
|
||||
[styles.active]: active && selected,
|
||||
})}
|
||||
replace
|
||||
>
|
||||
{children}
|
||||
{!value && `(${label || formatMessage(labels.unknown)})`}
|
||||
{value && (
|
||||
<Link href={resolveUrl({ [id]: value })} className={styles.label} replace>
|
||||
{safeDecodeURI(label || value)}
|
||||
</Link>
|
||||
)}
|
||||
|
|
|
@ -1,13 +1,18 @@
|
|||
.row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.row .inactive {
|
||||
.row.inactive {
|
||||
color: var(--base500);
|
||||
}
|
||||
|
||||
.row .active {
|
||||
.row.inactive img {
|
||||
opacity: 0.35;
|
||||
}
|
||||
|
||||
.row.active {
|
||||
color: var(--base900);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
|
|
@ -23,7 +23,7 @@ export function DateFilter({ websiteId, value, className }) {
|
|||
if (data) {
|
||||
setDateRange({ value, ...getDateRangeValues(new Date(data.createdAt), Date.now()) });
|
||||
}
|
||||
} else {
|
||||
} else if (value !== 'all') {
|
||||
setDateRange(value);
|
||||
}
|
||||
}
|
||||
|
@ -61,7 +61,7 @@ export function DateFilter({ websiteId, value, className }) {
|
|||
value: '90day',
|
||||
},
|
||||
{ label: formatMessage(labels.thisYear), value: '1year' },
|
||||
{
|
||||
websiteId && {
|
||||
label: formatMessage(labels.allTime),
|
||||
value: 'all',
|
||||
divider: true,
|
||||
|
@ -71,7 +71,7 @@ export function DateFilter({ websiteId, value, className }) {
|
|||
value: 'custom',
|
||||
divider: true,
|
||||
},
|
||||
];
|
||||
].filter(n => n);
|
||||
|
||||
const renderValue = value => {
|
||||
return value === 'custom' ? (
|
||||
|
|
|
@ -11,9 +11,14 @@ export function CountriesTable({ websiteId, ...props }) {
|
|||
|
||||
function renderLink({ x: code }) {
|
||||
return (
|
||||
<div className={locale}>
|
||||
<FilterLink id="country" value={countryNames[code] && code} label={countryNames[code]} />
|
||||
</div>
|
||||
<FilterLink
|
||||
id="country"
|
||||
className={locale}
|
||||
value={countryNames[code] && code}
|
||||
label={countryNames[code]}
|
||||
>
|
||||
<img src={`/images/flags/${code?.toLowerCase() || 'xx'}.png`} alt={code} />
|
||||
</FilterLink>
|
||||
);
|
||||
}
|
||||
|
||||
|
|
|
@ -15,11 +15,11 @@ export function RegionsTable({ websiteId, ...props }) {
|
|||
return regions[x] ? `${regions[x]}, ${countryNames[x.split('-')[0]]}` : x;
|
||||
};
|
||||
|
||||
const renderLink = ({ x }) => {
|
||||
const renderLink = ({ x: code }) => {
|
||||
return (
|
||||
<div className={locale}>
|
||||
<FilterLink id="region" value={x} label={renderLabel(x)} />
|
||||
</div>
|
||||
<FilterLink id="region" className={locale} value={code} label={renderLabel(code)}>
|
||||
<img src={`/images/flags/${code?.split('-')?.[0]?.toLowerCase() || 'xx'}.png`} alt={code} />
|
||||
</FilterLink>
|
||||
);
|
||||
};
|
||||
|
||||
|
|
|
@ -18,10 +18,15 @@ export function UsersList() {
|
|||
const { toast, showToast } = useToast();
|
||||
const hasData = data && data.length !== 0;
|
||||
|
||||
const handleSave = () => refetch();
|
||||
const handleSave = () => {
|
||||
refetch().then(() => showToast({ message: formatMessage(messages.saved), variant: 'success' }));
|
||||
};
|
||||
|
||||
const handleDelete = () =>
|
||||
showToast({ message: formatMessage(messages.userDeleted), variant: 'success' });
|
||||
const handleDelete = () => {
|
||||
refetch().then(() =>
|
||||
showToast({ message: formatMessage(messages.userDeleted), variant: 'success' }),
|
||||
);
|
||||
};
|
||||
|
||||
return (
|
||||
<Page loading={isLoading} error={error}>
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
{
|
||||
"label.access-code": "Access code",
|
||||
"label.actions": "اجراءات",
|
||||
"label.access-code": "كود الدعوة",
|
||||
"label.actions": "الإجراءات",
|
||||
"label.activity-log": "سجل الأحداث",
|
||||
"label.add-website": "إضافة موقع",
|
||||
"label.admin": "مدير",
|
||||
|
@ -59,7 +59,7 @@
|
|||
"label.more": "المزيد",
|
||||
"label.name": "الإسم",
|
||||
"label.new-password": "كلمة مرور جديدة",
|
||||
"label.none": "لا شيء",
|
||||
"label.none": "غير معرف",
|
||||
"label.operating-systems": "نظام التشغيل",
|
||||
"label.owner": "المالك",
|
||||
"label.page-views": "مشاهدات الصفحة",
|
||||
|
@ -109,8 +109,8 @@
|
|||
"label.users": "المستخدمين",
|
||||
"label.view": "عرض",
|
||||
"label.view-details": "عرض التفاصيل",
|
||||
"label.views": "مشاهدات",
|
||||
"label.visitors": "زوار",
|
||||
"label.views": "المشاهدات",
|
||||
"label.visitors": "الزوار",
|
||||
"label.website-id": "معرف الموقع",
|
||||
"label.websites": "المواقع",
|
||||
"label.yesterday": "الأمس",
|
||||
|
@ -136,7 +136,7 @@
|
|||
"message.saved": "تم الحفظ بنجاح.",
|
||||
"message.share-url": "هذا الرابط الذي تم مشاركته بشكل عام لـ {target}.",
|
||||
"message.team-already-member": "أنت عضو في المجموعة",
|
||||
"message.team-not-found": "لم يتم العثور على المجموعة found.",
|
||||
"message.team-not-found": "لم يتم العثور على المجموعة",
|
||||
"message.tracking-code": "كود التتبع",
|
||||
"message.user-deleted": "تم حذف المستخدم.",
|
||||
"message.visitor-log": "زائر من {country} يستخدم {browser} على {os} {device}",
|
||||
|
|
108
lang/es-MX.json
|
@ -1,39 +1,39 @@
|
|||
{
|
||||
"label.access-code": "Access code",
|
||||
"label.access-code": "Código de acceso",
|
||||
"label.actions": "Acciones",
|
||||
"label.activity-log": "Activity log",
|
||||
"label.add-website": "Agregar sitio",
|
||||
"label.activity-log": "Registro de actividad",
|
||||
"label.add-website": "Nuevo sitio web",
|
||||
"label.admin": "Administrador",
|
||||
"label.all": "Todos",
|
||||
"label.all-time": "Todos los tiempos",
|
||||
"label.analytics": "Analytics",
|
||||
"label.analytics": "Analíticas",
|
||||
"label.average-visit-time": "Tiempo promedio de visita",
|
||||
"label.back": "Atrás",
|
||||
"label.bounce-rate": "Porcentaje de rebote",
|
||||
"label.browsers": "Navegadores",
|
||||
"label.cancel": "Cancelar",
|
||||
"label.change-password": "Cambiar contraseña",
|
||||
"label.cities": "Cities",
|
||||
"label.clear-all": "Clear all",
|
||||
"label.confirm": "Confirm",
|
||||
"label.cities": "Ciudades",
|
||||
"label.clear-all": "Limpiar todo",
|
||||
"label.confirm": "Confirmar",
|
||||
"label.confirm-password": "Confirmar contraseña",
|
||||
"label.continue": "Continue",
|
||||
"label.continue": "Continuar",
|
||||
"label.countries": "Países",
|
||||
"label.create-team": "Create team",
|
||||
"label.create-user": "Create user",
|
||||
"label.created": "Created",
|
||||
"label.create-team": "Crear equipo",
|
||||
"label.create-user": "Crear usuario",
|
||||
"label.created": "Creado",
|
||||
"label.current-password": "Contraseña actual",
|
||||
"label.custom-range": "Intervalo personalizado",
|
||||
"label.dashboard": "Panel de control",
|
||||
"label.data": "Data",
|
||||
"label.date-range": "Fechas",
|
||||
"label.data": "Datos",
|
||||
"label.date-range": "Intervalo de fechas",
|
||||
"label.default-date-range": "Intervalo por defecto",
|
||||
"label.delete": "Eliminar",
|
||||
"label.delete-team": "Delete team",
|
||||
"label.delete-user": "Delete user",
|
||||
"label.delete-team": "Eliminar team",
|
||||
"label.delete-user": "Eliminar usuario",
|
||||
"label.delete-website": "Eliminar sitio",
|
||||
"label.desktop": "Escritorio",
|
||||
"label.details": "Details",
|
||||
"label.details": "Detalles",
|
||||
"label.devices": "Dispositivos",
|
||||
"label.dismiss": "Ignorar",
|
||||
"label.domain": "Dominio",
|
||||
|
@ -43,18 +43,18 @@
|
|||
"label.events": "Eventos",
|
||||
"label.filter-combined": "Combinado",
|
||||
"label.filter-raw": "Personalizado",
|
||||
"label.join": "Join",
|
||||
"label.join-team": "Join team",
|
||||
"label.join": "Unir",
|
||||
"label.join-team": "Unir a equipo",
|
||||
"label.language": "Idioma",
|
||||
"label.languages": "Idiomas",
|
||||
"label.laptop": "Portátil",
|
||||
"label.last-days": "Últimos {x} días",
|
||||
"label.last-hours": "Últimas {x} horas",
|
||||
"label.leave": "Leave",
|
||||
"label.leave-team": "Leave team",
|
||||
"label.leave": "Abandonar",
|
||||
"label.leave-team": "Abandonar equipo",
|
||||
"label.login": "Iniciar sesión",
|
||||
"label.logout": "Cerrar sesión",
|
||||
"label.members": "Members",
|
||||
"label.members": "Miembros",
|
||||
"label.mobile": "Móvil",
|
||||
"label.more": "Más",
|
||||
"label.name": "Nombre",
|
||||
|
@ -67,80 +67,80 @@
|
|||
"label.password": "Contraseña",
|
||||
"label.powered-by": "Analíticas de {name}",
|
||||
"label.profile": "Perfil",
|
||||
"label.queries": "Queries",
|
||||
"label.queries": "Consultas",
|
||||
"label.query-parameters": "Parámetros de petición",
|
||||
"label.realtime": "Tiempo real",
|
||||
"label.referrers": "Referido desde",
|
||||
"label.refresh": "Actualizar",
|
||||
"label.regenerate": "Regenerate",
|
||||
"label.regions": "Regions",
|
||||
"label.remove": "Remove",
|
||||
"label.regenerate": "Regenerar",
|
||||
"label.regions": "Regiones",
|
||||
"label.remove": "Quitar",
|
||||
"label.required": "Obligatorio",
|
||||
"label.reset": "Reiniciar",
|
||||
"label.reset-website": "Reiniciar estadísticas",
|
||||
"label.role": "Role",
|
||||
"label.role": "Rol",
|
||||
"label.save": "Guardar",
|
||||
"label.screens": "Pantallas",
|
||||
"label.select-website": "Select website",
|
||||
"label.sessions": "Sessions",
|
||||
"label.select-website": "Seleccionar sitio web",
|
||||
"label.sessions": "Sesiones",
|
||||
"label.settings": "Configuraciones",
|
||||
"label.share-url": "Compartir URL",
|
||||
"label.single-day": "Dia",
|
||||
"label.single-day": "Día",
|
||||
"label.tablet": "Tableta",
|
||||
"label.team": "Team",
|
||||
"label.team-guest": "Team guest",
|
||||
"label.team-id": "Team ID",
|
||||
"label.team-member": "Team member",
|
||||
"label.team-owner": "Team owner",
|
||||
"label.teams": "Teams",
|
||||
"label.team": "Equipo",
|
||||
"label.team-guest": "Invitado de equipo",
|
||||
"label.team-id": "ID de equipo",
|
||||
"label.team-member": "Miembro de equipo",
|
||||
"label.team-owner": "Admin. del equipo",
|
||||
"label.teams": "Equipos",
|
||||
"label.theme": "Tema",
|
||||
"label.this-month": "Este mes",
|
||||
"label.this-week": "Esta semana",
|
||||
"label.this-year": "Este año",
|
||||
"label.timezone": "Zona horaria",
|
||||
"label.title": "Title",
|
||||
"label.title": "Título",
|
||||
"label.today": "Hoy",
|
||||
"label.toggle-charts": "Alternar gráficas",
|
||||
"label.tracking-code": "Código de rastreo",
|
||||
"label.unique-visitors": "Visitantes únicos",
|
||||
"label.unknown": "Desconocida",
|
||||
"label.user": "User",
|
||||
"label.user": "Usuario",
|
||||
"label.username": "Nombre de usuario",
|
||||
"label.users": "Users",
|
||||
"label.view": "View",
|
||||
"label.users": "Usuarios",
|
||||
"label.view": "Visualizar",
|
||||
"label.view-details": "Ver detalles",
|
||||
"label.views": "Vistas",
|
||||
"label.visitors": "Visitantes",
|
||||
"label.website-id": "Website ID",
|
||||
"label.website-id": "ID del sitio web",
|
||||
"label.websites": "Sitios",
|
||||
"label.yesterday": "Ayer",
|
||||
"message.active-users": "{x} {x, plural, one {activo} other {activos}}",
|
||||
"message.confirm-delete": "¿Estás seguro(a) de querer eliminar {target}?",
|
||||
"message.confirm-leave": "Are you sure you want to leave {target}?",
|
||||
"message.confirm-reset": "¿Seguro que deseas restablecer las estadísticas de {target}?",
|
||||
"message.delete-website": "Eliminar sitio",
|
||||
"message.confirm-delete": "¿Seguro que quieres eliminar {target}?",
|
||||
"message.confirm-leave": "¿Seguro que quieres abandonar {target}?",
|
||||
"message.confirm-reset": "¿Seguro que quieres BORRAR las analíticas de {target}?",
|
||||
"message.delete-website": "Eliminar sitio web",
|
||||
"message.delete-website-warning": "Toda la información relacionada será eliminada.",
|
||||
"message.error": "Algo falló.",
|
||||
"message.event-log": "{event} on {url}",
|
||||
"message.event-log": "{event} en {url}",
|
||||
"message.go-to-settings": "Ir a la configuración",
|
||||
"message.incorrect-username-password": "Nombre de usuario o contraseña incorrectos.",
|
||||
"message.invalid-domain": "Dominio inválido",
|
||||
"message.min-password-length": "Minimum length of {n} characters",
|
||||
"message.min-password-length": "Longitud mínima de {n} caracteres",
|
||||
"message.no-data-available": "No hay información disponible.",
|
||||
"message.no-match-password": "Las contraseñas no coinciden",
|
||||
"message.no-teams": "You have not created any teams.",
|
||||
"message.no-users": "There are no users.",
|
||||
"message.no-teams": "No has creado ningún equipo.",
|
||||
"message.no-users": "No hay usuarios.",
|
||||
"message.page-not-found": "Página no encontrada",
|
||||
"message.reset-website": "Reiniciar estadísticas",
|
||||
"message.reset-website-warning": "Todas las estadísticas de esta página serán eliminadas, pero el código de rastreo permanecerá intacto.",
|
||||
"message.saved": "Guardado exitosamente.",
|
||||
"message.saved": "Guardado.",
|
||||
"message.share-url": "Esta es la URL compartida públicamente para {target}.",
|
||||
"message.team-already-member": "You are already a member of the team.",
|
||||
"message.team-not-found": "Team not found.",
|
||||
"message.team-already-member": "Ya eres miembro de este equipo.",
|
||||
"message.team-not-found": "Equipo no encontrado.",
|
||||
"message.tracking-code": "Código de rastreo",
|
||||
"message.user-deleted": "User deleted.",
|
||||
"message.user-deleted": "Usuario eliminado.",
|
||||
"message.visitor-log": "Visitante desde {country} usando {browser} en {os} {device}",
|
||||
"messages.no-team-websites": "This team does not have any websites.",
|
||||
"messages.no-team-websites": "Este equipo no tiene ningún sitio web configurado.",
|
||||
"messages.no-websites-configured": "No tienes ningún sitio configurado.",
|
||||
"messages.team-websites-info": "Websites can be viewed by anyone on the team."
|
||||
"messages.team-websites-info": "Las analíticas de tus sitios pueden verse por cualquier miembro del equipo."
|
||||
}
|
||||
|
|
|
@ -1,27 +1,27 @@
|
|||
{
|
||||
"label.access-code": "Access code",
|
||||
"label.access-code": "Kod dostępu",
|
||||
"label.actions": "Działania",
|
||||
"label.activity-log": "Activity log",
|
||||
"label.activity-log": "Dziennik aktywności",
|
||||
"label.add-website": "Dodaj witrynę",
|
||||
"label.admin": "Administrator",
|
||||
"label.all": "Wszystkie",
|
||||
"label.all-time": "Cały czas",
|
||||
"label.analytics": "Analytics",
|
||||
"label.analytics": "Analityka",
|
||||
"label.average-visit-time": "Średni czas wizyty",
|
||||
"label.back": "Powrót",
|
||||
"label.bounce-rate": "Współczynnik odrzuceń",
|
||||
"label.browsers": "Przeglądarki",
|
||||
"label.cancel": "Anuluj",
|
||||
"label.change-password": "Zmień hasło",
|
||||
"label.cities": "Cities",
|
||||
"label.clear-all": "Clear all",
|
||||
"label.confirm": "Confirm",
|
||||
"label.cities": "Miasta",
|
||||
"label.clear-all": "Wyczyść wszystko",
|
||||
"label.confirm": "Potwierdź",
|
||||
"label.confirm-password": "Potwierdź hasło",
|
||||
"label.continue": "Continue",
|
||||
"label.continue": "Kontynuuj",
|
||||
"label.countries": "Kraje",
|
||||
"label.create-team": "Create team",
|
||||
"label.create-user": "Create user",
|
||||
"label.created": "Created",
|
||||
"label.create-team": "Utwórz zespół",
|
||||
"label.create-user": "Utwórz użytkownika",
|
||||
"label.created": "Utworzony",
|
||||
"label.current-password": "Aktualne hasło",
|
||||
"label.custom-range": "Zakres niestandardowy",
|
||||
"label.dashboard": "Panel",
|
||||
|
@ -29,11 +29,11 @@
|
|||
"label.date-range": "Zakres dat",
|
||||
"label.default-date-range": "Domyślny zakres dat",
|
||||
"label.delete": "Usuń",
|
||||
"label.delete-team": "Delete team",
|
||||
"label.delete-user": "Delete user",
|
||||
"label.delete-team": "Usuń zespół",
|
||||
"label.delete-user": "Usuń użytkownika",
|
||||
"label.delete-website": "Usuń witrynę",
|
||||
"label.desktop": "Komputer",
|
||||
"label.details": "Details",
|
||||
"label.details": "Szczegóły",
|
||||
"label.devices": "Urządzenia",
|
||||
"label.dismiss": "Odrzuć",
|
||||
"label.domain": "Domena",
|
||||
|
@ -43,18 +43,18 @@
|
|||
"label.events": "Zdarzenia",
|
||||
"label.filter-combined": "Połączone",
|
||||
"label.filter-raw": "Surowe dane",
|
||||
"label.join": "Join",
|
||||
"label.join-team": "Join team",
|
||||
"label.join": "Dołącz",
|
||||
"label.join-team": "Dołącz do zespołu",
|
||||
"label.language": "Język",
|
||||
"label.languages": "Języki",
|
||||
"label.laptop": "Laptop",
|
||||
"label.last-days": "Ostatnie {x} dni",
|
||||
"label.last-hours": "Ostatnie {x} godzin",
|
||||
"label.leave": "Leave",
|
||||
"label.leave-team": "Leave team",
|
||||
"label.leave": "Opuść",
|
||||
"label.leave-team": "Opuść zespół",
|
||||
"label.login": "Zaloguj się",
|
||||
"label.logout": "Wyloguj",
|
||||
"label.members": "Members",
|
||||
"label.members": "Członkowie",
|
||||
"label.mobile": "Smartfon",
|
||||
"label.more": "Więcej",
|
||||
"label.name": "Nazwa",
|
||||
|
@ -67,80 +67,80 @@
|
|||
"label.password": "Hasło",
|
||||
"label.powered-by": "Obsługiwane przez {name}",
|
||||
"label.profile": "Profil",
|
||||
"label.queries": "Queries",
|
||||
"label.queries": "Zapytania",
|
||||
"label.query-parameters": "Parametry query",
|
||||
"label.realtime": "Czas rzeczywisty",
|
||||
"label.referrers": "Źródła odsyłające",
|
||||
"label.refresh": "Odśwież",
|
||||
"label.regenerate": "Regenerate",
|
||||
"label.regions": "Regions",
|
||||
"label.remove": "Remove",
|
||||
"label.regenerate": "Wygeneruj ponownie",
|
||||
"label.regions": "Regiony",
|
||||
"label.remove": "Usuń",
|
||||
"label.required": "Wymagany",
|
||||
"label.reset": "Zresetuj",
|
||||
"label.reset-website": "Zresetuj statystyki",
|
||||
"label.role": "Role",
|
||||
"label.save": "Zapisz",
|
||||
"label.screens": "Ekrany",
|
||||
"label.select-website": "Select website",
|
||||
"label.sessions": "Sessions",
|
||||
"label.select-website": "Wybierz witrynę",
|
||||
"label.sessions": "Sesje",
|
||||
"label.settings": "Ustawienia",
|
||||
"label.share-url": "Udostępnij adres URL",
|
||||
"label.single-day": "W tym dniu",
|
||||
"label.tablet": "Tablet",
|
||||
"label.team": "Team",
|
||||
"label.team-guest": "Team guest",
|
||||
"label.team-id": "Team ID",
|
||||
"label.team-member": "Team member",
|
||||
"label.team-owner": "Team owner",
|
||||
"label.teams": "Teams",
|
||||
"label.team": "Zespół",
|
||||
"label.team-guest": "Gość zespołu",
|
||||
"label.team-id": "ID zespołu",
|
||||
"label.team-member": "Członek zespołu",
|
||||
"label.team-owner": "Właściciel zespołu",
|
||||
"label.teams": "Zespoły",
|
||||
"label.theme": "Motyw",
|
||||
"label.this-month": "W tym miesiącu",
|
||||
"label.this-week": "W tym tygodniu",
|
||||
"label.this-year": "W tym roku",
|
||||
"label.timezone": "Strefa czasowa",
|
||||
"label.title": "Title",
|
||||
"label.title": "Tytuł",
|
||||
"label.today": "Dzisiaj",
|
||||
"label.toggle-charts": "Przełącz wykresy",
|
||||
"label.tracking-code": "Kod śledzenia",
|
||||
"label.unique-visitors": "Unikalni odwiedzający",
|
||||
"label.unknown": "Nieznany",
|
||||
"label.user": "User",
|
||||
"label.user": "Użytkownik",
|
||||
"label.username": "Nazwa użytkownika",
|
||||
"label.users": "Users",
|
||||
"label.view": "View",
|
||||
"label.users": "Użytkownicy",
|
||||
"label.view": "Zobacz",
|
||||
"label.view-details": "Pokaż szczegóły",
|
||||
"label.views": "Wyświetlenia",
|
||||
"label.visitors": "Odwiedzający",
|
||||
"label.website-id": "Website ID",
|
||||
"label.website-id": "ID witryny",
|
||||
"label.websites": "Witryny",
|
||||
"label.yesterday": "Wczoraj",
|
||||
"message.active-users": "{x} aktualnie {x, plural, one {odwiedzający} other {odwiedzających}}",
|
||||
"message.confirm-delete": "Czy na pewno chcesz usunąć {target}?",
|
||||
"message.confirm-leave": "Are you sure you want to leave {target}?",
|
||||
"message.confirm-leave": "Czy na pewno chcesz opuścić {target}?",
|
||||
"message.confirm-reset": "Czy na pewno chcesz zresetować statystyki {target}?",
|
||||
"message.delete-website": "Usuń witrynę",
|
||||
"message.delete-website-warning": "Wszystkie powiązane dane również zostaną usunięte.",
|
||||
"message.error": "Coś poszło nie tak.",
|
||||
"message.event-log": "{event} on {url}",
|
||||
"message.event-log": "{event} na {url}",
|
||||
"message.go-to-settings": "Przejdź do ustawień",
|
||||
"message.incorrect-username-password": "Nieprawidłowa nazwa użytkownika/hasło.",
|
||||
"message.invalid-domain": "Nieprawidłowa witryna",
|
||||
"message.min-password-length": "Minimum length of {n} characters",
|
||||
"message.min-password-length": "Minimalna długość {n} znaków",
|
||||
"message.no-data-available": "Brak dostępnych danych.",
|
||||
"message.no-match-password": "Hasła się nie zgadzają",
|
||||
"message.no-teams": "You have not created any teams.",
|
||||
"message.no-users": "There are no users.",
|
||||
"message.no-teams": "Nie stworzyłeś żadnych zespołów.",
|
||||
"message.no-users": "Nie ma żadnych użytkowników.",
|
||||
"message.page-not-found": "Strona nie znaleziona.",
|
||||
"message.reset-website": "Zresetuj statystyki",
|
||||
"message.reset-website-warning": "Wszystkie statystyki tej witryny zostaną usunięte, ale kod śledzenia pozostanie nienaruszony.",
|
||||
"message.saved": "Zapisano pomyślnie.",
|
||||
"message.share-url": "To jest publicznie udostępniany adres URL dla {target}.",
|
||||
"message.team-already-member": "You are already a member of the team.",
|
||||
"message.team-not-found": "Team not found.",
|
||||
"message.team-already-member": "Jesteś już członkiem zespołu.",
|
||||
"message.team-not-found": "Nie znaleziono zespołu.",
|
||||
"message.tracking-code": "Kod śledzenia",
|
||||
"message.user-deleted": "User deleted.",
|
||||
"message.user-deleted": "Użytkownik usunięty.",
|
||||
"message.visitor-log": "Odwiedzający z {country} używa {browser} na {os} {device}",
|
||||
"messages.no-team-websites": "This team does not have any websites.",
|
||||
"messages.no-team-websites": "Ten zespół nie ma żadnych witryn internetowych.",
|
||||
"messages.no-websites-configured": "Nie masz skonfigurowanych żadnych witryn internetowych.",
|
||||
"messages.team-websites-info": "Websites can be viewed by anyone on the team."
|
||||
"messages.team-websites-info": "Strony internetowe mogą być przeglądane przez każdego członka zespołu."
|
||||
}
|
||||
|
|
|
@ -53,7 +53,7 @@ export function parseDateRange(value, locale = 'en-US') {
|
|||
|
||||
const match = value.match(/^(?<num>[0-9-]+)(?<unit>hour|day|week|month|year)$/);
|
||||
|
||||
if (!match) return;
|
||||
if (!match) return {};
|
||||
|
||||
const { num, unit } = match.groups;
|
||||
|
||||
|
|
|
@ -56,12 +56,24 @@ export function getDevice(screen, os) {
|
|||
}
|
||||
}
|
||||
|
||||
export async function getLocation(ip) {
|
||||
export async function getLocation(ip, req) {
|
||||
// Ignore local ips
|
||||
if (await isLocalhost(ip)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (req.headers['x-vercel-ip-country']) {
|
||||
const country = req.headers['x-vercel-ip-country'];
|
||||
const region = req.headers['x-vercel-ip-country-region'];
|
||||
const city = req.headers['x-vercel-ip-city'];
|
||||
|
||||
return {
|
||||
country,
|
||||
subdivision1: region,
|
||||
city: city ? decodeURIComponent(city) : undefined,
|
||||
};
|
||||
}
|
||||
|
||||
// Database lookup
|
||||
if (!lookup) {
|
||||
const dir = path.join(process.cwd(), 'geo');
|
||||
|
@ -70,18 +82,21 @@ export async function getLocation(ip) {
|
|||
}
|
||||
|
||||
const result = lookup.get(ip);
|
||||
const country = result?.country?.iso_code ?? result?.registered_country?.iso_code;
|
||||
const subdivision1 = result?.subdivisions?.[0]?.iso_code;
|
||||
const subdivision2 = result?.subdivisions?.[1]?.names?.en;
|
||||
const city = result?.city?.names?.en;
|
||||
|
||||
return { country, subdivision1, subdivision2, city };
|
||||
if (result) {
|
||||
return {
|
||||
country: result.country?.iso_code ?? result?.registered_country?.iso_code,
|
||||
subdivision1: result.subdivisions?.[0]?.iso_code,
|
||||
subdivision2: result.subdivisions?.[1]?.names?.en,
|
||||
city: result.city?.names?.en,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
export async function getClientInfo(req: NextApiRequestCollect, { screen }) {
|
||||
const userAgent = req.headers['user-agent'];
|
||||
const ip = getIpAddress(req);
|
||||
const location = await getLocation(ip);
|
||||
const location = await getLocation(ip, req);
|
||||
const country = location?.country;
|
||||
const subdivision1 = location?.subdivision1;
|
||||
const subdivision2 = location?.subdivision2;
|
||||
|
|
|
@ -20,6 +20,7 @@ export const useCors = createMiddleware(
|
|||
);
|
||||
|
||||
export const useSession = createMiddleware(async (req, res, next) => {
|
||||
try {
|
||||
const session = await findSession(req as NextApiRequestCollect);
|
||||
|
||||
if (!session) {
|
||||
|
@ -28,6 +29,10 @@ export const useSession = createMiddleware(async (req, res, next) => {
|
|||
}
|
||||
|
||||
(req as any).session = session;
|
||||
} catch (e: any) {
|
||||
return badRequest(res, e.message);
|
||||
}
|
||||
|
||||
next();
|
||||
});
|
||||
|
||||
|
|
|
@ -11,7 +11,7 @@ export async function findSession(req: NextApiRequestCollect) {
|
|||
const { payload } = getJsonBody<CollectRequestBody>(req);
|
||||
|
||||
if (!payload) {
|
||||
return null;
|
||||
throw new Error('Invalid payload.');
|
||||
}
|
||||
|
||||
// Check if cache token is passed
|
||||
|
@ -29,14 +29,14 @@ export async function findSession(req: NextApiRequestCollect) {
|
|||
const { website: websiteId, hostname, screen, language } = payload;
|
||||
|
||||
if (!validate(websiteId)) {
|
||||
return null;
|
||||
throw new Error('Invalid website ID.');
|
||||
}
|
||||
|
||||
// Find website
|
||||
const website = await loadWebsite(websiteId);
|
||||
|
||||
if (!website) {
|
||||
throw new Error(`Website not found: ${websiteId}`);
|
||||
throw new Error(`Website not found: ${websiteId}.`);
|
||||
}
|
||||
|
||||
const { userAgent, browser, os, ip, country, subdivision1, subdivision2, city, device } =
|
||||
|
|
|
@ -65,10 +65,10 @@ const redirects = [
|
|||
},
|
||||
];
|
||||
|
||||
if (process.env.CLOUD_MODE) {
|
||||
if (process.env.CLOUD_MODE && process.env.DISABLE_LOGIN && process.env.CLOUD_URL) {
|
||||
redirects.push({
|
||||
source: '/login',
|
||||
destination: CLOUD_URL,
|
||||
destination: process.env.CLOUD_URL,
|
||||
permanent: false,
|
||||
});
|
||||
}
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
{
|
||||
"name": "umami",
|
||||
"version": "2.1.0",
|
||||
"version": "2.2.0",
|
||||
"description": "A simple, fast, privacy-focused alternative to Google Analytics.",
|
||||
"author": "Mike Cao <mike@mikecao.com>",
|
||||
"license": "MIT",
|
||||
|
@ -32,7 +32,7 @@
|
|||
"merge-messages": "node scripts/merge-messages.js",
|
||||
"generate-lang": "npm-run-all extract-messages merge-messages",
|
||||
"format-lang": "node scripts/format-lang.js",
|
||||
"compile-lang": "formatjs compile-folder --ast build public/intl/messages",
|
||||
"compile-lang": "formatjs compile-folder --ast build/messages public/intl/messages",
|
||||
"check-lang": "node scripts/check-lang.js",
|
||||
"download-country-names": "node scripts/download-country-names.js",
|
||||
"download-language-names": "node scripts/download-language-names.js",
|
||||
|
@ -97,6 +97,7 @@
|
|||
"react-basics": "^0.77.0",
|
||||
"react-beautiful-dnd": "^13.1.0",
|
||||
"react-dom": "^18.2.0",
|
||||
"react-error-boundary": "^4.0.4",
|
||||
"react-intl": "^5.24.7",
|
||||
"react-simple-maps": "^2.3.0",
|
||||
"react-spring": "^9.4.4",
|
||||
|
|
|
@ -3,6 +3,7 @@ import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
|
|||
import Head from 'next/head';
|
||||
import Script from 'next/script';
|
||||
import { useRouter } from 'next/router';
|
||||
import ErrorBoundary from 'components/common/ErrorBoundary';
|
||||
import useLocale from 'hooks/useLocale';
|
||||
import useConfig from 'hooks/useConfig';
|
||||
import '@fontsource/inter/400.css';
|
||||
|
@ -41,11 +42,26 @@ export default function App({ Component, pageProps }) {
|
|||
textComponent={Wrapper}
|
||||
onError={() => null}
|
||||
>
|
||||
<ErrorBoundary>
|
||||
<Head>
|
||||
<link rel="icon" href={`${basePath}/favicon.ico`} />
|
||||
<link rel="apple-touch-icon" sizes="180x180" href={`${basePath}/apple-touch-icon.png`} />
|
||||
<link rel="icon" type="image/png" sizes="32x32" href={`${basePath}/favicon-32x32.png`} />
|
||||
<link rel="icon" type="image/png" sizes="16x16" href={`${basePath}/favicon-16x16.png`} />
|
||||
<link
|
||||
rel="apple-touch-icon"
|
||||
sizes="180x180"
|
||||
href={`${basePath}/apple-touch-icon.png`}
|
||||
/>
|
||||
<link
|
||||
rel="icon"
|
||||
type="image/png"
|
||||
sizes="32x32"
|
||||
href={`${basePath}/favicon-32x32.png`}
|
||||
/>
|
||||
<link
|
||||
rel="icon"
|
||||
type="image/png"
|
||||
sizes="16x16"
|
||||
href={`${basePath}/favicon-16x16.png`}
|
||||
/>
|
||||
<link rel="manifest" href={`${basePath}/site.webmanifest`} />
|
||||
<link rel="mask-icon" href={`${basePath}/safari-pinned-tab.svg`} color="#5bbad5" />
|
||||
<meta name="msapplication-TileColor" content="#da532c" />
|
||||
|
@ -56,6 +72,7 @@ export default function App({ Component, pageProps }) {
|
|||
</Head>
|
||||
<Component {...pageProps} />
|
||||
{!pathname.includes('/share/') && <Script src={`${basePath}/telemetry.js`} />}
|
||||
</ErrorBoundary>
|
||||
</IntlProvider>
|
||||
</QueryClientProvider>
|
||||
);
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
import debug from 'debug';
|
||||
import { NextApiResponse } from 'next';
|
||||
import {
|
||||
ok,
|
||||
|
@ -13,6 +14,8 @@ import { secret } from 'lib/crypto';
|
|||
import { NextApiRequestQueryBody, User } from 'lib/types';
|
||||
import { setAuthKey } from 'lib/auth';
|
||||
|
||||
const log = debug('umami:auth');
|
||||
|
||||
export interface LoginRequestBody {
|
||||
username: string;
|
||||
password: string;
|
||||
|
@ -51,6 +54,8 @@ export default async (
|
|||
});
|
||||
}
|
||||
|
||||
log('Login failed:', { username, user });
|
||||
|
||||
return unauthorized(res, 'message.incorrect-username-password');
|
||||
}
|
||||
|
||||
|
|
|
@ -16,7 +16,7 @@ export default function LoginPage({ disabled }) {
|
|||
export async function getServerSideProps() {
|
||||
return {
|
||||
props: {
|
||||
disabled: !!(process.env.DISABLE_LOGIN || process.env.CLOUD_MODE),
|
||||
disabled: !!process.env.DISABLE_LOGIN,
|
||||
},
|
||||
};
|
||||
}
|
||||
|
|
After Width: | Height: | Size: 235 B |
After Width: | Height: | Size: 122 B |
After Width: | Height: | Size: 296 B |
After Width: | Height: | Size: 281 B |
After Width: | Height: | Size: 245 B |
After Width: | Height: | Size: 183 B |
After Width: | Height: | Size: 110 B |
After Width: | Height: | Size: 201 B |
After Width: | Height: | Size: 186 B |
After Width: | Height: | Size: 141 B |
After Width: | Height: | Size: 237 B |
After Width: | Height: | Size: 102 B |
After Width: | Height: | Size: 211 B |
After Width: | Height: | Size: 139 B |
After Width: | Height: | Size: 163 B |
After Width: | Height: | Size: 150 B |
After Width: | Height: | Size: 175 B |
After Width: | Height: | Size: 149 B |
After Width: | Height: | Size: 129 B |
After Width: | Height: | Size: 105 B |
After Width: | Height: | Size: 140 B |
After Width: | Height: | Size: 97 B |
After Width: | Height: | Size: 153 B |
After Width: | Height: | Size: 264 B |
After Width: | Height: | Size: 107 B |
After Width: | Height: | Size: 353 B |
After Width: | Height: | Size: 286 B |
After Width: | Height: | Size: 350 B |
After Width: | Height: | Size: 145 B |
After Width: | Height: | Size: 321 B |
After Width: | Height: | Size: 260 B |
After Width: | Height: | Size: 147 B |
After Width: | Height: | Size: 316 B |
After Width: | Height: | Size: 150 B |
After Width: | Height: | Size: 114 B |
After Width: | Height: | Size: 145 B |
After Width: | Height: | Size: 253 B |
After Width: | Height: | Size: 171 B |
After Width: | Height: | Size: 211 B |
After Width: | Height: | Size: 233 B |
After Width: | Height: | Size: 206 B |
After Width: | Height: | Size: 164 B |
After Width: | Height: | Size: 129 B |
After Width: | Height: | Size: 114 B |
After Width: | Height: | Size: 230 B |
After Width: | Height: | Size: 144 B |
After Width: | Height: | Size: 135 B |
After Width: | Height: | Size: 144 B |
After Width: | Height: | Size: 112 B |
After Width: | Height: | Size: 142 B |
After Width: | Height: | Size: 163 B |
After Width: | Height: | Size: 146 B |
After Width: | Height: | Size: 154 B |
After Width: | Height: | Size: 240 B |
After Width: | Height: | Size: 196 B |
After Width: | Height: | Size: 196 B |
After Width: | Height: | Size: 97 B |
After Width: | Height: | Size: 234 B |
After Width: | Height: | Size: 127 B |
After Width: | Height: | Size: 200 B |
After Width: | Height: | Size: 164 B |
After Width: | Height: | Size: 160 B |
After Width: | Height: | Size: 237 B |
After Width: | Height: | Size: 117 B |
After Width: | Height: | Size: 160 B |
After Width: | Height: | Size: 230 B |
After Width: | Height: | Size: 204 B |
After Width: | Height: | Size: 190 B |
After Width: | Height: | Size: 215 B |
After Width: | Height: | Size: 121 B |
After Width: | Height: | Size: 257 B |
After Width: | Height: | Size: 285 B |
After Width: | Height: | Size: 156 B |
After Width: | Height: | Size: 147 B |
After Width: | Height: | Size: 105 B |
After Width: | Height: | Size: 98 B |