feature(dashboard,api): add timezone support
* feat(dashboard): add support for today, yesterday etc (timezones) * fix(db): escape js dates * fix(dashboard): ensure we support default timezone * final fixes * remove complete series and add sql with fill instead
This commit is contained in:
committed by
GitHub
parent
46bfeee131
commit
680727355b
@@ -1,5 +1,5 @@
|
|||||||
import Chat from '@/components/chat/chat';
|
import Chat from '@/components/chat/chat';
|
||||||
import { db, getOrganizationBySlug } from '@openpanel/db';
|
import { db, getOrganizationById } from '@openpanel/db';
|
||||||
import type { UIMessage } from 'ai';
|
import type { UIMessage } from 'ai';
|
||||||
|
|
||||||
export default async function ChatPage({
|
export default async function ChatPage({
|
||||||
@@ -9,7 +9,7 @@ export default async function ChatPage({
|
|||||||
}) {
|
}) {
|
||||||
const { projectId } = await params;
|
const { projectId } = await params;
|
||||||
const [organization, chat] = await Promise.all([
|
const [organization, chat] = await Promise.all([
|
||||||
getOrganizationBySlug(params.organizationSlug),
|
getOrganizationById(params.organizationSlug),
|
||||||
db.chat.findFirst({
|
db.chat.findFirst({
|
||||||
where: {
|
where: {
|
||||||
projectId,
|
projectId,
|
||||||
|
|||||||
@@ -24,7 +24,6 @@ import { Sheet, SheetContent, SheetTrigger } from '@/components/ui/sheet';
|
|||||||
import { useAppParams } from '@/hooks/useAppParams';
|
import { useAppParams } from '@/hooks/useAppParams';
|
||||||
import { useDispatch, useSelector } from '@/redux';
|
import { useDispatch, useSelector } from '@/redux';
|
||||||
import { bind } from 'bind-event-listener';
|
import { bind } from 'bind-event-listener';
|
||||||
import { endOfDay, startOfDay } from 'date-fns';
|
|
||||||
import { GanttChartSquareIcon } from 'lucide-react';
|
import { GanttChartSquareIcon } from 'lucide-react';
|
||||||
import { useEffect } from 'react';
|
import { useEffect } from 'react';
|
||||||
|
|
||||||
@@ -89,12 +88,8 @@ export default function ReportEditor({
|
|||||||
dispatch(changeDateRanges(value));
|
dispatch(changeDateRanges(value));
|
||||||
}}
|
}}
|
||||||
value={report.range}
|
value={report.range}
|
||||||
onStartDateChange={(date) =>
|
onStartDateChange={(date) => dispatch(changeStartDate(date))}
|
||||||
dispatch(changeStartDate(startOfDay(date).toISOString()))
|
onEndDateChange={(date) => dispatch(changeEndDate(date))}
|
||||||
}
|
|
||||||
onEndDateChange={(date) =>
|
|
||||||
dispatch(changeEndDate(endOfDay(date).toISOString()))
|
|
||||||
}
|
|
||||||
endDate={report.endDate}
|
endDate={report.endDate}
|
||||||
startDate={report.startDate}
|
startDate={report.startDate}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -1,20 +1,19 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { InputWithLabel } from '@/components/forms/input-with-label';
|
import { InputWithLabel, WithLabel } from '@/components/forms/input-with-label';
|
||||||
import { Button } from '@/components/ui/button';
|
import { Button } from '@/components/ui/button';
|
||||||
import { Widget, WidgetBody, WidgetHead } from '@/components/widget';
|
import { Widget, WidgetBody, WidgetHead } from '@/components/widget';
|
||||||
import { api, handleError } from '@/trpc/client';
|
import { api, handleError } from '@/trpc/client';
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
import { useForm } from 'react-hook-form';
|
import { Controller, useForm } from 'react-hook-form';
|
||||||
import { toast } from 'sonner';
|
import { toast } from 'sonner';
|
||||||
import { z } from 'zod';
|
import type { z } from 'zod';
|
||||||
|
|
||||||
|
import { Combobox } from '@/components/ui/combobox';
|
||||||
import type { IServiceOrganization } from '@openpanel/db';
|
import type { IServiceOrganization } from '@openpanel/db';
|
||||||
|
import { zEditOrganization } from '@openpanel/validation';
|
||||||
|
|
||||||
const validator = z.object({
|
const validator = zEditOrganization;
|
||||||
id: z.string().min(2),
|
|
||||||
name: z.string().min(2),
|
|
||||||
});
|
|
||||||
|
|
||||||
type IForm = z.infer<typeof validator>;
|
type IForm = z.infer<typeof validator>;
|
||||||
interface EditOrganizationProps {
|
interface EditOrganizationProps {
|
||||||
@@ -25,8 +24,12 @@ export default function EditOrganization({
|
|||||||
}: EditOrganizationProps) {
|
}: EditOrganizationProps) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const { register, handleSubmit, formState, reset } = useForm<IForm>({
|
const { register, handleSubmit, formState, reset, control } = useForm<IForm>({
|
||||||
defaultValues: organization ?? undefined,
|
defaultValues: {
|
||||||
|
id: organization.id,
|
||||||
|
name: organization.name,
|
||||||
|
timezone: organization.timezone ?? undefined,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const mutation = api.organization.update.useMutation({
|
const mutation = api.organization.update.useMutation({
|
||||||
@@ -34,7 +37,10 @@ export default function EditOrganization({
|
|||||||
toast('Organization updated', {
|
toast('Organization updated', {
|
||||||
description: 'Your organization has been updated.',
|
description: 'Your organization has been updated.',
|
||||||
});
|
});
|
||||||
reset(res);
|
reset({
|
||||||
|
...res,
|
||||||
|
timezone: res.timezone!,
|
||||||
|
});
|
||||||
router.refresh();
|
router.refresh();
|
||||||
},
|
},
|
||||||
onError: handleError,
|
onError: handleError,
|
||||||
@@ -50,14 +56,37 @@ export default function EditOrganization({
|
|||||||
<WidgetHead className="flex items-center justify-between">
|
<WidgetHead className="flex items-center justify-between">
|
||||||
<span className="title">Details</span>
|
<span className="title">Details</span>
|
||||||
</WidgetHead>
|
</WidgetHead>
|
||||||
<WidgetBody className="flex items-end gap-2">
|
<WidgetBody className="gap-4 col">
|
||||||
<InputWithLabel
|
<InputWithLabel
|
||||||
className="flex-1"
|
className="flex-1"
|
||||||
label="Name"
|
label="Name"
|
||||||
{...register('name')}
|
{...register('name')}
|
||||||
defaultValue={organization?.name}
|
defaultValue={organization?.name}
|
||||||
/>
|
/>
|
||||||
<Button size="sm" type="submit" disabled={!formState.isDirty}>
|
<Controller
|
||||||
|
name="timezone"
|
||||||
|
control={control}
|
||||||
|
render={({ field }) => (
|
||||||
|
<WithLabel label="Timezone">
|
||||||
|
<Combobox
|
||||||
|
placeholder="Select timezone"
|
||||||
|
items={Intl.supportedValuesOf('timeZone').map((item) => ({
|
||||||
|
value: item,
|
||||||
|
label: item,
|
||||||
|
}))}
|
||||||
|
value={field.value}
|
||||||
|
onChange={field.onChange}
|
||||||
|
className="w-full"
|
||||||
|
/>
|
||||||
|
</WithLabel>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
disabled={!formState.isDirty}
|
||||||
|
className="self-end"
|
||||||
|
>
|
||||||
Save
|
Save
|
||||||
</Button>
|
</Button>
|
||||||
</WidgetBody>
|
</WidgetBody>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import { notFound } from 'next/navigation';
|
|||||||
import { parseAsStringEnum } from 'nuqs/server';
|
import { parseAsStringEnum } from 'nuqs/server';
|
||||||
|
|
||||||
import { auth } from '@openpanel/auth/nextjs';
|
import { auth } from '@openpanel/auth/nextjs';
|
||||||
import { db, transformOrganization } from '@openpanel/db';
|
import { db } from '@openpanel/db';
|
||||||
|
|
||||||
import InvitesServer from './invites';
|
import InvitesServer from './invites';
|
||||||
import MembersServer from './members';
|
import MembersServer from './members';
|
||||||
|
|||||||
@@ -83,12 +83,7 @@ export default function EditProjectDetails({ project }: Props) {
|
|||||||
<span className="title">Details</span>
|
<span className="title">Details</span>
|
||||||
</WidgetHead>
|
</WidgetHead>
|
||||||
<WidgetBody>
|
<WidgetBody>
|
||||||
<form
|
<form onSubmit={form.handleSubmit(onSubmit)} className="col gap-4">
|
||||||
onSubmit={form.handleSubmit(onSubmit, (errors) => {
|
|
||||||
console.log(errors);
|
|
||||||
})}
|
|
||||||
className="col gap-4"
|
|
||||||
>
|
|
||||||
<InputWithLabel
|
<InputWithLabel
|
||||||
label="Name"
|
label="Name"
|
||||||
{...form.register('name')}
|
{...form.register('name')}
|
||||||
|
|||||||
@@ -0,0 +1,61 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { differenceInHours } from 'date-fns';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
import { ProjectLink } from '@/components/links';
|
||||||
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
|
import { ModalHeader } from '@/modals/Modal/Container';
|
||||||
|
import type { IServiceOrganization } from '@openpanel/db';
|
||||||
|
import { useOpenPanel } from '@openpanel/nextjs';
|
||||||
|
import { FREE_PRODUCT_IDS } from '@openpanel/payments';
|
||||||
|
import Billing from './settings/organization/organization/billing';
|
||||||
|
|
||||||
|
interface SideEffectsProps {
|
||||||
|
organization: IServiceOrganization;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function SideEffectsFreePlan({
|
||||||
|
organization,
|
||||||
|
}: SideEffectsProps) {
|
||||||
|
const op = useOpenPanel();
|
||||||
|
const willEndInHours = organization.subscriptionEndsAt
|
||||||
|
? differenceInHours(organization.subscriptionEndsAt, new Date())
|
||||||
|
: null;
|
||||||
|
const [isFreePlan, setIsFreePlan] = useState<boolean>(
|
||||||
|
!!organization.subscriptionProductId &&
|
||||||
|
FREE_PRODUCT_IDS.includes(organization.subscriptionProductId),
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isFreePlan) {
|
||||||
|
op.track('free_plan_removed');
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={isFreePlan} onOpenChange={setIsFreePlan}>
|
||||||
|
<DialogContent className="max-w-xl">
|
||||||
|
<ModalHeader
|
||||||
|
onClose={() => setIsFreePlan(false)}
|
||||||
|
title={'Free plan has been removed'}
|
||||||
|
text={
|
||||||
|
<>
|
||||||
|
Please upgrade your plan to continue using OpenPanel. Select a
|
||||||
|
tier which is appropriate for your needs or{' '}
|
||||||
|
<ProjectLink
|
||||||
|
href="/settings/organization?tab=billing"
|
||||||
|
className="underline text-foreground"
|
||||||
|
>
|
||||||
|
manage billing
|
||||||
|
</ProjectLink>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<div className="-mx-4 mt-4">
|
||||||
|
<Billing organization={organization} />
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,91 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { Combobox } from '@/components/ui/combobox';
|
||||||
|
import { Dialog, DialogContent, DialogFooter } from '@/components/ui/dialog';
|
||||||
|
import { ModalHeader } from '@/modals/Modal/Container';
|
||||||
|
import { api, handleError } from '@/trpc/client';
|
||||||
|
import { TIMEZONES } from '@openpanel/common';
|
||||||
|
import type { IServiceOrganization } from '@openpanel/db';
|
||||||
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
|
interface SideEffectsProps {
|
||||||
|
organization: IServiceOrganization;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function SideEffectsTimezone({
|
||||||
|
organization,
|
||||||
|
}: SideEffectsProps) {
|
||||||
|
const [isMissingTimezone, setIsMissingTimezone] = useState<boolean>(
|
||||||
|
!organization.timezone,
|
||||||
|
);
|
||||||
|
const defaultTimezone = Intl.DateTimeFormat().resolvedOptions().timeZone;
|
||||||
|
const [timezone, setTimezone] = useState<string>(
|
||||||
|
TIMEZONES.includes(defaultTimezone) ? defaultTimezone : '',
|
||||||
|
);
|
||||||
|
|
||||||
|
const mutation = api.organization.update.useMutation({
|
||||||
|
onSuccess(res) {
|
||||||
|
toast('Timezone updated', {
|
||||||
|
description: 'Your timezone has been updated.',
|
||||||
|
});
|
||||||
|
window.location.reload();
|
||||||
|
},
|
||||||
|
onError: handleError,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={isMissingTimezone} onOpenChange={setIsMissingTimezone}>
|
||||||
|
<DialogContent
|
||||||
|
className="max-w-xl"
|
||||||
|
onEscapeKeyDown={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
}}
|
||||||
|
onInteractOutside={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<ModalHeader
|
||||||
|
onClose={false}
|
||||||
|
title="Select your timezone"
|
||||||
|
text={
|
||||||
|
<>
|
||||||
|
We have introduced new features that requires your timezone.
|
||||||
|
Please select the timezone you want to use for your organization.
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<Combobox
|
||||||
|
items={TIMEZONES.map((item) => ({
|
||||||
|
value: item,
|
||||||
|
label: item,
|
||||||
|
}))}
|
||||||
|
value={timezone}
|
||||||
|
onChange={setTimezone}
|
||||||
|
placeholder="Select a timezone"
|
||||||
|
searchable
|
||||||
|
size="lg"
|
||||||
|
className="w-full px-4"
|
||||||
|
/>
|
||||||
|
<DialogFooter className="mt-4">
|
||||||
|
<Button
|
||||||
|
size="lg"
|
||||||
|
disabled={!TIMEZONES.includes(timezone)}
|
||||||
|
loading={mutation.isLoading}
|
||||||
|
onClick={() =>
|
||||||
|
mutation.mutate({
|
||||||
|
id: organization.id,
|
||||||
|
name: organization.name,
|
||||||
|
timezone: timezone ?? '',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Save
|
||||||
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { differenceInHours } from 'date-fns';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
import { ProjectLink } from '@/components/links';
|
||||||
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
|
import { ModalHeader } from '@/modals/Modal/Container';
|
||||||
|
import type { IServiceOrganization } from '@openpanel/db';
|
||||||
|
import { useOpenPanel } from '@openpanel/nextjs';
|
||||||
|
import Billing from './settings/organization/organization/billing';
|
||||||
|
|
||||||
|
interface SideEffectsProps {
|
||||||
|
organization: IServiceOrganization;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function SideEffectsTrial({ organization }: SideEffectsProps) {
|
||||||
|
const op = useOpenPanel();
|
||||||
|
const willEndInHours = organization.subscriptionEndsAt
|
||||||
|
? differenceInHours(organization.subscriptionEndsAt, new Date())
|
||||||
|
: null;
|
||||||
|
|
||||||
|
const [isTrialDialogOpen, setIsTrialDialogOpen] = useState<boolean>(
|
||||||
|
willEndInHours !== null &&
|
||||||
|
organization.subscriptionStatus === 'trialing' &&
|
||||||
|
organization.subscriptionEndsAt !== null &&
|
||||||
|
willEndInHours <= 48,
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isTrialDialogOpen) {
|
||||||
|
op.track('trial_expires_soon');
|
||||||
|
}
|
||||||
|
}, [isTrialDialogOpen]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Dialog open={isTrialDialogOpen} onOpenChange={setIsTrialDialogOpen}>
|
||||||
|
<DialogContent className="max-w-xl">
|
||||||
|
<ModalHeader
|
||||||
|
onClose={() => setIsTrialDialogOpen(false)}
|
||||||
|
title={
|
||||||
|
willEndInHours !== null && willEndInHours > 0
|
||||||
|
? `Your trial is ending in ${willEndInHours} hours`
|
||||||
|
: 'Your trial has ended'
|
||||||
|
}
|
||||||
|
text={
|
||||||
|
<>
|
||||||
|
Please upgrade your plan to continue using OpenPanel. Select a
|
||||||
|
tier which is appropriate for your needs or{' '}
|
||||||
|
<ProjectLink
|
||||||
|
href="/settings/organization?tab=billing"
|
||||||
|
className="underline text-foreground"
|
||||||
|
>
|
||||||
|
manage billing
|
||||||
|
</ProjectLink>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<div className="-mx-4 mt-4">
|
||||||
|
<Billing organization={organization} />
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -4,12 +4,16 @@ import { differenceInHours } from 'date-fns';
|
|||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import { ProjectLink } from '@/components/links';
|
import { ProjectLink } from '@/components/links';
|
||||||
|
import { Combobox } from '@/components/ui/combobox';
|
||||||
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
import { ModalHeader } from '@/modals/Modal/Container';
|
import { ModalHeader } from '@/modals/Modal/Container';
|
||||||
import type { IServiceOrganization } from '@openpanel/db';
|
import type { IServiceOrganization } from '@openpanel/db';
|
||||||
import { useOpenPanel } from '@openpanel/nextjs';
|
import { useOpenPanel } from '@openpanel/nextjs';
|
||||||
import { FREE_PRODUCT_IDS } from '@openpanel/payments';
|
import { FREE_PRODUCT_IDS } from '@openpanel/payments';
|
||||||
import Billing from './settings/organization/organization/billing';
|
import Billing from './settings/organization/organization/billing';
|
||||||
|
import SideEffectsFreePlan from './side-effects-free-plan';
|
||||||
|
import SideEffectsTimezone from './side-effects-timezone';
|
||||||
|
import SideEffectsTrial from './side-effects-trial';
|
||||||
|
|
||||||
interface SideEffectsProps {
|
interface SideEffectsProps {
|
||||||
organization: IServiceOrganization;
|
organization: IServiceOrganization;
|
||||||
@@ -17,41 +21,11 @@ interface SideEffectsProps {
|
|||||||
|
|
||||||
export default function SideEffects({ organization }: SideEffectsProps) {
|
export default function SideEffects({ organization }: SideEffectsProps) {
|
||||||
const [mounted, setMounted] = useState(false);
|
const [mounted, setMounted] = useState(false);
|
||||||
const op = useOpenPanel();
|
|
||||||
const willEndInHours = organization.subscriptionEndsAt
|
|
||||||
? differenceInHours(organization.subscriptionEndsAt, new Date())
|
|
||||||
: null;
|
|
||||||
const [isTrialDialogOpen, setIsTrialDialogOpen] = useState<boolean>(false);
|
|
||||||
const [isFreePlan, setIsFreePlan] = useState<boolean>(false);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!mounted) {
|
setMounted(true);
|
||||||
setMounted(true);
|
|
||||||
}
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (
|
|
||||||
willEndInHours !== null &&
|
|
||||||
organization.subscriptionStatus === 'trialing' &&
|
|
||||||
organization.subscriptionEndsAt !== null &&
|
|
||||||
willEndInHours <= 48
|
|
||||||
) {
|
|
||||||
setIsTrialDialogOpen(true);
|
|
||||||
op.track('trial_expires_soon');
|
|
||||||
}
|
|
||||||
}, [mounted, organization]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (
|
|
||||||
organization.subscriptionProductId &&
|
|
||||||
FREE_PRODUCT_IDS.includes(organization.subscriptionProductId)
|
|
||||||
) {
|
|
||||||
setIsFreePlan(true);
|
|
||||||
op.track('free_plan_removed');
|
|
||||||
}
|
|
||||||
}, [mounted, organization]);
|
|
||||||
|
|
||||||
// Avoids hydration errors
|
// Avoids hydration errors
|
||||||
if (!mounted) {
|
if (!mounted) {
|
||||||
return null;
|
return null;
|
||||||
@@ -59,56 +33,9 @@ export default function SideEffects({ organization }: SideEffectsProps) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Dialog open={isTrialDialogOpen} onOpenChange={setIsTrialDialogOpen}>
|
<SideEffectsTimezone organization={organization} />
|
||||||
<DialogContent className="max-w-xl">
|
<SideEffectsTrial organization={organization} />
|
||||||
<ModalHeader
|
<SideEffectsFreePlan organization={organization} />
|
||||||
onClose={() => setIsTrialDialogOpen(false)}
|
|
||||||
title={
|
|
||||||
willEndInHours !== null && willEndInHours > 0
|
|
||||||
? `Your trial is ending in ${willEndInHours} hours`
|
|
||||||
: 'Your trial has ended'
|
|
||||||
}
|
|
||||||
text={
|
|
||||||
<>
|
|
||||||
Please upgrade your plan to continue using OpenPanel. Select a
|
|
||||||
tier which is appropriate for your needs or{' '}
|
|
||||||
<ProjectLink
|
|
||||||
href="/settings/organization?tab=billing"
|
|
||||||
className="underline text-foreground"
|
|
||||||
>
|
|
||||||
manage billing
|
|
||||||
</ProjectLink>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<div className="-mx-4 mt-4">
|
|
||||||
<Billing organization={organization} />
|
|
||||||
</div>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
<Dialog open={isFreePlan} onOpenChange={setIsFreePlan}>
|
|
||||||
<DialogContent className="max-w-xl">
|
|
||||||
<ModalHeader
|
|
||||||
onClose={() => setIsFreePlan(false)}
|
|
||||||
title={'Free plan has been removed'}
|
|
||||||
text={
|
|
||||||
<>
|
|
||||||
Please upgrade your plan to continue using OpenPanel. Select a
|
|
||||||
tier which is appropriate for your needs or{' '}
|
|
||||||
<ProjectLink
|
|
||||||
href="/settings/organization?tab=billing"
|
|
||||||
className="underline text-foreground"
|
|
||||||
>
|
|
||||||
manage billing
|
|
||||||
</ProjectLink>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<div className="-mx-4 mt-4">
|
|
||||||
<Billing organization={organization} />
|
|
||||||
</div>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,6 +51,7 @@ export const OnboardingCreateProject = ({
|
|||||||
resolver: zodResolver(zOnboardingProject),
|
resolver: zodResolver(zOnboardingProject),
|
||||||
defaultValues: {
|
defaultValues: {
|
||||||
organization: '',
|
organization: '',
|
||||||
|
timezone: Intl.DateTimeFormat().resolvedOptions().timeZone,
|
||||||
project: '',
|
project: '',
|
||||||
domain: '',
|
domain: '',
|
||||||
cors: [],
|
cors: [],
|
||||||
@@ -130,13 +131,33 @@ export const OnboardingCreateProject = ({
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<InputWithLabel
|
<>
|
||||||
label="Workspace name"
|
<InputWithLabel
|
||||||
info="This is the name of your workspace. It can be anything you like."
|
label="Workspace name"
|
||||||
placeholder="Eg. The Music Company"
|
info="This is the name of your workspace. It can be anything you like."
|
||||||
error={form.formState.errors.organization?.message}
|
placeholder="Eg. The Music Company"
|
||||||
{...form.register('organization')}
|
error={form.formState.errors.organization?.message}
|
||||||
/>
|
{...form.register('organization')}
|
||||||
|
/>
|
||||||
|
<Controller
|
||||||
|
name="timezone"
|
||||||
|
control={form.control}
|
||||||
|
render={({ field }) => (
|
||||||
|
<WithLabel label="Timezone">
|
||||||
|
<Combobox
|
||||||
|
placeholder="Select timezone"
|
||||||
|
items={Intl.supportedValuesOf('timeZone').map((item) => ({
|
||||||
|
value: item,
|
||||||
|
label: item,
|
||||||
|
}))}
|
||||||
|
value={field.value}
|
||||||
|
onChange={field.onChange}
|
||||||
|
className="w-full"
|
||||||
|
/>
|
||||||
|
</WithLabel>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
<InputWithLabel
|
<InputWithLabel
|
||||||
label="Project name"
|
label="Project name"
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ import { notFound } from 'next/navigation';
|
|||||||
|
|
||||||
import { ShareEnterPassword } from '@/components/auth/share-enter-password';
|
import { ShareEnterPassword } from '@/components/auth/share-enter-password';
|
||||||
import { OverviewRange } from '@/components/overview/overview-range';
|
import { OverviewRange } from '@/components/overview/overview-range';
|
||||||
import { getOrganizationBySlug, getShareOverviewById } from '@openpanel/db';
|
import { getOrganizationById, getShareOverviewById } from '@openpanel/db';
|
||||||
import { cookies } from 'next/headers';
|
import { cookies } from 'next/headers';
|
||||||
|
|
||||||
interface PageProps {
|
interface PageProps {
|
||||||
@@ -35,7 +35,7 @@ export default async function Page({
|
|||||||
return notFound();
|
return notFound();
|
||||||
}
|
}
|
||||||
const projectId = share.projectId;
|
const projectId = share.projectId;
|
||||||
const organization = await getOrganizationBySlug(share.organizationId);
|
const organization = await getOrganizationById(share.organizationId);
|
||||||
|
|
||||||
if (share.password) {
|
if (share.password) {
|
||||||
const cookie = cookies().get(`shared-overview-${share.id}`)?.value;
|
const cookie = cookies().get(`shared-overview-${share.id}`)?.value;
|
||||||
|
|||||||
@@ -34,10 +34,7 @@ export function SignInEmailForm() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<form
|
<form onSubmit={form.handleSubmit(onSubmit)} className="col gap-6">
|
||||||
onSubmit={form.handleSubmit(onSubmit, (err) => console.log(err))}
|
|
||||||
className="col gap-6"
|
|
||||||
>
|
|
||||||
<h3 className="text-2xl font-medium text-left">Sign in with email</h3>
|
<h3 className="text-2xl font-medium text-left">Sign in with email</h3>
|
||||||
<InputWithLabel
|
<InputWithLabel
|
||||||
{...form.register('email')}
|
{...form.register('email')}
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import type {
|
|||||||
IChartType,
|
IChartType,
|
||||||
IInterval,
|
IInterval,
|
||||||
} from '@openpanel/validation';
|
} from '@openpanel/validation';
|
||||||
import { endOfDay, startOfDay } from 'date-fns';
|
|
||||||
import { SaveIcon } from 'lucide-react';
|
import { SaveIcon } from 'lucide-react';
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { ReportChart } from '../report-chart';
|
import { ReportChart } from '../report-chart';
|
||||||
@@ -50,10 +49,8 @@ export function ChatReport({
|
|||||||
className="min-w-0"
|
className="min-w-0"
|
||||||
onChange={setRange}
|
onChange={setRange}
|
||||||
value={report.range}
|
value={report.range}
|
||||||
onStartDateChange={(date) =>
|
onStartDateChange={setStartDate}
|
||||||
setStartDate(startOfDay(date).toISOString())
|
onEndDateChange={setEndDate}
|
||||||
}
|
|
||||||
onEndDateChange={(date) => setEndDate(endOfDay(date).toISOString())}
|
|
||||||
endDate={report.endDate}
|
endDate={report.endDate}
|
||||||
startDate={report.startDate}
|
startDate={report.startDate}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -173,7 +173,9 @@ export function OverviewMetricCardNumber({
|
|||||||
<div className={cn('flex min-w-0 flex-col gap-2', className)}>
|
<div className={cn('flex min-w-0 flex-col gap-2', className)}>
|
||||||
<div className="flex items-center justify-between gap-2">
|
<div className="flex items-center justify-between gap-2">
|
||||||
<div className="flex min-w-0 items-center gap-2 text-left">
|
<div className="flex min-w-0 items-center gap-2 text-left">
|
||||||
<span className="truncate text-muted-foreground">{label}</span>
|
<span className="truncate text-sm font-medium text-muted-foreground">
|
||||||
|
{label}
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
|
|||||||
@@ -4,15 +4,19 @@ import { useOverviewOptions } from '@/components/overview/useOverviewOptions';
|
|||||||
import { useEventQueryFilters } from '@/hooks/useEventQueryFilters';
|
import { useEventQueryFilters } from '@/hooks/useEventQueryFilters';
|
||||||
import { cn } from '@/utils/cn';
|
import { cn } from '@/utils/cn';
|
||||||
|
|
||||||
|
import { useDashedStroke } from '@/hooks/use-dashed-stroke';
|
||||||
import { useFormatDateInterval } from '@/hooks/useFormatDateInterval';
|
import { useFormatDateInterval } from '@/hooks/useFormatDateInterval';
|
||||||
import { useNumber } from '@/hooks/useNumerFormatter';
|
import { useNumber } from '@/hooks/useNumerFormatter';
|
||||||
import { type RouterOutputs, api } from '@/trpc/client';
|
import { type RouterOutputs, api } from '@/trpc/client';
|
||||||
import { getChartColor } from '@/utils/theme';
|
import { getChartColor } from '@/utils/theme';
|
||||||
import { getPreviousMetric } from '@openpanel/common';
|
import { getPreviousMetric } from '@openpanel/common';
|
||||||
import type { IInterval } from '@openpanel/validation';
|
import type { IInterval } from '@openpanel/validation';
|
||||||
|
import { isSameDay, isSameHour, isSameMonth, isSameWeek } from 'date-fns';
|
||||||
|
import { last } from 'ramda';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import {
|
import {
|
||||||
CartesianGrid,
|
CartesianGrid,
|
||||||
|
Customized,
|
||||||
Line,
|
Line,
|
||||||
LineChart,
|
LineChart,
|
||||||
ResponsiveContainer,
|
ResponsiveContainer,
|
||||||
@@ -93,6 +97,44 @@ export default function OverviewMetrics({ projectId }: OverviewMetricsProps) {
|
|||||||
const xAxisProps = useXAxisProps({ interval });
|
const xAxisProps = useXAxisProps({ interval });
|
||||||
const yAxisProps = useYAxisProps();
|
const yAxisProps = useYAxisProps();
|
||||||
|
|
||||||
|
let dotIndex = undefined;
|
||||||
|
if (range === 'today') {
|
||||||
|
// Find closest index based on times
|
||||||
|
dotIndex = data.findIndex((item) => {
|
||||||
|
return isSameHour(item.date, new Date());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const { calcStrokeDasharray, handleAnimationEnd, getStrokeDasharray } =
|
||||||
|
useDashedStroke({
|
||||||
|
dotIndex,
|
||||||
|
});
|
||||||
|
|
||||||
|
const lastSerieDataItem = last(data)?.date || new Date();
|
||||||
|
const useDashedLastLine = (() => {
|
||||||
|
if (range === 'today') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (interval === 'hour') {
|
||||||
|
return isSameHour(lastSerieDataItem, new Date());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (interval === 'day') {
|
||||||
|
return isSameDay(lastSerieDataItem, new Date());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (interval === 'month') {
|
||||||
|
return isSameMonth(lastSerieDataItem, new Date());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (interval === 'week') {
|
||||||
|
return isSameWeek(lastSerieDataItem, new Date());
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
})();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="relative -top-0.5 col-span-6 -m-4 mb-0 mt-0 md:m-0">
|
<div className="relative -top-0.5 col-span-6 -m-4 mb-0 mt-0 md:m-0">
|
||||||
@@ -127,7 +169,7 @@ export default function OverviewMetrics({ projectId }: OverviewMetricsProps) {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="card p-4">
|
<div className="card p-4">
|
||||||
<div className="text-center text-muted-foreground mb-2">
|
<div className="text-center mb-3 -mt-1 text-sm font-medium text-muted-foreground">
|
||||||
{activeMetric.title}
|
{activeMetric.title}
|
||||||
</div>
|
</div>
|
||||||
<div className="w-full h-[150px]">
|
<div className="w-full h-[150px]">
|
||||||
@@ -135,6 +177,13 @@ export default function OverviewMetrics({ projectId }: OverviewMetricsProps) {
|
|||||||
<TooltipProvider metric={activeMetric} interval={interval}>
|
<TooltipProvider metric={activeMetric} interval={interval}>
|
||||||
<ResponsiveContainer width="100%" height="100%">
|
<ResponsiveContainer width="100%" height="100%">
|
||||||
<LineChart data={data}>
|
<LineChart data={data}>
|
||||||
|
<Customized component={calcStrokeDasharray} />
|
||||||
|
<Line
|
||||||
|
dataKey="calcStrokeDasharray"
|
||||||
|
legendType="none"
|
||||||
|
animationDuration={0}
|
||||||
|
onAnimationEnd={handleAnimationEnd}
|
||||||
|
/>
|
||||||
<Tooltip />
|
<Tooltip />
|
||||||
<YAxis
|
<YAxis
|
||||||
{...yAxisProps}
|
{...yAxisProps}
|
||||||
@@ -184,6 +233,11 @@ export default function OverviewMetrics({ projectId }: OverviewMetricsProps) {
|
|||||||
dataKey={activeMetric.key}
|
dataKey={activeMetric.key}
|
||||||
stroke={getChartColor(0)}
|
stroke={getChartColor(0)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
|
strokeDasharray={
|
||||||
|
useDashedLastLine
|
||||||
|
? getStrokeDasharray(activeMetric.key)
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
isAnimationActive={false}
|
isAnimationActive={false}
|
||||||
dot={
|
dot={
|
||||||
data.length > 90
|
data.length > 90
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
import { useOverviewOptions } from '@/components/overview/useOverviewOptions';
|
import { useOverviewOptions } from '@/components/overview/useOverviewOptions';
|
||||||
import { TimeWindowPicker } from '@/components/time-window-picker';
|
import { TimeWindowPicker } from '@/components/time-window-picker';
|
||||||
import { endOfDay, formatISO, startOfDay } from 'date-fns';
|
|
||||||
|
|
||||||
export function OverviewRange() {
|
export function OverviewRange() {
|
||||||
const { range, setRange, setStartDate, setEndDate, endDate, startDate } =
|
const { range, setRange, setStartDate, setEndDate, endDate, startDate } =
|
||||||
@@ -12,14 +11,8 @@ export function OverviewRange() {
|
|||||||
<TimeWindowPicker
|
<TimeWindowPicker
|
||||||
onChange={setRange}
|
onChange={setRange}
|
||||||
value={range}
|
value={range}
|
||||||
onStartDateChange={(date) => {
|
onStartDateChange={setStartDate}
|
||||||
const d = formatISO(startOfDay(new Date(date)));
|
onEndDateChange={setEndDate}
|
||||||
setStartDate(d);
|
|
||||||
}}
|
|
||||||
onEndDateChange={(date) => {
|
|
||||||
const d = formatISO(endOfDay(new Date(date)));
|
|
||||||
setEndDate(d);
|
|
||||||
}}
|
|
||||||
endDate={endDate}
|
endDate={endDate}
|
||||||
startDate={startDate}
|
startDate={startDate}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -6,13 +6,20 @@ import { api } from '@/trpc/client';
|
|||||||
import type { IChartData } from '@/trpc/client';
|
import type { IChartData } from '@/trpc/client';
|
||||||
import { cn } from '@/utils/cn';
|
import { cn } from '@/utils/cn';
|
||||||
import { getChartColor } from '@/utils/theme';
|
import { getChartColor } from '@/utils/theme';
|
||||||
import { isSameDay, isSameHour, isSameMonth } from 'date-fns';
|
import {
|
||||||
|
isFuture,
|
||||||
|
isSameDay,
|
||||||
|
isSameHour,
|
||||||
|
isSameMonth,
|
||||||
|
isSameWeek,
|
||||||
|
} from 'date-fns';
|
||||||
import { last } from 'ramda';
|
import { last } from 'ramda';
|
||||||
import React, { useCallback } from 'react';
|
import React, { useCallback, useState } from 'react';
|
||||||
import {
|
import {
|
||||||
Area,
|
Area,
|
||||||
CartesianGrid,
|
CartesianGrid,
|
||||||
ComposedChart,
|
ComposedChart,
|
||||||
|
Customized,
|
||||||
Legend,
|
Legend,
|
||||||
Line,
|
Line,
|
||||||
ReferenceLine,
|
ReferenceLine,
|
||||||
@@ -22,6 +29,7 @@ import {
|
|||||||
YAxis,
|
YAxis,
|
||||||
} from 'recharts';
|
} from 'recharts';
|
||||||
|
|
||||||
|
import { useDashedStroke, useStrokeDasharray } from '@/hooks/use-dashed-stroke';
|
||||||
import { useXAxisProps, useYAxisProps } from '../common/axis';
|
import { useXAxisProps, useYAxisProps } from '../common/axis';
|
||||||
import { SolidToDashedGradient } from '../common/linear-gradient';
|
import { SolidToDashedGradient } from '../common/linear-gradient';
|
||||||
import { ReportChartTooltip } from '../common/report-chart-tooltip';
|
import { ReportChartTooltip } from '../common/report-chart-tooltip';
|
||||||
@@ -63,15 +71,20 @@ export function Chart({ data }: Props) {
|
|||||||
const { series, setVisibleSeries } = useVisibleSeries(data);
|
const { series, setVisibleSeries } = useVisibleSeries(data);
|
||||||
const rechartData = useRechartDataModel(series);
|
const rechartData = useRechartDataModel(series);
|
||||||
|
|
||||||
// great care should be taken when computing lastIntervalPercent
|
let dotIndex = undefined;
|
||||||
// the expression below works for data.length - 1 equal intervals
|
if (range === 'today') {
|
||||||
// but if there are numeric x values in a "linear" axis, the formula
|
// Find closest index based on times
|
||||||
// should be updated to use those values
|
dotIndex = rechartData.findIndex((item) => {
|
||||||
const lastIntervalPercent =
|
return isSameHour(item.date, new Date());
|
||||||
((rechartData.length - 2) * 100) / (rechartData.length - 1);
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const lastSerieDataItem = last(series[0]?.data || [])?.date || new Date();
|
const lastSerieDataItem = last(series[0]?.data || [])?.date || new Date();
|
||||||
const useDashedLastLine = (() => {
|
const useDashedLastLine = (() => {
|
||||||
|
if (range === 'today') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (interval === 'hour') {
|
if (interval === 'hour') {
|
||||||
return isSameHour(lastSerieDataItem, new Date());
|
return isSameHour(lastSerieDataItem, new Date());
|
||||||
}
|
}
|
||||||
@@ -84,9 +97,18 @@ export function Chart({ data }: Props) {
|
|||||||
return isSameMonth(lastSerieDataItem, new Date());
|
return isSameMonth(lastSerieDataItem, new Date());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (interval === 'week') {
|
||||||
|
return isSameWeek(lastSerieDataItem, new Date());
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
const { getStrokeDasharray, calcStrokeDasharray, handleAnimationEnd } =
|
||||||
|
useDashedStroke({
|
||||||
|
dotIndex,
|
||||||
|
});
|
||||||
|
|
||||||
const CustomLegend = useCallback(() => {
|
const CustomLegend = useCallback(() => {
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-wrap justify-center gap-x-4 gap-y-1 text-xs mt-4 -mb-2">
|
<div className="flex flex-wrap justify-center gap-x-4 gap-y-1 text-xs mt-4 -mb-2">
|
||||||
@@ -117,6 +139,13 @@ export function Chart({ data }: Props) {
|
|||||||
<div className={cn('h-full w-full', isEditMode && 'card p-4')}>
|
<div className={cn('h-full w-full', isEditMode && 'card p-4')}>
|
||||||
<ResponsiveContainer>
|
<ResponsiveContainer>
|
||||||
<ComposedChart data={rechartData}>
|
<ComposedChart data={rechartData}>
|
||||||
|
<Customized component={calcStrokeDasharray} />
|
||||||
|
<Line
|
||||||
|
dataKey="calcStrokeDasharray"
|
||||||
|
legendType="none"
|
||||||
|
animationDuration={0}
|
||||||
|
onAnimationEnd={handleAnimationEnd}
|
||||||
|
/>
|
||||||
<CartesianGrid
|
<CartesianGrid
|
||||||
strokeDasharray="3 3"
|
strokeDasharray="3 3"
|
||||||
horizontal={true}
|
horizontal={true}
|
||||||
@@ -166,13 +195,6 @@ export function Chart({ data }: Props) {
|
|||||||
/>
|
/>
|
||||||
</linearGradient>
|
</linearGradient>
|
||||||
)}
|
)}
|
||||||
{useDashedLastLine && (
|
|
||||||
<SolidToDashedGradient
|
|
||||||
percentage={lastIntervalPercent}
|
|
||||||
baseColor={color}
|
|
||||||
id={`stroke${color}`}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</defs>
|
</defs>
|
||||||
<Line
|
<Line
|
||||||
dot={isAreaStyle && dataLength <= 8}
|
dot={isAreaStyle && dataLength <= 8}
|
||||||
@@ -181,7 +203,12 @@ export function Chart({ data }: Props) {
|
|||||||
isAnimationActive={false}
|
isAnimationActive={false}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
dataKey={`${serie.id}:count`}
|
dataKey={`${serie.id}:count`}
|
||||||
stroke={useDashedLastLine ? `url(#stroke${color})` : color}
|
stroke={color}
|
||||||
|
strokeDasharray={
|
||||||
|
useDashedLastLine
|
||||||
|
? getStrokeDasharray(`${serie.id}:count`)
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
// Use for legend
|
// Use for legend
|
||||||
fill={color}
|
fill={color}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -1,16 +1,9 @@
|
|||||||
import { createSlice } from '@reduxjs/toolkit';
|
|
||||||
import type { PayloadAction } from '@reduxjs/toolkit';
|
import type { PayloadAction } from '@reduxjs/toolkit';
|
||||||
import {
|
import { createSlice } from '@reduxjs/toolkit';
|
||||||
endOfDay,
|
import { endOfDay, format, isSameDay, isSameMonth, startOfDay } from 'date-fns';
|
||||||
formatISO,
|
|
||||||
isSameDay,
|
|
||||||
isSameMonth,
|
|
||||||
startOfDay,
|
|
||||||
} from 'date-fns';
|
|
||||||
|
|
||||||
import { shortId } from '@openpanel/common';
|
import { shortId } from '@openpanel/common';
|
||||||
import {
|
import {
|
||||||
alphabetIds,
|
|
||||||
getDefaultIntervalByDates,
|
getDefaultIntervalByDates,
|
||||||
getDefaultIntervalByRange,
|
getDefaultIntervalByRange,
|
||||||
isHourIntervalEnabledByRange,
|
isHourIntervalEnabledByRange,
|
||||||
@@ -192,31 +185,10 @@ export const reportSlice = createSlice({
|
|||||||
state.lineType = action.payload;
|
state.lineType = action.payload;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Custom start and end date
|
|
||||||
changeDates: (
|
|
||||||
state,
|
|
||||||
action: PayloadAction<{
|
|
||||||
startDate: string;
|
|
||||||
endDate: string;
|
|
||||||
}>,
|
|
||||||
) => {
|
|
||||||
state.dirty = true;
|
|
||||||
state.startDate = formatISO(startOfDay(action.payload.startDate));
|
|
||||||
state.endDate = formatISO(endOfDay(action.payload.endDate));
|
|
||||||
|
|
||||||
if (isSameDay(state.startDate, state.endDate)) {
|
|
||||||
state.interval = 'hour';
|
|
||||||
} else if (isSameMonth(state.startDate, state.endDate)) {
|
|
||||||
state.interval = 'day';
|
|
||||||
} else {
|
|
||||||
state.interval = 'month';
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// Date range
|
// Date range
|
||||||
changeStartDate: (state, action: PayloadAction<string>) => {
|
changeStartDate: (state, action: PayloadAction<string>) => {
|
||||||
state.dirty = true;
|
state.dirty = true;
|
||||||
state.startDate = formatISO(startOfDay(action.payload));
|
state.startDate = action.payload;
|
||||||
|
|
||||||
const interval = getDefaultIntervalByDates(
|
const interval = getDefaultIntervalByDates(
|
||||||
state.startDate,
|
state.startDate,
|
||||||
@@ -230,7 +202,7 @@ export const reportSlice = createSlice({
|
|||||||
// Date range
|
// Date range
|
||||||
changeEndDate: (state, action: PayloadAction<string>) => {
|
changeEndDate: (state, action: PayloadAction<string>) => {
|
||||||
state.dirty = true;
|
state.dirty = true;
|
||||||
state.endDate = formatISO(endOfDay(action.payload));
|
state.endDate = action.payload;
|
||||||
|
|
||||||
const interval = getDefaultIntervalByDates(
|
const interval = getDefaultIntervalByDates(
|
||||||
state.startDate,
|
state.startDate,
|
||||||
@@ -263,8 +235,6 @@ export const reportSlice = createSlice({
|
|||||||
},
|
},
|
||||||
|
|
||||||
changeUnit(state, action: PayloadAction<string | undefined>) {
|
changeUnit(state, action: PayloadAction<string | undefined>) {
|
||||||
console.log('here?!?!', action.payload);
|
|
||||||
|
|
||||||
state.dirty = true;
|
state.dirty = true;
|
||||||
state.unit = action.payload || undefined;
|
state.unit = action.payload || undefined;
|
||||||
},
|
},
|
||||||
@@ -305,7 +275,6 @@ export const {
|
|||||||
removeBreakdown,
|
removeBreakdown,
|
||||||
changeBreakdown,
|
changeBreakdown,
|
||||||
changeInterval,
|
changeInterval,
|
||||||
changeDates,
|
|
||||||
changeStartDate,
|
changeStartDate,
|
||||||
changeEndDate,
|
changeEndDate,
|
||||||
changeDateRanges,
|
changeDateRanges,
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import { useCallback, useEffect, useRef } from 'react';
|
|||||||
import { shouldIgnoreKeypress } from '@/utils/should-ignore-keypress';
|
import { shouldIgnoreKeypress } from '@/utils/should-ignore-keypress';
|
||||||
import { timeWindows } from '@openpanel/constants';
|
import { timeWindows } from '@openpanel/constants';
|
||||||
import type { IChartRange } from '@openpanel/validation';
|
import type { IChartRange } from '@openpanel/validation';
|
||||||
|
import { endOfDay, format, startOfDay } from 'date-fns';
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
value: IChartRange;
|
value: IChartRange;
|
||||||
@@ -46,8 +47,8 @@ export function TimeWindowPicker({
|
|||||||
const handleCustom = useCallback(() => {
|
const handleCustom = useCallback(() => {
|
||||||
pushModal('DateRangerPicker', {
|
pushModal('DateRangerPicker', {
|
||||||
onChange: ({ startDate, endDate }) => {
|
onChange: ({ startDate, endDate }) => {
|
||||||
onStartDateChange(startDate.toISOString());
|
onStartDateChange(format(startOfDay(startDate), 'yyyy-MM-dd HH:mm:ss'));
|
||||||
onEndDateChange(endDate.toISOString());
|
onEndDateChange(format(endOfDay(endDate), 'yyyy-MM-dd HH:mm:ss'));
|
||||||
onChange('custom');
|
onChange('custom');
|
||||||
},
|
},
|
||||||
startDate: startDate ? new Date(startDate) : undefined,
|
startDate: startDate ? new Date(startDate) : undefined,
|
||||||
@@ -113,6 +114,12 @@ export function TimeWindowPicker({
|
|||||||
{timeWindows.today.shortcut}
|
{timeWindows.today.shortcut}
|
||||||
</DropdownMenuShortcut>
|
</DropdownMenuShortcut>
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
|
<DropdownMenuItem onClick={() => onChange(timeWindows.yesterday.key)}>
|
||||||
|
{timeWindows.yesterday.label}
|
||||||
|
<DropdownMenuShortcut>
|
||||||
|
{timeWindows.yesterday.shortcut}
|
||||||
|
</DropdownMenuShortcut>
|
||||||
|
</DropdownMenuItem>
|
||||||
</DropdownMenuGroup>
|
</DropdownMenuGroup>
|
||||||
|
|
||||||
<DropdownMenuSeparator />
|
<DropdownMenuSeparator />
|
||||||
@@ -130,6 +137,18 @@ export function TimeWindowPicker({
|
|||||||
{timeWindows['30d'].shortcut}
|
{timeWindows['30d'].shortcut}
|
||||||
</DropdownMenuShortcut>
|
</DropdownMenuShortcut>
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
|
<DropdownMenuItem onClick={() => onChange(timeWindows['6m'].key)}>
|
||||||
|
{timeWindows['6m'].label}
|
||||||
|
<DropdownMenuShortcut>
|
||||||
|
{timeWindows['6m'].shortcut}
|
||||||
|
</DropdownMenuShortcut>
|
||||||
|
</DropdownMenuItem>
|
||||||
|
<DropdownMenuItem onClick={() => onChange(timeWindows['12m'].key)}>
|
||||||
|
{timeWindows['12m'].label}
|
||||||
|
<DropdownMenuShortcut>
|
||||||
|
{timeWindows['12m'].shortcut}
|
||||||
|
</DropdownMenuShortcut>
|
||||||
|
</DropdownMenuItem>
|
||||||
</DropdownMenuGroup>
|
</DropdownMenuGroup>
|
||||||
|
|
||||||
<DropdownMenuSeparator />
|
<DropdownMenuSeparator />
|
||||||
|
|||||||
@@ -18,8 +18,6 @@ export function InputEnter({
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (value !== internalValue) {
|
if (value !== internalValue) {
|
||||||
console.log(value, internalValue);
|
|
||||||
|
|
||||||
setInternalValue(value ?? '');
|
setInternalValue(value ?? '');
|
||||||
}
|
}
|
||||||
}, [value]);
|
}, [value]);
|
||||||
|
|||||||
199
apps/dashboard/src/hooks/use-dashed-stroke.tsx
Normal file
199
apps/dashboard/src/hooks/use-dashed-stroke.tsx
Normal file
@@ -0,0 +1,199 @@
|
|||||||
|
import { forwardRef, useCallback, useRef, useState } from 'react';
|
||||||
|
import { Customized, Line } from 'recharts';
|
||||||
|
export type GraphicalItemPoint = {
|
||||||
|
/**
|
||||||
|
* x point coordinate.
|
||||||
|
*/
|
||||||
|
x?: number;
|
||||||
|
/**
|
||||||
|
* y point coordinate.
|
||||||
|
*/
|
||||||
|
y?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type GraphicalItemProps = {
|
||||||
|
/**
|
||||||
|
* graphical item points.
|
||||||
|
*/
|
||||||
|
points?: GraphicalItemPoint[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ItemProps = {
|
||||||
|
/**
|
||||||
|
* item data key.
|
||||||
|
*/
|
||||||
|
dataKey?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ItemType = {
|
||||||
|
/**
|
||||||
|
* recharts item display name.
|
||||||
|
*/
|
||||||
|
displayName?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Item = {
|
||||||
|
/**
|
||||||
|
* item props.
|
||||||
|
*/
|
||||||
|
props?: ItemProps;
|
||||||
|
/**
|
||||||
|
* recharts item class.
|
||||||
|
*/
|
||||||
|
type?: ItemType;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type GraphicalItem = {
|
||||||
|
/**
|
||||||
|
* from recharts internal state and props of chart.
|
||||||
|
*/
|
||||||
|
props?: GraphicalItemProps;
|
||||||
|
/**
|
||||||
|
* from recharts internal state and props of chart.
|
||||||
|
*/
|
||||||
|
item?: Item;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type RechartsChartProps = {
|
||||||
|
/**
|
||||||
|
* from recharts internal state and props of chart.
|
||||||
|
*/
|
||||||
|
formattedGraphicalItems?: GraphicalItem[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type CalculateStrokeDasharray = (props?: any) => any;
|
||||||
|
|
||||||
|
export type LineStrokeDasharray = {
|
||||||
|
/**
|
||||||
|
* line name.
|
||||||
|
*/
|
||||||
|
name?: string;
|
||||||
|
/**
|
||||||
|
* line strokeDasharray.
|
||||||
|
*/
|
||||||
|
strokeDasharray?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type LinesStrokeDasharray = LineStrokeDasharray[];
|
||||||
|
|
||||||
|
export type LineProps = {
|
||||||
|
/**
|
||||||
|
* line name.
|
||||||
|
*/
|
||||||
|
name?: string;
|
||||||
|
/**
|
||||||
|
* specifies the starting index of the first dot in the dash pattern.
|
||||||
|
*/
|
||||||
|
dotIndex?: number;
|
||||||
|
/**
|
||||||
|
* defines the pattern of dashes and gaps. an array of [gap length, dash length].
|
||||||
|
*/
|
||||||
|
strokeDasharray?: [number, number];
|
||||||
|
/**
|
||||||
|
* adjusts the percentage correction of the first line segment for better alignment in curved lines.
|
||||||
|
*/
|
||||||
|
curveCorrection?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type UseStrokeDasharrayProps = {
|
||||||
|
/**
|
||||||
|
* an array of properties to target specific line(s) and override default settings.
|
||||||
|
*/
|
||||||
|
linesProps?: LineProps[];
|
||||||
|
} & LineProps;
|
||||||
|
|
||||||
|
export function useStrokeDasharray({
|
||||||
|
linesProps = [],
|
||||||
|
dotIndex = -2,
|
||||||
|
strokeDasharray: restStroke = [5, 3],
|
||||||
|
curveCorrection = 1,
|
||||||
|
}: UseStrokeDasharrayProps): [CalculateStrokeDasharray, LinesStrokeDasharray] {
|
||||||
|
const linesStrokeDasharray = useRef<LinesStrokeDasharray>([]);
|
||||||
|
|
||||||
|
const calculateStrokeDasharray = useCallback(
|
||||||
|
(props: RechartsChartProps): null => {
|
||||||
|
const items = props?.formattedGraphicalItems;
|
||||||
|
|
||||||
|
const getLineWidth = (points: GraphicalItemPoint[]) => {
|
||||||
|
const width = points?.reduce((acc, point, index) => {
|
||||||
|
if (!index) return acc;
|
||||||
|
|
||||||
|
const prevPoint = points?.[index - 1];
|
||||||
|
|
||||||
|
const xAxis = point?.x || 0;
|
||||||
|
const prevXAxis = prevPoint?.x || 0;
|
||||||
|
const xWidth = xAxis - prevXAxis;
|
||||||
|
|
||||||
|
const yAxis = point?.y || 0;
|
||||||
|
const prevYAxis = prevPoint?.y || 0;
|
||||||
|
const yWidth = Math.abs(yAxis - prevYAxis);
|
||||||
|
|
||||||
|
const hypotenuse = Math.sqrt(xWidth * xWidth + yWidth * yWidth);
|
||||||
|
acc += hypotenuse;
|
||||||
|
return acc;
|
||||||
|
}, 0);
|
||||||
|
|
||||||
|
return width || 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
items?.forEach((line) => {
|
||||||
|
const linePoints = line?.props?.points;
|
||||||
|
const lineWidth = getLineWidth(linePoints || []);
|
||||||
|
|
||||||
|
const name = line?.item?.props?.dataKey;
|
||||||
|
const targetLine = linesProps?.find((target) => target?.name === name);
|
||||||
|
const targetIndex = targetLine?.dotIndex ?? dotIndex;
|
||||||
|
const dashedPoints = linePoints?.slice(targetIndex);
|
||||||
|
const dashedWidth = getLineWidth(dashedPoints || []);
|
||||||
|
|
||||||
|
if (!lineWidth || !dashedWidth) return;
|
||||||
|
|
||||||
|
const firstWidth = lineWidth - dashedWidth;
|
||||||
|
const targetCurve = targetLine?.curveCorrection ?? curveCorrection;
|
||||||
|
const correctionWidth = (firstWidth * targetCurve) / 100;
|
||||||
|
const firstDasharray = firstWidth + correctionWidth;
|
||||||
|
|
||||||
|
const targetRestStroke = targetLine?.strokeDasharray || restStroke;
|
||||||
|
const gapDashWidth = targetRestStroke?.[0] + targetRestStroke?.[1] || 1;
|
||||||
|
const restDasharrayLength = dashedWidth / gapDashWidth;
|
||||||
|
const restDasharray = new Array(Math.ceil(restDasharrayLength)).fill(
|
||||||
|
targetRestStroke.join(' '),
|
||||||
|
);
|
||||||
|
|
||||||
|
const strokeDasharray = `${firstDasharray} ${restDasharray.join(' ')}`;
|
||||||
|
const lineStrokeDasharray = { name, strokeDasharray };
|
||||||
|
|
||||||
|
const dasharrayIndex = linesStrokeDasharray.current.findIndex((d) => {
|
||||||
|
return d.name === line?.item?.props?.dataKey;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (dasharrayIndex === -1) {
|
||||||
|
linesStrokeDasharray.current.push(lineStrokeDasharray);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
linesStrokeDasharray.current[dasharrayIndex] = lineStrokeDasharray;
|
||||||
|
});
|
||||||
|
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
[dotIndex],
|
||||||
|
);
|
||||||
|
|
||||||
|
return [calculateStrokeDasharray, linesStrokeDasharray.current];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useDashedStroke(options: UseStrokeDasharrayProps = {}) {
|
||||||
|
const [calcStrokeDasharray, strokes] = useStrokeDasharray(options);
|
||||||
|
const [strokeDasharray, setStrokeDasharray] = useState([...strokes]);
|
||||||
|
const handleAnimationEnd = () => setStrokeDasharray([...strokes]);
|
||||||
|
const getStrokeDasharray = (name: string) => {
|
||||||
|
return strokeDasharray.find((s) => s?.name === name)?.strokeDasharray;
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
calcStrokeDasharray,
|
||||||
|
getStrokeDasharray,
|
||||||
|
handleAnimationEnd,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -83,12 +83,7 @@ export default function AddProject() {
|
|||||||
return (
|
return (
|
||||||
<ModalContent>
|
<ModalContent>
|
||||||
<ModalHeader title="Create project" />
|
<ModalHeader title="Create project" />
|
||||||
<form
|
<form onSubmit={form.handleSubmit(onSubmit)} className="col gap-4">
|
||||||
onSubmit={form.handleSubmit(onSubmit, (errors) => {
|
|
||||||
console.log(errors);
|
|
||||||
})}
|
|
||||||
className="col gap-4"
|
|
||||||
>
|
|
||||||
<InputWithLabel label="Name" {...form.register('name')} />
|
<InputWithLabel label="Name" {...form.register('name')} />
|
||||||
|
|
||||||
<div className="-mb-2 flex gap-2 items-center justify-between">
|
<div className="-mb-2 flex gap-2 items-center justify-between">
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
export * from './src/date';
|
export * from './src/date';
|
||||||
|
export * from './src/timezones';
|
||||||
export * from './src/object';
|
export * from './src/object';
|
||||||
export * from './src/names';
|
export * from './src/names';
|
||||||
export * from './src/string';
|
export * from './src/string';
|
||||||
export * from './src/math';
|
export * from './src/math';
|
||||||
export * from './src/slug';
|
export * from './src/slug';
|
||||||
export * from './src/fill-series';
|
|
||||||
export * from './src/url';
|
export * from './src/url';
|
||||||
export * from './src/id';
|
export * from './src/id';
|
||||||
export * from './src/get-previous-metric';
|
export * from './src/get-previous-metric';
|
||||||
|
export * from './src/group-by-labels';
|
||||||
|
|||||||
@@ -8,6 +8,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@openpanel/constants": "workspace:*",
|
"@openpanel/constants": "workspace:*",
|
||||||
"date-fns": "^3.3.1",
|
"date-fns": "^3.3.1",
|
||||||
|
"luxon": "^3.6.1",
|
||||||
"mathjs": "^12.3.2",
|
"mathjs": "^12.3.2",
|
||||||
"nanoid": "^5.0.7",
|
"nanoid": "^5.0.7",
|
||||||
"ramda": "^0.29.1",
|
"ramda": "^0.29.1",
|
||||||
@@ -19,6 +20,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@openpanel/tsconfig": "workspace:*",
|
"@openpanel/tsconfig": "workspace:*",
|
||||||
"@openpanel/validation": "workspace:*",
|
"@openpanel/validation": "workspace:*",
|
||||||
|
"@types/luxon": "^3.6.2",
|
||||||
"@types/node": "20.14.8",
|
"@types/node": "20.14.8",
|
||||||
"@types/ramda": "^0.29.6",
|
"@types/ramda": "^0.29.6",
|
||||||
"@types/ua-parser-js": "^0.7.39",
|
"@types/ua-parser-js": "^0.7.39",
|
||||||
|
|||||||
@@ -1,58 +1,7 @@
|
|||||||
|
import { DateTime } from 'luxon';
|
||||||
|
|
||||||
|
export { DateTime };
|
||||||
|
|
||||||
export function getTime(date: string | number | Date) {
|
export function getTime(date: string | number | Date) {
|
||||||
return new Date(date).getTime();
|
return new Date(date).getTime();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function toISOString(date: string | number | Date) {
|
|
||||||
return new Date(date).toISOString();
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getTimezoneFromDateString(_date: string) {
|
|
||||||
const mapper: Record<string, string> = {
|
|
||||||
'+00:00': 'UTC',
|
|
||||||
'+01:00': 'Europe/Paris',
|
|
||||||
'+02:00': 'Europe/Stockholm',
|
|
||||||
'+03:00': 'Europe/Moscow',
|
|
||||||
'+04:00': 'Asia/Dubai',
|
|
||||||
'+05:00': 'Asia/Karachi',
|
|
||||||
'+06:00': 'Asia/Dhaka',
|
|
||||||
'+07:00': 'Asia/Bangkok',
|
|
||||||
'+08:00': 'Asia/Shanghai',
|
|
||||||
'+09:00': 'Asia/Tokyo',
|
|
||||||
'+10:00': 'Australia/Sydney',
|
|
||||||
'+11:00': 'Pacific/Noumea',
|
|
||||||
'+12:00': 'Pacific/Fiji',
|
|
||||||
'-02:00': 'America/Noronha',
|
|
||||||
'-03:00': 'America/Sao_Paulo',
|
|
||||||
'-04:00': 'America/Santiago',
|
|
||||||
'-05:00': 'America/Bogota',
|
|
||||||
'-06:00': 'America/Mexico_City',
|
|
||||||
'-07:00': 'America/Phoenix',
|
|
||||||
'-08:00': 'America/Los_Angeles',
|
|
||||||
'-09:00': 'America/Anchorage',
|
|
||||||
'-10:00': 'Pacific/Honolulu',
|
|
||||||
'-11:00': 'Pacific/Midway',
|
|
||||||
'-12:00': 'Pacific/Tarawa',
|
|
||||||
// Additional time zones
|
|
||||||
'+05:30': 'Asia/Kolkata',
|
|
||||||
'+05:45': 'Asia/Kathmandu',
|
|
||||||
'+08:45': 'Australia/Eucla',
|
|
||||||
'+09:30': 'Australia/Darwin',
|
|
||||||
'+10:30': 'Australia/Adelaide',
|
|
||||||
'+12:45': 'Pacific/Chatham',
|
|
||||||
'+13:00': 'Pacific/Apia',
|
|
||||||
'+14:00': 'Pacific/Kiritimati',
|
|
||||||
'-02:30': 'America/St_Johns',
|
|
||||||
'-03:30': 'America/St_Johns',
|
|
||||||
'-04:30': 'America/Caracas',
|
|
||||||
'-09:30': 'Pacific/Marquesas',
|
|
||||||
};
|
|
||||||
|
|
||||||
const defaultTimezone = 'UTC';
|
|
||||||
|
|
||||||
const match = _date.match(/([+-][0-9]{2}):([0-9]{2})$/)?.[0];
|
|
||||||
if (match) {
|
|
||||||
return mapper[match] ?? defaultTimezone;
|
|
||||||
}
|
|
||||||
|
|
||||||
return defaultTimezone;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,143 +0,0 @@
|
|||||||
import {
|
|
||||||
addDays,
|
|
||||||
addHours,
|
|
||||||
addMinutes,
|
|
||||||
addMonths,
|
|
||||||
addWeeks,
|
|
||||||
format,
|
|
||||||
parseISO,
|
|
||||||
startOfDay,
|
|
||||||
startOfHour,
|
|
||||||
startOfMinute,
|
|
||||||
startOfMonth,
|
|
||||||
startOfWeek,
|
|
||||||
} from 'date-fns';
|
|
||||||
|
|
||||||
import { NOT_SET_VALUE } from '@openpanel/constants';
|
|
||||||
import type { IInterval } from '@openpanel/validation';
|
|
||||||
|
|
||||||
// Define the data structure
|
|
||||||
export interface ISerieDataItem {
|
|
||||||
label_0: string | null | undefined;
|
|
||||||
label_1?: string | null | undefined;
|
|
||||||
label_2?: string | null | undefined;
|
|
||||||
label_3?: string | null | undefined;
|
|
||||||
count: number;
|
|
||||||
date: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ISerieDataItemComplete {
|
|
||||||
labels: string[];
|
|
||||||
count: number;
|
|
||||||
date: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Function to round down the date to the nearest interval
|
|
||||||
function roundDate(date: Date, interval: IInterval): Date {
|
|
||||||
switch (interval) {
|
|
||||||
case 'minute':
|
|
||||||
return startOfMinute(date);
|
|
||||||
case 'hour':
|
|
||||||
return startOfHour(date);
|
|
||||||
case 'day':
|
|
||||||
return startOfDay(date);
|
|
||||||
case 'week':
|
|
||||||
return startOfWeek(date);
|
|
||||||
case 'month':
|
|
||||||
return startOfMonth(date);
|
|
||||||
default:
|
|
||||||
return startOfMinute(date);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function filterFalsyAfterTruthy(array: (string | undefined | null)[]) {
|
|
||||||
let foundTruthy = false;
|
|
||||||
const filtered = array.filter((item) => {
|
|
||||||
if (foundTruthy) {
|
|
||||||
// After a truthy, filter out falsy values
|
|
||||||
return !!item;
|
|
||||||
}
|
|
||||||
if (item) {
|
|
||||||
// Mark when the first truthy is encountered
|
|
||||||
foundTruthy = true;
|
|
||||||
}
|
|
||||||
// Return all elements until the first truthy is found
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (filtered.some((item) => !!item)) {
|
|
||||||
return filtered;
|
|
||||||
}
|
|
||||||
|
|
||||||
return [null];
|
|
||||||
}
|
|
||||||
|
|
||||||
function concatLabels(entry: ISerieDataItem): string {
|
|
||||||
return filterFalsyAfterTruthy([
|
|
||||||
entry.label_0,
|
|
||||||
entry.label_1,
|
|
||||||
entry.label_2,
|
|
||||||
entry.label_3,
|
|
||||||
])
|
|
||||||
.map((label) => label || NOT_SET_VALUE)
|
|
||||||
.join(':::');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Function to complete the timeline for each label
|
|
||||||
export function completeSerie(
|
|
||||||
data: ISerieDataItem[],
|
|
||||||
_startDate: string,
|
|
||||||
_endDate: string,
|
|
||||||
interval: IInterval,
|
|
||||||
) {
|
|
||||||
const startDate = parseISO(_startDate);
|
|
||||||
const endDate = parseISO(_endDate);
|
|
||||||
// Group data by label
|
|
||||||
const labelsMap = new Map<string, Map<string, number>>();
|
|
||||||
data.forEach((entry) => {
|
|
||||||
const roundedDate = roundDate(parseISO(entry.date), interval);
|
|
||||||
const dateKey = format(roundedDate, 'yyyy-MM-dd HH:mm:ss');
|
|
||||||
const label = concatLabels(entry) || NOT_SET_VALUE;
|
|
||||||
if (!labelsMap.has(label)) {
|
|
||||||
labelsMap.set(label, new Map());
|
|
||||||
}
|
|
||||||
const labelData = labelsMap.get(label)!;
|
|
||||||
labelData.set(dateKey, (labelData.get(dateKey) || 0) + (entry.count || 0));
|
|
||||||
});
|
|
||||||
|
|
||||||
// Complete the timeline for each label
|
|
||||||
const result: Record<string, ISerieDataItemComplete[]> = {};
|
|
||||||
labelsMap.forEach((counts, label) => {
|
|
||||||
let currentDate = roundDate(startDate, interval);
|
|
||||||
result[label] = [];
|
|
||||||
while (currentDate <= endDate) {
|
|
||||||
const dateKey = format(currentDate, 'yyyy-MM-dd HH:mm:ss');
|
|
||||||
result[label]!.push({
|
|
||||||
labels: label.split(':::'),
|
|
||||||
date: dateKey,
|
|
||||||
count: counts.get(dateKey) || 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Increment the current date based on the interval
|
|
||||||
switch (interval) {
|
|
||||||
case 'minute':
|
|
||||||
currentDate = addMinutes(currentDate, 1);
|
|
||||||
break;
|
|
||||||
case 'hour':
|
|
||||||
currentDate = addHours(currentDate, 1);
|
|
||||||
break;
|
|
||||||
case 'day':
|
|
||||||
currentDate = addDays(currentDate, 1);
|
|
||||||
break;
|
|
||||||
case 'week':
|
|
||||||
currentDate = addWeeks(currentDate, 1);
|
|
||||||
break;
|
|
||||||
case 'month':
|
|
||||||
currentDate = addMonths(currentDate, 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
70
packages/common/src/group-by-labels.ts
Normal file
70
packages/common/src/group-by-labels.ts
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
export interface ISerieDataItem {
|
||||||
|
label_0: string | null | undefined;
|
||||||
|
label_1?: string | null | undefined;
|
||||||
|
label_2?: string | null | undefined;
|
||||||
|
label_3?: string | null | undefined;
|
||||||
|
count: number;
|
||||||
|
date: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface GroupedDataPoint {
|
||||||
|
date: string;
|
||||||
|
count: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface GroupedResult {
|
||||||
|
name: string[]; // [label_0, label_1, label_2, label_3]
|
||||||
|
data: GroupedDataPoint[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function groupByLabels(data: ISerieDataItem[]): GroupedResult[] {
|
||||||
|
const groupedMap = new Map<string, GroupedResult>();
|
||||||
|
const timestamps = new Set<string>();
|
||||||
|
data.forEach((row) => {
|
||||||
|
timestamps.add(row.date);
|
||||||
|
const labels = Object.keys(row)
|
||||||
|
.filter((key) => key.startsWith('label_'))
|
||||||
|
.sort((a, b) => {
|
||||||
|
const numA = Number.parseInt(a.replace('label_', ''));
|
||||||
|
const numB = Number.parseInt(b.replace('label_', ''));
|
||||||
|
return numA - numB;
|
||||||
|
})
|
||||||
|
.map((key) => (row as any)[key])
|
||||||
|
.filter((label): label is string => !!label);
|
||||||
|
|
||||||
|
const labelKey = labels.join(':::');
|
||||||
|
|
||||||
|
if (!groupedMap.has(labelKey)) {
|
||||||
|
groupedMap.set(labelKey, {
|
||||||
|
name: labels,
|
||||||
|
data: [],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const group = groupedMap.get(labelKey)!;
|
||||||
|
group.data.push({
|
||||||
|
date: row.date,
|
||||||
|
count: row.count,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const result = Array.from(groupedMap.values()).map((group) => ({
|
||||||
|
...group,
|
||||||
|
data: group.data.sort(
|
||||||
|
(a, b) => new Date(a.date).getTime() - new Date(b.date).getTime(),
|
||||||
|
),
|
||||||
|
}));
|
||||||
|
|
||||||
|
return result
|
||||||
|
.filter((group) => group.name.length > 0)
|
||||||
|
.map((group) => {
|
||||||
|
return {
|
||||||
|
...group,
|
||||||
|
// This will ensure that all dates are present in the data array
|
||||||
|
data: Array.from(timestamps).map((date) => {
|
||||||
|
const dataPoint = group.data.find((dp) => dp.date === date);
|
||||||
|
return dataPoint || { date, count: 0 };
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
600
packages/common/src/timezones.ts
Normal file
600
packages/common/src/timezones.ts
Normal file
@@ -0,0 +1,600 @@
|
|||||||
|
// List of available timezones from Clickhouse `select time_zone from system.time_zones`
|
||||||
|
export const TIMEZONES = [
|
||||||
|
'Africa/Abidjan',
|
||||||
|
'Africa/Accra',
|
||||||
|
'Africa/Addis_Ababa',
|
||||||
|
'Africa/Algiers',
|
||||||
|
'Africa/Asmara',
|
||||||
|
'Africa/Asmera',
|
||||||
|
'Africa/Bamako',
|
||||||
|
'Africa/Bangui',
|
||||||
|
'Africa/Banjul',
|
||||||
|
'Africa/Bissau',
|
||||||
|
'Africa/Blantyre',
|
||||||
|
'Africa/Brazzaville',
|
||||||
|
'Africa/Bujumbura',
|
||||||
|
'Africa/Cairo',
|
||||||
|
'Africa/Casablanca',
|
||||||
|
'Africa/Ceuta',
|
||||||
|
'Africa/Conakry',
|
||||||
|
'Africa/Dakar',
|
||||||
|
'Africa/Dar_es_Salaam',
|
||||||
|
'Africa/Djibouti',
|
||||||
|
'Africa/Douala',
|
||||||
|
'Africa/El_Aaiun',
|
||||||
|
'Africa/Freetown',
|
||||||
|
'Africa/Gaborone',
|
||||||
|
'Africa/Harare',
|
||||||
|
'Africa/Johannesburg',
|
||||||
|
'Africa/Juba',
|
||||||
|
'Africa/Kampala',
|
||||||
|
'Africa/Khartoum',
|
||||||
|
'Africa/Kigali',
|
||||||
|
'Africa/Kinshasa',
|
||||||
|
'Africa/Lagos',
|
||||||
|
'Africa/Libreville',
|
||||||
|
'Africa/Lome',
|
||||||
|
'Africa/Luanda',
|
||||||
|
'Africa/Lubumbashi',
|
||||||
|
'Africa/Lusaka',
|
||||||
|
'Africa/Malabo',
|
||||||
|
'Africa/Maputo',
|
||||||
|
'Africa/Maseru',
|
||||||
|
'Africa/Mbabane',
|
||||||
|
'Africa/Mogadishu',
|
||||||
|
'Africa/Monrovia',
|
||||||
|
'Africa/Nairobi',
|
||||||
|
'Africa/Ndjamena',
|
||||||
|
'Africa/Niamey',
|
||||||
|
'Africa/Nouakchott',
|
||||||
|
'Africa/Ouagadougou',
|
||||||
|
'Africa/Porto-Novo',
|
||||||
|
'Africa/Sao_Tome',
|
||||||
|
'Africa/Timbuktu',
|
||||||
|
'Africa/Tripoli',
|
||||||
|
'Africa/Tunis',
|
||||||
|
'Africa/Windhoek',
|
||||||
|
'America/Adak',
|
||||||
|
'America/Anchorage',
|
||||||
|
'America/Anguilla',
|
||||||
|
'America/Antigua',
|
||||||
|
'America/Araguaina',
|
||||||
|
'America/Argentina/Buenos_Aires',
|
||||||
|
'America/Argentina/Catamarca',
|
||||||
|
'America/Argentina/ComodRivadavia',
|
||||||
|
'America/Argentina/Cordoba',
|
||||||
|
'America/Argentina/Jujuy',
|
||||||
|
'America/Argentina/La_Rioja',
|
||||||
|
'America/Argentina/Mendoza',
|
||||||
|
'America/Argentina/Rio_Gallegos',
|
||||||
|
'America/Argentina/Salta',
|
||||||
|
'America/Argentina/San_Juan',
|
||||||
|
'America/Argentina/San_Luis',
|
||||||
|
'America/Argentina/Tucuman',
|
||||||
|
'America/Argentina/Ushuaia',
|
||||||
|
'America/Aruba',
|
||||||
|
'America/Asuncion',
|
||||||
|
'America/Atikokan',
|
||||||
|
'America/Atka',
|
||||||
|
'America/Bahia',
|
||||||
|
'America/Bahia_Banderas',
|
||||||
|
'America/Barbados',
|
||||||
|
'America/Belem',
|
||||||
|
'America/Belize',
|
||||||
|
'America/Blanc-Sablon',
|
||||||
|
'America/Boa_Vista',
|
||||||
|
'America/Bogota',
|
||||||
|
'America/Boise',
|
||||||
|
'America/Buenos_Aires',
|
||||||
|
'America/Cambridge_Bay',
|
||||||
|
'America/Campo_Grande',
|
||||||
|
'America/Cancun',
|
||||||
|
'America/Caracas',
|
||||||
|
'America/Catamarca',
|
||||||
|
'America/Cayenne',
|
||||||
|
'America/Cayman',
|
||||||
|
'America/Chicago',
|
||||||
|
'America/Chihuahua',
|
||||||
|
'America/Ciudad_Juarez',
|
||||||
|
'America/Coral_Harbour',
|
||||||
|
'America/Cordoba',
|
||||||
|
'America/Costa_Rica',
|
||||||
|
'America/Creston',
|
||||||
|
'America/Cuiaba',
|
||||||
|
'America/Curacao',
|
||||||
|
'America/Danmarkshavn',
|
||||||
|
'America/Dawson',
|
||||||
|
'America/Dawson_Creek',
|
||||||
|
'America/Denver',
|
||||||
|
'America/Detroit',
|
||||||
|
'America/Dominica',
|
||||||
|
'America/Edmonton',
|
||||||
|
'America/Eirunepe',
|
||||||
|
'America/El_Salvador',
|
||||||
|
'America/Ensenada',
|
||||||
|
'America/Fort_Nelson',
|
||||||
|
'America/Fort_Wayne',
|
||||||
|
'America/Fortaleza',
|
||||||
|
'America/Glace_Bay',
|
||||||
|
'America/Godthab',
|
||||||
|
'America/Goose_Bay',
|
||||||
|
'America/Grand_Turk',
|
||||||
|
'America/Grenada',
|
||||||
|
'America/Guadeloupe',
|
||||||
|
'America/Guatemala',
|
||||||
|
'America/Guayaquil',
|
||||||
|
'America/Guyana',
|
||||||
|
'America/Halifax',
|
||||||
|
'America/Havana',
|
||||||
|
'America/Hermosillo',
|
||||||
|
'America/Indiana/Indianapolis',
|
||||||
|
'America/Indiana/Knox',
|
||||||
|
'America/Indiana/Marengo',
|
||||||
|
'America/Indiana/Petersburg',
|
||||||
|
'America/Indiana/Tell_City',
|
||||||
|
'America/Indiana/Vevay',
|
||||||
|
'America/Indiana/Vincennes',
|
||||||
|
'America/Indiana/Winamac',
|
||||||
|
'America/Indianapolis',
|
||||||
|
'America/Inuvik',
|
||||||
|
'America/Iqaluit',
|
||||||
|
'America/Jamaica',
|
||||||
|
'America/Jujuy',
|
||||||
|
'America/Juneau',
|
||||||
|
'America/Kentucky/Louisville',
|
||||||
|
'America/Kentucky/Monticello',
|
||||||
|
'America/Knox_IN',
|
||||||
|
'America/Kralendijk',
|
||||||
|
'America/La_Paz',
|
||||||
|
'America/Lima',
|
||||||
|
'America/Los_Angeles',
|
||||||
|
'America/Louisville',
|
||||||
|
'America/Lower_Princes',
|
||||||
|
'America/Maceio',
|
||||||
|
'America/Managua',
|
||||||
|
'America/Manaus',
|
||||||
|
'America/Marigot',
|
||||||
|
'America/Martinique',
|
||||||
|
'America/Matamoros',
|
||||||
|
'America/Mazatlan',
|
||||||
|
'America/Mendoza',
|
||||||
|
'America/Menominee',
|
||||||
|
'America/Merida',
|
||||||
|
'America/Metlakatla',
|
||||||
|
'America/Mexico_City',
|
||||||
|
'America/Miquelon',
|
||||||
|
'America/Moncton',
|
||||||
|
'America/Monterrey',
|
||||||
|
'America/Montevideo',
|
||||||
|
'America/Montreal',
|
||||||
|
'America/Montserrat',
|
||||||
|
'America/Nassau',
|
||||||
|
'America/New_York',
|
||||||
|
'America/Nipigon',
|
||||||
|
'America/Nome',
|
||||||
|
'America/Noronha',
|
||||||
|
'America/North_Dakota/Beulah',
|
||||||
|
'America/North_Dakota/Center',
|
||||||
|
'America/North_Dakota/New_Salem',
|
||||||
|
'America/Nuuk',
|
||||||
|
'America/Ojinaga',
|
||||||
|
'America/Panama',
|
||||||
|
'America/Pangnirtung',
|
||||||
|
'America/Paramaribo',
|
||||||
|
'America/Phoenix',
|
||||||
|
'America/Port-au-Prince',
|
||||||
|
'America/Port_of_Spain',
|
||||||
|
'America/Porto_Acre',
|
||||||
|
'America/Porto_Velho',
|
||||||
|
'America/Puerto_Rico',
|
||||||
|
'America/Punta_Arenas',
|
||||||
|
'America/Rainy_River',
|
||||||
|
'America/Rankin_Inlet',
|
||||||
|
'America/Recife',
|
||||||
|
'America/Regina',
|
||||||
|
'America/Resolute',
|
||||||
|
'America/Rio_Branco',
|
||||||
|
'America/Rosario',
|
||||||
|
'America/Santa_Isabel',
|
||||||
|
'America/Santarem',
|
||||||
|
'America/Santiago',
|
||||||
|
'America/Santo_Domingo',
|
||||||
|
'America/Sao_Paulo',
|
||||||
|
'America/Scoresbysund',
|
||||||
|
'America/Shiprock',
|
||||||
|
'America/Sitka',
|
||||||
|
'America/St_Barthelemy',
|
||||||
|
'America/St_Johns',
|
||||||
|
'America/St_Kitts',
|
||||||
|
'America/St_Lucia',
|
||||||
|
'America/St_Thomas',
|
||||||
|
'America/St_Vincent',
|
||||||
|
'America/Swift_Current',
|
||||||
|
'America/Tegucigalpa',
|
||||||
|
'America/Thule',
|
||||||
|
'America/Thunder_Bay',
|
||||||
|
'America/Tijuana',
|
||||||
|
'America/Toronto',
|
||||||
|
'America/Tortola',
|
||||||
|
'America/Vancouver',
|
||||||
|
'America/Virgin',
|
||||||
|
'America/Whitehorse',
|
||||||
|
'America/Winnipeg',
|
||||||
|
'America/Yakutat',
|
||||||
|
'America/Yellowknife',
|
||||||
|
'Antarctica/Casey',
|
||||||
|
'Antarctica/Davis',
|
||||||
|
'Antarctica/DumontDUrville',
|
||||||
|
'Antarctica/Macquarie',
|
||||||
|
'Antarctica/Mawson',
|
||||||
|
'Antarctica/McMurdo',
|
||||||
|
'Antarctica/Palmer',
|
||||||
|
'Antarctica/Rothera',
|
||||||
|
'Antarctica/South_Pole',
|
||||||
|
'Antarctica/Syowa',
|
||||||
|
'Antarctica/Troll',
|
||||||
|
'Antarctica/Vostok',
|
||||||
|
'Arctic/Longyearbyen',
|
||||||
|
'Asia/Aden',
|
||||||
|
'Asia/Almaty',
|
||||||
|
'Asia/Amman',
|
||||||
|
'Asia/Anadyr',
|
||||||
|
'Asia/Aqtau',
|
||||||
|
'Asia/Aqtobe',
|
||||||
|
'Asia/Ashgabat',
|
||||||
|
'Asia/Ashkhabad',
|
||||||
|
'Asia/Atyrau',
|
||||||
|
'Asia/Baghdad',
|
||||||
|
'Asia/Bahrain',
|
||||||
|
'Asia/Baku',
|
||||||
|
'Asia/Bangkok',
|
||||||
|
'Asia/Barnaul',
|
||||||
|
'Asia/Beirut',
|
||||||
|
'Asia/Bishkek',
|
||||||
|
'Asia/Brunei',
|
||||||
|
'Asia/Calcutta',
|
||||||
|
'Asia/Chita',
|
||||||
|
'Asia/Choibalsan',
|
||||||
|
'Asia/Chongqing',
|
||||||
|
'Asia/Chungking',
|
||||||
|
'Asia/Colombo',
|
||||||
|
'Asia/Dacca',
|
||||||
|
'Asia/Damascus',
|
||||||
|
'Asia/Dhaka',
|
||||||
|
'Asia/Dili',
|
||||||
|
'Asia/Dubai',
|
||||||
|
'Asia/Dushanbe',
|
||||||
|
'Asia/Famagusta',
|
||||||
|
'Asia/Gaza',
|
||||||
|
'Asia/Harbin',
|
||||||
|
'Asia/Hebron',
|
||||||
|
'Asia/Ho_Chi_Minh',
|
||||||
|
'Asia/Hong_Kong',
|
||||||
|
'Asia/Hovd',
|
||||||
|
'Asia/Irkutsk',
|
||||||
|
'Asia/Istanbul',
|
||||||
|
'Asia/Jakarta',
|
||||||
|
'Asia/Jayapura',
|
||||||
|
'Asia/Jerusalem',
|
||||||
|
'Asia/Kabul',
|
||||||
|
'Asia/Kamchatka',
|
||||||
|
'Asia/Karachi',
|
||||||
|
'Asia/Kashgar',
|
||||||
|
'Asia/Kathmandu',
|
||||||
|
'Asia/Katmandu',
|
||||||
|
'Asia/Khandyga',
|
||||||
|
'Asia/Kolkata',
|
||||||
|
'Asia/Krasnoyarsk',
|
||||||
|
'Asia/Kuala_Lumpur',
|
||||||
|
'Asia/Kuching',
|
||||||
|
'Asia/Kuwait',
|
||||||
|
'Asia/Macao',
|
||||||
|
'Asia/Macau',
|
||||||
|
'Asia/Magadan',
|
||||||
|
'Asia/Makassar',
|
||||||
|
'Asia/Manila',
|
||||||
|
'Asia/Muscat',
|
||||||
|
'Asia/Nicosia',
|
||||||
|
'Asia/Novokuznetsk',
|
||||||
|
'Asia/Novosibirsk',
|
||||||
|
'Asia/Omsk',
|
||||||
|
'Asia/Oral',
|
||||||
|
'Asia/Phnom_Penh',
|
||||||
|
'Asia/Pontianak',
|
||||||
|
'Asia/Pyongyang',
|
||||||
|
'Asia/Qatar',
|
||||||
|
'Asia/Qostanay',
|
||||||
|
'Asia/Qyzylorda',
|
||||||
|
'Asia/Rangoon',
|
||||||
|
'Asia/Riyadh',
|
||||||
|
'Asia/Saigon',
|
||||||
|
'Asia/Sakhalin',
|
||||||
|
'Asia/Samarkand',
|
||||||
|
'Asia/Seoul',
|
||||||
|
'Asia/Shanghai',
|
||||||
|
'Asia/Singapore',
|
||||||
|
'Asia/Srednekolymsk',
|
||||||
|
'Asia/Taipei',
|
||||||
|
'Asia/Tashkent',
|
||||||
|
'Asia/Tbilisi',
|
||||||
|
'Asia/Tehran',
|
||||||
|
'Asia/Tel_Aviv',
|
||||||
|
'Asia/Thimbu',
|
||||||
|
'Asia/Thimphu',
|
||||||
|
'Asia/Tokyo',
|
||||||
|
'Asia/Tomsk',
|
||||||
|
'Asia/Ujung_Pandang',
|
||||||
|
'Asia/Ulaanbaatar',
|
||||||
|
'Asia/Ulan_Bator',
|
||||||
|
'Asia/Urumqi',
|
||||||
|
'Asia/Ust-Nera',
|
||||||
|
'Asia/Vientiane',
|
||||||
|
'Asia/Vladivostok',
|
||||||
|
'Asia/Yakutsk',
|
||||||
|
'Asia/Yangon',
|
||||||
|
'Asia/Yekaterinburg',
|
||||||
|
'Asia/Yerevan',
|
||||||
|
'Atlantic/Azores',
|
||||||
|
'Atlantic/Bermuda',
|
||||||
|
'Atlantic/Canary',
|
||||||
|
'Atlantic/Cape_Verde',
|
||||||
|
'Atlantic/Faeroe',
|
||||||
|
'Atlantic/Faroe',
|
||||||
|
'Atlantic/Jan_Mayen',
|
||||||
|
'Atlantic/Madeira',
|
||||||
|
'Atlantic/Reykjavik',
|
||||||
|
'Atlantic/South_Georgia',
|
||||||
|
'Atlantic/St_Helena',
|
||||||
|
'Atlantic/Stanley',
|
||||||
|
'Australia/ACT',
|
||||||
|
'Australia/Adelaide',
|
||||||
|
'Australia/Brisbane',
|
||||||
|
'Australia/Broken_Hill',
|
||||||
|
'Australia/Canberra',
|
||||||
|
'Australia/Currie',
|
||||||
|
'Australia/Darwin',
|
||||||
|
'Australia/Eucla',
|
||||||
|
'Australia/Hobart',
|
||||||
|
'Australia/LHI',
|
||||||
|
'Australia/Lindeman',
|
||||||
|
'Australia/Lord_Howe',
|
||||||
|
'Australia/Melbourne',
|
||||||
|
'Australia/NSW',
|
||||||
|
'Australia/North',
|
||||||
|
'Australia/Perth',
|
||||||
|
'Australia/Queensland',
|
||||||
|
'Australia/South',
|
||||||
|
'Australia/Sydney',
|
||||||
|
'Australia/Tasmania',
|
||||||
|
'Australia/Victoria',
|
||||||
|
'Australia/West',
|
||||||
|
'Australia/Yancowinna',
|
||||||
|
'Brazil/Acre',
|
||||||
|
'Brazil/DeNoronha',
|
||||||
|
'Brazil/East',
|
||||||
|
'Brazil/West',
|
||||||
|
'CET',
|
||||||
|
'CST6CDT',
|
||||||
|
'Canada/Atlantic',
|
||||||
|
'Canada/Central',
|
||||||
|
'Canada/Eastern',
|
||||||
|
'Canada/Mountain',
|
||||||
|
'Canada/Newfoundland',
|
||||||
|
'Canada/Pacific',
|
||||||
|
'Canada/Saskatchewan',
|
||||||
|
'Canada/Yukon',
|
||||||
|
'Chile/Continental',
|
||||||
|
'Chile/EasterIsland',
|
||||||
|
'Cuba',
|
||||||
|
'EET',
|
||||||
|
'EST',
|
||||||
|
'EST5EDT',
|
||||||
|
'Egypt',
|
||||||
|
'Eire',
|
||||||
|
'Etc/GMT',
|
||||||
|
'Etc/GMT+0',
|
||||||
|
'Etc/GMT+1',
|
||||||
|
'Etc/GMT+10',
|
||||||
|
'Etc/GMT+11',
|
||||||
|
'Etc/GMT+12',
|
||||||
|
'Etc/GMT+2',
|
||||||
|
'Etc/GMT+3',
|
||||||
|
'Etc/GMT+4',
|
||||||
|
'Etc/GMT+5',
|
||||||
|
'Etc/GMT+6',
|
||||||
|
'Etc/GMT+7',
|
||||||
|
'Etc/GMT+8',
|
||||||
|
'Etc/GMT+9',
|
||||||
|
'Etc/GMT-0',
|
||||||
|
'Etc/GMT-1',
|
||||||
|
'Etc/GMT-10',
|
||||||
|
'Etc/GMT-11',
|
||||||
|
'Etc/GMT-12',
|
||||||
|
'Etc/GMT-13',
|
||||||
|
'Etc/GMT-14',
|
||||||
|
'Etc/GMT-2',
|
||||||
|
'Etc/GMT-3',
|
||||||
|
'Etc/GMT-4',
|
||||||
|
'Etc/GMT-5',
|
||||||
|
'Etc/GMT-6',
|
||||||
|
'Etc/GMT-7',
|
||||||
|
'Etc/GMT-8',
|
||||||
|
'Etc/GMT-9',
|
||||||
|
'Etc/GMT0',
|
||||||
|
'Etc/Greenwich',
|
||||||
|
'Etc/UCT',
|
||||||
|
'Etc/UTC',
|
||||||
|
'Etc/Universal',
|
||||||
|
'Etc/Zulu',
|
||||||
|
'Europe/Amsterdam',
|
||||||
|
'Europe/Andorra',
|
||||||
|
'Europe/Astrakhan',
|
||||||
|
'Europe/Athens',
|
||||||
|
'Europe/Belfast',
|
||||||
|
'Europe/Belgrade',
|
||||||
|
'Europe/Berlin',
|
||||||
|
'Europe/Bratislava',
|
||||||
|
'Europe/Brussels',
|
||||||
|
'Europe/Bucharest',
|
||||||
|
'Europe/Budapest',
|
||||||
|
'Europe/Busingen',
|
||||||
|
'Europe/Chisinau',
|
||||||
|
'Europe/Copenhagen',
|
||||||
|
'Europe/Dublin',
|
||||||
|
'Europe/Gibraltar',
|
||||||
|
'Europe/Guernsey',
|
||||||
|
'Europe/Helsinki',
|
||||||
|
'Europe/Isle_of_Man',
|
||||||
|
'Europe/Istanbul',
|
||||||
|
'Europe/Jersey',
|
||||||
|
'Europe/Kaliningrad',
|
||||||
|
'Europe/Kiev',
|
||||||
|
'Europe/Kirov',
|
||||||
|
'Europe/Kyiv',
|
||||||
|
'Europe/Lisbon',
|
||||||
|
'Europe/Ljubljana',
|
||||||
|
'Europe/London',
|
||||||
|
'Europe/Luxembourg',
|
||||||
|
'Europe/Madrid',
|
||||||
|
'Europe/Malta',
|
||||||
|
'Europe/Mariehamn',
|
||||||
|
'Europe/Minsk',
|
||||||
|
'Europe/Monaco',
|
||||||
|
'Europe/Moscow',
|
||||||
|
'Europe/Nicosia',
|
||||||
|
'Europe/Oslo',
|
||||||
|
'Europe/Paris',
|
||||||
|
'Europe/Podgorica',
|
||||||
|
'Europe/Prague',
|
||||||
|
'Europe/Riga',
|
||||||
|
'Europe/Rome',
|
||||||
|
'Europe/Samara',
|
||||||
|
'Europe/San_Marino',
|
||||||
|
'Europe/Sarajevo',
|
||||||
|
'Europe/Saratov',
|
||||||
|
'Europe/Simferopol',
|
||||||
|
'Europe/Skopje',
|
||||||
|
'Europe/Sofia',
|
||||||
|
'Europe/Stockholm',
|
||||||
|
'Europe/Tallinn',
|
||||||
|
'Europe/Tirane',
|
||||||
|
'Europe/Tiraspol',
|
||||||
|
'Europe/Ulyanovsk',
|
||||||
|
'Europe/Uzhgorod',
|
||||||
|
'Europe/Vaduz',
|
||||||
|
'Europe/Vatican',
|
||||||
|
'Europe/Vienna',
|
||||||
|
'Europe/Vilnius',
|
||||||
|
'Europe/Volgograd',
|
||||||
|
'Europe/Warsaw',
|
||||||
|
'Europe/Zagreb',
|
||||||
|
'Europe/Zaporozhye',
|
||||||
|
'Europe/Zurich',
|
||||||
|
'Factory',
|
||||||
|
'GB',
|
||||||
|
'GB-Eire',
|
||||||
|
'GMT',
|
||||||
|
'GMT+0',
|
||||||
|
'GMT-0',
|
||||||
|
'GMT0',
|
||||||
|
'Greenwich',
|
||||||
|
'HST',
|
||||||
|
'Hongkong',
|
||||||
|
'Iceland',
|
||||||
|
'Indian/Antananarivo',
|
||||||
|
'Indian/Chagos',
|
||||||
|
'Indian/Christmas',
|
||||||
|
'Indian/Cocos',
|
||||||
|
'Indian/Comoro',
|
||||||
|
'Indian/Kerguelen',
|
||||||
|
'Indian/Mahe',
|
||||||
|
'Indian/Maldives',
|
||||||
|
'Indian/Mauritius',
|
||||||
|
'Indian/Mayotte',
|
||||||
|
'Indian/Reunion',
|
||||||
|
'Iran',
|
||||||
|
'Israel',
|
||||||
|
'Jamaica',
|
||||||
|
'Japan',
|
||||||
|
'Kwajalein',
|
||||||
|
'Libya',
|
||||||
|
'MET',
|
||||||
|
'MST',
|
||||||
|
'MST7MDT',
|
||||||
|
'Mexico/BajaNorte',
|
||||||
|
'Mexico/BajaSur',
|
||||||
|
'Mexico/General',
|
||||||
|
'NZ',
|
||||||
|
'NZ-CHAT',
|
||||||
|
'Navajo',
|
||||||
|
'PRC',
|
||||||
|
'PST8PDT',
|
||||||
|
'Pacific/Apia',
|
||||||
|
'Pacific/Auckland',
|
||||||
|
'Pacific/Bougainville',
|
||||||
|
'Pacific/Chatham',
|
||||||
|
'Pacific/Chuuk',
|
||||||
|
'Pacific/Easter',
|
||||||
|
'Pacific/Efate',
|
||||||
|
'Pacific/Enderbury',
|
||||||
|
'Pacific/Fakaofo',
|
||||||
|
'Pacific/Fiji',
|
||||||
|
'Pacific/Funafuti',
|
||||||
|
'Pacific/Galapagos',
|
||||||
|
'Pacific/Gambier',
|
||||||
|
'Pacific/Guadalcanal',
|
||||||
|
'Pacific/Guam',
|
||||||
|
'Pacific/Honolulu',
|
||||||
|
'Pacific/Johnston',
|
||||||
|
'Pacific/Kanton',
|
||||||
|
'Pacific/Kiritimati',
|
||||||
|
'Pacific/Kosrae',
|
||||||
|
'Pacific/Kwajalein',
|
||||||
|
'Pacific/Majuro',
|
||||||
|
'Pacific/Marquesas',
|
||||||
|
'Pacific/Midway',
|
||||||
|
'Pacific/Nauru',
|
||||||
|
'Pacific/Niue',
|
||||||
|
'Pacific/Norfolk',
|
||||||
|
'Pacific/Noumea',
|
||||||
|
'Pacific/Pago_Pago',
|
||||||
|
'Pacific/Palau',
|
||||||
|
'Pacific/Pitcairn',
|
||||||
|
'Pacific/Pohnpei',
|
||||||
|
'Pacific/Ponape',
|
||||||
|
'Pacific/Port_Moresby',
|
||||||
|
'Pacific/Rarotonga',
|
||||||
|
'Pacific/Saipan',
|
||||||
|
'Pacific/Samoa',
|
||||||
|
'Pacific/Tahiti',
|
||||||
|
'Pacific/Tarawa',
|
||||||
|
'Pacific/Tongatapu',
|
||||||
|
'Pacific/Truk',
|
||||||
|
'Pacific/Wake',
|
||||||
|
'Pacific/Wallis',
|
||||||
|
'Pacific/Yap',
|
||||||
|
'Poland',
|
||||||
|
'Portugal',
|
||||||
|
'ROC',
|
||||||
|
'ROK',
|
||||||
|
'Singapore',
|
||||||
|
'Turkey',
|
||||||
|
'UCT',
|
||||||
|
'US/Alaska',
|
||||||
|
'US/Aleutian',
|
||||||
|
'US/Arizona',
|
||||||
|
'US/Central',
|
||||||
|
'US/East-Indiana',
|
||||||
|
'US/Eastern',
|
||||||
|
'US/Hawaii',
|
||||||
|
'US/Indiana-Starke',
|
||||||
|
'US/Michigan',
|
||||||
|
'US/Mountain',
|
||||||
|
'US/Pacific',
|
||||||
|
'US/Samoa',
|
||||||
|
'UTC',
|
||||||
|
'Universal',
|
||||||
|
'W-SU',
|
||||||
|
'WET',
|
||||||
|
'Zulu',
|
||||||
|
];
|
||||||
@@ -16,9 +16,14 @@ export const timeWindows = {
|
|||||||
},
|
},
|
||||||
today: {
|
today: {
|
||||||
key: 'today',
|
key: 'today',
|
||||||
label: '24 hours',
|
label: 'Today',
|
||||||
shortcut: 'D',
|
shortcut: 'D',
|
||||||
},
|
},
|
||||||
|
yesterday: {
|
||||||
|
key: 'yesterday',
|
||||||
|
label: 'Yesterday',
|
||||||
|
shortcut: 'E',
|
||||||
|
},
|
||||||
'7d': {
|
'7d': {
|
||||||
key: '7d',
|
key: '7d',
|
||||||
label: 'Last 7 days',
|
label: 'Last 7 days',
|
||||||
@@ -29,6 +34,16 @@ export const timeWindows = {
|
|||||||
label: 'Last 30 days',
|
label: 'Last 30 days',
|
||||||
shortcut: 'T',
|
shortcut: 'T',
|
||||||
},
|
},
|
||||||
|
'6m': {
|
||||||
|
key: '6m',
|
||||||
|
label: 'Last 6 months',
|
||||||
|
shortcut: '6',
|
||||||
|
},
|
||||||
|
'12m': {
|
||||||
|
key: '12m',
|
||||||
|
label: 'Last 12 months',
|
||||||
|
shortcut: '0',
|
||||||
|
},
|
||||||
monthToDate: {
|
monthToDate: {
|
||||||
key: 'monthToDate',
|
key: 'monthToDate',
|
||||||
label: 'Month to Date',
|
label: 'Month to Date',
|
||||||
@@ -167,7 +182,10 @@ export function isMinuteIntervalEnabledByRange(
|
|||||||
|
|
||||||
export function isHourIntervalEnabledByRange(range: keyof typeof timeWindows) {
|
export function isHourIntervalEnabledByRange(range: keyof typeof timeWindows) {
|
||||||
return (
|
return (
|
||||||
isMinuteIntervalEnabledByRange(range) || range === 'today' || range === '7d'
|
isMinuteIntervalEnabledByRange(range) ||
|
||||||
|
range === 'today' ||
|
||||||
|
range === 'yesterday' ||
|
||||||
|
range === '7d'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -177,7 +195,7 @@ export function getDefaultIntervalByRange(
|
|||||||
if (range === '30min' || range === 'lastHour') {
|
if (range === '30min' || range === 'lastHour') {
|
||||||
return 'minute';
|
return 'minute';
|
||||||
}
|
}
|
||||||
if (range === 'today') {
|
if (range === 'today' || range === 'yesterday') {
|
||||||
return 'hour';
|
return 'hour';
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
|
|||||||
@@ -0,0 +1,2 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "organizations" ADD COLUMN "timezone" TEXT;
|
||||||
@@ -54,6 +54,7 @@ model Organization {
|
|||||||
ShareOverview ShareOverview[]
|
ShareOverview ShareOverview[]
|
||||||
integrations Integration[]
|
integrations Integration[]
|
||||||
invites Invite[]
|
invites Invite[]
|
||||||
|
timezone String?
|
||||||
|
|
||||||
// Subscription
|
// Subscription
|
||||||
subscriptionId String?
|
subscriptionId String?
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { ResponseJSON } from '@clickhouse/client';
|
import type { ClickHouseSettings, ResponseJSON } from '@clickhouse/client';
|
||||||
import { ClickHouseLogLevel, createClient } from '@clickhouse/client';
|
import { ClickHouseLogLevel, createClient } from '@clickhouse/client';
|
||||||
import { escape } from 'sqlstring';
|
import { escape } from 'sqlstring';
|
||||||
|
|
||||||
@@ -11,7 +11,6 @@ export { createClient };
|
|||||||
const logger = createLogger({ name: 'clickhouse' });
|
const logger = createLogger({ name: 'clickhouse' });
|
||||||
|
|
||||||
import type { Logger } from '@clickhouse/client';
|
import type { Logger } from '@clickhouse/client';
|
||||||
import { getTimezoneFromDateString } from '@openpanel/common';
|
|
||||||
|
|
||||||
// All three LogParams types are exported by the client
|
// All three LogParams types are exported by the client
|
||||||
interface LogParams {
|
interface LogParams {
|
||||||
@@ -142,10 +141,12 @@ export const ch = new Proxy(originalCh, {
|
|||||||
|
|
||||||
export async function chQueryWithMeta<T extends Record<string, any>>(
|
export async function chQueryWithMeta<T extends Record<string, any>>(
|
||||||
query: string,
|
query: string,
|
||||||
|
clickhouseSettings?: ClickHouseSettings,
|
||||||
): Promise<ResponseJSON<T>> {
|
): Promise<ResponseJSON<T>> {
|
||||||
const start = Date.now();
|
const start = Date.now();
|
||||||
const res = await ch.query({
|
const res = await ch.query({
|
||||||
query,
|
query,
|
||||||
|
clickhouse_settings: clickhouseSettings,
|
||||||
});
|
});
|
||||||
const json = await res.json<T>();
|
const json = await res.json<T>();
|
||||||
const keys = Object.keys(json.data[0] || {});
|
const keys = Object.keys(json.data[0] || {});
|
||||||
@@ -170,6 +171,7 @@ export async function chQueryWithMeta<T extends Record<string, any>>(
|
|||||||
rows: json.rows,
|
rows: json.rows,
|
||||||
stats: response.statistics,
|
stats: response.statistics,
|
||||||
elapsed: Date.now() - start,
|
elapsed: Date.now() - start,
|
||||||
|
clickhouseSettings,
|
||||||
});
|
});
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
@@ -177,8 +179,9 @@ export async function chQueryWithMeta<T extends Record<string, any>>(
|
|||||||
|
|
||||||
export async function chQuery<T extends Record<string, any>>(
|
export async function chQuery<T extends Record<string, any>>(
|
||||||
query: string,
|
query: string,
|
||||||
|
clickhouseSettings?: ClickHouseSettings,
|
||||||
): Promise<T[]> {
|
): Promise<T[]> {
|
||||||
return (await chQueryWithMeta<T>(query)).data;
|
return (await chQueryWithMeta<T>(query, clickhouseSettings)).data;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function formatClickhouseDate(
|
export function formatClickhouseDate(
|
||||||
@@ -188,7 +191,10 @@ export function formatClickhouseDate(
|
|||||||
if (skipTime) {
|
if (skipTime) {
|
||||||
return new Date(date).toISOString().split('T')[0]!;
|
return new Date(date).toISOString().split('T')[0]!;
|
||||||
}
|
}
|
||||||
return new Date(date).toISOString().replace('T', ' ').replace(/Z+$/, '');
|
return new Date(date)
|
||||||
|
.toISOString()
|
||||||
|
.replace('T', ' ')
|
||||||
|
.replace(/(\.\d{3})?Z+$/, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
export function toDate(str: string, interval?: IInterval) {
|
export function toDate(str: string, interval?: IInterval) {
|
||||||
|
|||||||
@@ -73,7 +73,11 @@ export class Query<T = any> {
|
|||||||
};
|
};
|
||||||
private _transform?: Record<string, (item: T) => any>;
|
private _transform?: Record<string, (item: T) => any>;
|
||||||
private _union?: Query;
|
private _union?: Query;
|
||||||
constructor(private client: ClickHouseClient) {}
|
private _dateRegex = /\d{4}-\d{2}-\d{2}([\s\:\d\.]+)?/g;
|
||||||
|
constructor(
|
||||||
|
private client: ClickHouseClient,
|
||||||
|
private timezone: string,
|
||||||
|
) {}
|
||||||
|
|
||||||
// Select methods
|
// Select methods
|
||||||
select<U>(
|
select<U>(
|
||||||
@@ -121,9 +125,14 @@ export class Query<T = any> {
|
|||||||
if (Array.isArray(value)) {
|
if (Array.isArray(value)) {
|
||||||
return `(${value.map((v) => this.escapeValue(v)).join(', ')})`;
|
return `(${value.map((v) => this.escapeValue(v)).join(', ')})`;
|
||||||
}
|
}
|
||||||
if (value instanceof Date) {
|
|
||||||
return escape(clix.datetime(value));
|
if (
|
||||||
|
(typeof value === 'string' && this._dateRegex.test(value)) ||
|
||||||
|
value instanceof Date
|
||||||
|
) {
|
||||||
|
return this.escapeDate(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
return escape(value);
|
return escape(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -249,10 +258,10 @@ export class Query<T = any> {
|
|||||||
|
|
||||||
private escapeDate(value: string | Date): string {
|
private escapeDate(value: string | Date): string {
|
||||||
if (value instanceof Date) {
|
if (value instanceof Date) {
|
||||||
return clix.datetime(value);
|
return escape(clix.datetime(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
return value.replaceAll(/\d{4}-\d{2}-\d{2}([\s\:\d\.]+)?/g, (match) => {
|
return value.replaceAll(this._dateRegex, (match) => {
|
||||||
return escape(match);
|
return escape(match);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -348,7 +357,10 @@ export class Query<T = any> {
|
|||||||
|
|
||||||
// SELECT
|
// SELECT
|
||||||
if (this._select.length > 0) {
|
if (this._select.length > 0) {
|
||||||
parts.push('SELECT', this._select.map(this.escapeDate).join(', '));
|
parts.push(
|
||||||
|
'SELECT',
|
||||||
|
this._select.map((col) => this.escapeDate(col)).join(', '),
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
parts.push('SELECT *');
|
parts.push('SELECT *');
|
||||||
}
|
}
|
||||||
@@ -483,26 +495,16 @@ export class Query<T = any> {
|
|||||||
// Execution methods
|
// Execution methods
|
||||||
async execute(): Promise<T[]> {
|
async execute(): Promise<T[]> {
|
||||||
const query = this.buildQuery();
|
const query = this.buildQuery();
|
||||||
console.log('TEST QUERY ----->');
|
console.log('query', query);
|
||||||
console.log(query);
|
|
||||||
console.log('<----------');
|
const result = await this.client.query({
|
||||||
const perf = performance.now();
|
query,
|
||||||
try {
|
clickhouse_settings: {
|
||||||
const result = await this.client.query({
|
session_timezone: this.timezone,
|
||||||
query,
|
},
|
||||||
});
|
});
|
||||||
const json = await result.json<T>();
|
const json = await result.json<T>();
|
||||||
const perf2 = performance.now();
|
return this.transformJson(json).data;
|
||||||
console.log(`PERF: ${perf2 - perf}ms`);
|
|
||||||
return this.transformJson(json).data;
|
|
||||||
} catch (error) {
|
|
||||||
console.log('ERROR ----->');
|
|
||||||
console.log(error);
|
|
||||||
console.log('<----------');
|
|
||||||
console.log(query);
|
|
||||||
console.log('<----------');
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Debug methods
|
// Debug methods
|
||||||
@@ -535,7 +537,7 @@ export class Query<T = any> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
clone(): Query<T> {
|
clone(): Query<T> {
|
||||||
return new Query(this.client).merge(this);
|
return new Query(this.client, this.timezone).merge(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add merge method
|
// Add merge method
|
||||||
@@ -629,12 +631,8 @@ export class WhereGroupBuilder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Helper function to create a new query
|
// Helper function to create a new query
|
||||||
export function createQuery(client: ClickHouseClient): Query {
|
export function clix(client: ClickHouseClient, timezone?: string): Query {
|
||||||
return new Query(client);
|
return new Query(client, timezone ?? 'UTC');
|
||||||
}
|
|
||||||
|
|
||||||
export function clix(client: ClickHouseClient): Query {
|
|
||||||
return new Query(client);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
clix.exp = (expr: string | Query<any>) =>
|
clix.exp = (expr: string | Query<any>) =>
|
||||||
@@ -654,7 +652,7 @@ clix.dynamicDatetime = (date: string | Date, interval: IInterval) => {
|
|||||||
return clix.datetime(date);
|
return clix.datetime(date);
|
||||||
};
|
};
|
||||||
|
|
||||||
clix.toStartOf = (node: string, interval: IInterval) => {
|
clix.toStartOf = (node: string, interval: IInterval, timezone?: string) => {
|
||||||
switch (interval) {
|
switch (interval) {
|
||||||
case 'minute': {
|
case 'minute': {
|
||||||
return `toStartOfMinute(${node})`;
|
return `toStartOfMinute(${node})`;
|
||||||
@@ -666,10 +664,12 @@ clix.toStartOf = (node: string, interval: IInterval) => {
|
|||||||
return `toStartOfDay(${node})`;
|
return `toStartOfDay(${node})`;
|
||||||
}
|
}
|
||||||
case 'week': {
|
case 'week': {
|
||||||
return `toStartOfWeek(${node})`;
|
// Does not respect timezone settings (session_timezone) so we need to pass it manually
|
||||||
|
return `toStartOfWeek(${node}${timezone ? `, 1, '${timezone}'` : ''})`;
|
||||||
}
|
}
|
||||||
case 'month': {
|
case 'month': {
|
||||||
return `toStartOfMonth(${node})`;
|
// Does not respect timezone settings (session_timezone) so we need to pass it manually
|
||||||
|
return `toStartOfMonth(${node}${timezone ? `, '${timezone}'` : ''})`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,19 +1,12 @@
|
|||||||
import { escape } from 'sqlstring';
|
import { escape } from 'sqlstring';
|
||||||
|
|
||||||
import {
|
import { stripLeadingAndTrailingSlashes } from '@openpanel/common';
|
||||||
getTimezoneFromDateString,
|
|
||||||
stripLeadingAndTrailingSlashes,
|
|
||||||
} from '@openpanel/common';
|
|
||||||
import type {
|
import type {
|
||||||
IChartEventFilter,
|
IChartEventFilter,
|
||||||
IGetChartDataInput,
|
IGetChartDataInput,
|
||||||
} from '@openpanel/validation';
|
} from '@openpanel/validation';
|
||||||
|
|
||||||
import {
|
import { TABLE_NAMES, formatClickhouseDate } from '../clickhouse/client';
|
||||||
TABLE_NAMES,
|
|
||||||
formatClickhouseDate,
|
|
||||||
toDate,
|
|
||||||
} from '../clickhouse/client';
|
|
||||||
import { createSqlBuilder } from '../sql-builder';
|
import { createSqlBuilder } from '../sql-builder';
|
||||||
|
|
||||||
export function transformPropertyKey(property: string) {
|
export function transformPropertyKey(property: string) {
|
||||||
@@ -61,9 +54,9 @@ export function getChartSql({
|
|||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
projectId,
|
projectId,
|
||||||
chartType,
|
|
||||||
limit,
|
limit,
|
||||||
}: IGetChartDataInput) {
|
timezone,
|
||||||
|
}: IGetChartDataInput & { timezone: string }) {
|
||||||
const {
|
const {
|
||||||
sb,
|
sb,
|
||||||
join,
|
join,
|
||||||
@@ -73,6 +66,7 @@ export function getChartSql({
|
|||||||
getSelect,
|
getSelect,
|
||||||
getOrderBy,
|
getOrderBy,
|
||||||
getGroupBy,
|
getGroupBy,
|
||||||
|
getFill,
|
||||||
} = createSqlBuilder();
|
} = createSqlBuilder();
|
||||||
|
|
||||||
sb.where = getEventFiltersWhereClause(event.filters);
|
sb.where = getEventFiltersWhereClause(event.filters);
|
||||||
@@ -99,34 +93,40 @@ export function getChartSql({
|
|||||||
sb.select.count = 'count(*) as count';
|
sb.select.count = 'count(*) as count';
|
||||||
switch (interval) {
|
switch (interval) {
|
||||||
case 'minute': {
|
case 'minute': {
|
||||||
sb.select.date = `toStartOfMinute(toTimeZone(created_at, '${getTimezoneFromDateString(startDate)}')) as date`;
|
sb.fill = `FROM toStartOfMinute(toDateTime('${startDate}')) TO toStartOfMinute(toDateTime('${endDate}')) STEP toIntervalMinute(1)`;
|
||||||
|
sb.select.date = 'toStartOfMinute(created_at) as date';
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 'hour': {
|
case 'hour': {
|
||||||
sb.select.date = `toStartOfHour(toTimeZone(created_at, '${getTimezoneFromDateString(startDate)}')) as date`;
|
sb.fill = `FROM toStartOfHour(toDateTime('${startDate}')) TO toStartOfHour(toDateTime('${endDate}')) STEP toIntervalHour(1)`;
|
||||||
|
sb.select.date = 'toStartOfHour(created_at) as date';
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 'day': {
|
case 'day': {
|
||||||
sb.select.date = `toStartOfDay(toTimeZone(created_at, '${getTimezoneFromDateString(startDate)}')) as date`;
|
sb.fill = `FROM toStartOfDay(toDateTime('${startDate}')) TO toStartOfDay(toDateTime('${endDate}')) STEP toIntervalDay(1)`;
|
||||||
|
sb.select.date = 'toStartOfDay(created_at) as date';
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 'week': {
|
case 'week': {
|
||||||
sb.select.date = `toStartOfWeek(toTimeZone(created_at, '${getTimezoneFromDateString(startDate)}')) as date`;
|
sb.fill = `FROM toStartOfWeek(toDateTime('${startDate}'), 1, '${timezone}') TO toStartOfWeek(toDateTime('${endDate}'), 1, '${timezone}') STEP toIntervalWeek(1)`;
|
||||||
|
sb.select.date = `toStartOfWeek(created_at, 1, '${timezone}') as date`;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 'month': {
|
case 'month': {
|
||||||
sb.select.date = `toStartOfMonth(toTimeZone(created_at, '${getTimezoneFromDateString(startDate)}')) as date`;
|
sb.fill = `FROM toStartOfMonth(toDateTime('${startDate}'), '${timezone}') TO toStartOfMonth(toDateTime('${endDate}'), '${timezone}') STEP toIntervalMonth(1)`;
|
||||||
|
sb.select.date = `toStartOfMonth(created_at, '${timezone}') as date`;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
sb.groupBy.date = 'date';
|
sb.groupBy.date = 'date';
|
||||||
|
sb.orderBy.date = 'date ASC';
|
||||||
|
|
||||||
if (startDate) {
|
if (startDate) {
|
||||||
sb.where.startDate = `${toDate('created_at', interval)} >= ${toDate(formatClickhouseDate(startDate), interval)}`;
|
sb.where.startDate = `created_at >= toDateTime('${formatClickhouseDate(startDate)}')`;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (endDate) {
|
if (endDate) {
|
||||||
sb.where.endDate = `${toDate('created_at', interval)} <= ${toDate(formatClickhouseDate(endDate), interval)}`;
|
sb.where.endDate = `created_at <= toDateTime('${formatClickhouseDate(endDate)}')`;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (breakdowns.length > 0 && limit) {
|
if (breakdowns.length > 0 && limit) {
|
||||||
@@ -179,18 +179,14 @@ export function getChartSql({
|
|||||||
ORDER BY profile_id, created_at DESC
|
ORDER BY profile_id, created_at DESC
|
||||||
) as subQuery`;
|
) as subQuery`;
|
||||||
|
|
||||||
console.log(
|
const sql = `${getSelect()} ${getFrom()} ${getJoins()} ${getWhere()} ${getGroupBy()} ${getOrderBy()} ${getFill()}`;
|
||||||
`${getSelect()} ${getFrom()} ${getJoins()} ${getWhere()} ${getGroupBy()} ${getOrderBy()}`,
|
console.log('CHART SQL', sql);
|
||||||
);
|
return sql;
|
||||||
|
|
||||||
return `${getSelect()} ${getFrom()} ${getJoins()} ${getWhere()} ${getGroupBy()} ${getOrderBy()}`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(
|
const sql = `${getSelect()} ${getFrom()} ${getJoins()} ${getWhere()} ${getGroupBy()} ${getOrderBy()} ${getFill()}`;
|
||||||
`${getSelect()} ${getFrom()} ${getJoins()} ${getWhere()} ${getGroupBy()} ${getOrderBy()}`,
|
console.log('CHART SQL', sql);
|
||||||
);
|
return sql;
|
||||||
|
|
||||||
return `${getSelect()} ${getFrom()} ${getJoins()} ${getWhere()} ${getGroupBy()} ${getOrderBy()}`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getEventFiltersWhereClause(filters: IChartEventFilter[]) {
|
export function getEventFiltersWhereClause(filters: IChartEventFilter[]) {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { ClickHouseClient } from '@clickhouse/client';
|
import type { ClickHouseClient } from '@clickhouse/client';
|
||||||
import { Query, createQuery } from '../clickhouse/query-builder';
|
import { clix } from '../clickhouse/query-builder';
|
||||||
|
|
||||||
export interface Insight {
|
export interface Insight {
|
||||||
type: string;
|
type: string;
|
||||||
@@ -73,7 +73,7 @@ export class InsightsService {
|
|||||||
constructor(private client: ClickHouseClient) {}
|
constructor(private client: ClickHouseClient) {}
|
||||||
|
|
||||||
private async getTrafficSpikes(projectId: string): Promise<Insight[]> {
|
private async getTrafficSpikes(projectId: string): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'referrer_name',
|
'referrer_name',
|
||||||
'toDate(created_at) as date',
|
'toDate(created_at) as date',
|
||||||
@@ -100,7 +100,7 @@ export class InsightsService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async getEventSurges(projectId: string): Promise<Insight[]> {
|
private async getEventSurges(projectId: string): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'toDate(created_at) as date',
|
'toDate(created_at) as date',
|
||||||
'COUNT(*) as event_count',
|
'COUNT(*) as event_count',
|
||||||
@@ -126,7 +126,7 @@ export class InsightsService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async getNewVisitorTrends(projectId: string): Promise<Insight[]> {
|
private async getNewVisitorTrends(projectId: string): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'toMonth(created_at) as month',
|
'toMonth(created_at) as month',
|
||||||
'COUNT(DISTINCT device_id) as new_visitors',
|
'COUNT(DISTINCT device_id) as new_visitors',
|
||||||
@@ -155,7 +155,7 @@ export class InsightsService {
|
|||||||
private async getReferralSourceHighlights(
|
private async getReferralSourceHighlights(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
): Promise<Insight[]> {
|
): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'referrer_name',
|
'referrer_name',
|
||||||
'COUNT(*) as count',
|
'COUNT(*) as count',
|
||||||
@@ -179,7 +179,7 @@ export class InsightsService {
|
|||||||
private async getSessionDurationChanges(
|
private async getSessionDurationChanges(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
): Promise<Insight[]> {
|
): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'toWeek(created_at) as week',
|
'toWeek(created_at) as week',
|
||||||
'avg(duration) as avg_duration',
|
'avg(duration) as avg_duration',
|
||||||
@@ -205,7 +205,7 @@ export class InsightsService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async getTopPerformingContent(projectId: string): Promise<Insight[]> {
|
private async getTopPerformingContent(projectId: string): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'path',
|
'path',
|
||||||
'COUNT(*) as view_count',
|
'COUNT(*) as view_count',
|
||||||
@@ -233,7 +233,7 @@ export class InsightsService {
|
|||||||
private async getBounceRateImprovements(
|
private async getBounceRateImprovements(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
): Promise<Insight[]> {
|
): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'toMonth(created_at) as month',
|
'toMonth(created_at) as month',
|
||||||
'sum(is_bounce) / COUNT(*) as bounce_rate',
|
'sum(is_bounce) / COUNT(*) as bounce_rate',
|
||||||
@@ -261,7 +261,7 @@ export class InsightsService {
|
|||||||
private async getReturningVisitorTrends(
|
private async getReturningVisitorTrends(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
): Promise<Insight[]> {
|
): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'toQuarter(created_at) as quarter',
|
'toQuarter(created_at) as quarter',
|
||||||
'COUNT(DISTINCT device_id) as returning_visitors',
|
'COUNT(DISTINCT device_id) as returning_visitors',
|
||||||
@@ -290,7 +290,7 @@ export class InsightsService {
|
|||||||
private async getGeographicInterestShifts(
|
private async getGeographicInterestShifts(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
): Promise<Insight[]> {
|
): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'country',
|
'country',
|
||||||
'COUNT(*) as visitor_count',
|
'COUNT(*) as visitor_count',
|
||||||
@@ -318,7 +318,7 @@ export class InsightsService {
|
|||||||
private async getEventCompletionChanges(
|
private async getEventCompletionChanges(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
): Promise<Insight[]> {
|
): Promise<Insight[]> {
|
||||||
const query = createQuery(this.client)
|
const query = clix(this.client)
|
||||||
.select([
|
.select([
|
||||||
'event_name',
|
'event_name',
|
||||||
'toMonth(created_at) as month',
|
'toMonth(created_at) as month',
|
||||||
|
|||||||
@@ -14,10 +14,6 @@ export type IServiceMember = Prisma.MemberGetPayload<{
|
|||||||
}> & { access: ProjectAccess[] };
|
}> & { access: ProjectAccess[] };
|
||||||
export type IServiceProjectAccess = ProjectAccess;
|
export type IServiceProjectAccess = ProjectAccess;
|
||||||
|
|
||||||
export function transformOrganization<T>(org: T) {
|
|
||||||
return org;
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getOrganizations(userId: string | null) {
|
export async function getOrganizations(userId: string | null) {
|
||||||
if (!userId) return [];
|
if (!userId) return [];
|
||||||
|
|
||||||
@@ -34,10 +30,10 @@ export async function getOrganizations(userId: string | null) {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return organizations.map(transformOrganization);
|
return organizations;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getOrganizationBySlug(slug: string) {
|
export function getOrganizationById(slug: string) {
|
||||||
return db.organization.findUniqueOrThrow({
|
return db.organization.findUniqueOrThrow({
|
||||||
where: {
|
where: {
|
||||||
id: slug,
|
id: slug,
|
||||||
@@ -59,7 +55,7 @@ export async function getOrganizationByProjectId(projectId: string) {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return transformOrganization(project.organization);
|
return project.organization;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getOrganizationByProjectIdCached = cacheable(
|
export const getOrganizationByProjectIdCached = cacheable(
|
||||||
@@ -258,3 +254,32 @@ export async function getOrganizationSubscriptionChartEndDate(
|
|||||||
|
|
||||||
return endDate;
|
return endDate;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const DEFAULT_TIMEZONE = 'UTC';
|
||||||
|
|
||||||
|
export async function getSettingsForOrganization(organizationId: string) {
|
||||||
|
const organization = await db.organization.findUniqueOrThrow({
|
||||||
|
where: {
|
||||||
|
id: organizationId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
timezone: organization.timezone || DEFAULT_TIMEZONE,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getSettingsForProject(projectId: string) {
|
||||||
|
const project = await db.project.findUniqueOrThrow({
|
||||||
|
where: {
|
||||||
|
id: projectId,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
organization: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
timezone: project.organization.timezone || DEFAULT_TIMEZONE,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|||||||
@@ -15,7 +15,9 @@ export const zGetMetricsInput = z.object({
|
|||||||
interval: zTimeInterval,
|
interval: zTimeInterval,
|
||||||
});
|
});
|
||||||
|
|
||||||
export type IGetMetricsInput = z.infer<typeof zGetMetricsInput>;
|
export type IGetMetricsInput = z.infer<typeof zGetMetricsInput> & {
|
||||||
|
timezone: string;
|
||||||
|
};
|
||||||
|
|
||||||
export const zGetTopPagesInput = z.object({
|
export const zGetTopPagesInput = z.object({
|
||||||
projectId: z.string(),
|
projectId: z.string(),
|
||||||
@@ -27,7 +29,9 @@ export const zGetTopPagesInput = z.object({
|
|||||||
limit: z.number().optional(),
|
limit: z.number().optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
export type IGetTopPagesInput = z.infer<typeof zGetTopPagesInput>;
|
export type IGetTopPagesInput = z.infer<typeof zGetTopPagesInput> & {
|
||||||
|
timezone: string;
|
||||||
|
};
|
||||||
|
|
||||||
export const zGetTopEntryExitInput = z.object({
|
export const zGetTopEntryExitInput = z.object({
|
||||||
projectId: z.string(),
|
projectId: z.string(),
|
||||||
@@ -40,7 +44,9 @@ export const zGetTopEntryExitInput = z.object({
|
|||||||
limit: z.number().optional(),
|
limit: z.number().optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
export type IGetTopEntryExitInput = z.infer<typeof zGetTopEntryExitInput>;
|
export type IGetTopEntryExitInput = z.infer<typeof zGetTopEntryExitInput> & {
|
||||||
|
timezone: string;
|
||||||
|
};
|
||||||
|
|
||||||
export const zGetTopGenericInput = z.object({
|
export const zGetTopGenericInput = z.object({
|
||||||
projectId: z.string(),
|
projectId: z.string(),
|
||||||
@@ -75,7 +81,9 @@ export const zGetTopGenericInput = z.object({
|
|||||||
limit: z.number().optional(),
|
limit: z.number().optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
export type IGetTopGenericInput = z.infer<typeof zGetTopGenericInput>;
|
export type IGetTopGenericInput = z.infer<typeof zGetTopGenericInput> & {
|
||||||
|
timezone: string;
|
||||||
|
};
|
||||||
|
|
||||||
export class OverviewService {
|
export class OverviewService {
|
||||||
private pendingQueries: Map<string, Promise<number | null>> = new Map();
|
private pendingQueries: Map<string, Promise<number | null>> = new Map();
|
||||||
@@ -91,11 +99,13 @@ export class OverviewService {
|
|||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
filters,
|
filters,
|
||||||
|
timezone,
|
||||||
}: {
|
}: {
|
||||||
projectId: string;
|
projectId: string;
|
||||||
startDate: string;
|
startDate: string;
|
||||||
endDate: string;
|
endDate: string;
|
||||||
filters: IChartEventFilter[];
|
filters: IChartEventFilter[];
|
||||||
|
timezone: string;
|
||||||
}) {
|
}) {
|
||||||
const where = this.getRawWhereClause('sessions', filters);
|
const where = this.getRawWhereClause('sessions', filters);
|
||||||
const key = `total_sessions_${projectId}_${startDate}_${endDate}_${JSON.stringify(filters)}`;
|
const key = `total_sessions_${projectId}_${startDate}_${endDate}_${JSON.stringify(filters)}`;
|
||||||
@@ -109,15 +119,15 @@ export class OverviewService {
|
|||||||
// Create new query promise and store it
|
// Create new query promise and store it
|
||||||
const queryPromise = getCache(key, 15, async () => {
|
const queryPromise = getCache(key, 15, async () => {
|
||||||
try {
|
try {
|
||||||
const result = await clix(this.client)
|
const result = await clix(this.client, timezone)
|
||||||
.select<{
|
.select<{
|
||||||
total_sessions: number;
|
total_sessions: number;
|
||||||
}>(['sum(sign) as total_sessions'])
|
}>(['sum(sign) as total_sessions'])
|
||||||
.from(TABLE_NAMES.sessions, true)
|
.from(TABLE_NAMES.sessions, true)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.rawWhere(where)
|
.rawWhere(where)
|
||||||
.having('sum(sign)', '>', 0)
|
.having('sum(sign)', '>', 0)
|
||||||
@@ -138,6 +148,7 @@ export class OverviewService {
|
|||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
interval,
|
interval,
|
||||||
|
timezone,
|
||||||
}: IGetMetricsInput): Promise<{
|
}: IGetMetricsInput): Promise<{
|
||||||
metrics: {
|
metrics: {
|
||||||
bounce_rate: number;
|
bounce_rate: number;
|
||||||
@@ -160,17 +171,17 @@ export class OverviewService {
|
|||||||
const where = this.getRawWhereClause('sessions', filters);
|
const where = this.getRawWhereClause('sessions', filters);
|
||||||
if (this.isPageFilter(filters)) {
|
if (this.isPageFilter(filters)) {
|
||||||
// Session aggregation with bounce rates
|
// Session aggregation with bounce rates
|
||||||
const sessionAggQuery = clix(this.client)
|
const sessionAggQuery = clix(this.client, timezone)
|
||||||
.select([
|
.select([
|
||||||
`${clix.toStartOfInterval('created_at', interval, startDate)} AS date`,
|
`${clix.toStartOf('created_at', interval, timezone)} AS date`,
|
||||||
'round((countIf(is_bounce = 1 AND sign = 1) * 100.) / countIf(sign = 1), 2) AS bounce_rate',
|
'round((countIf(is_bounce = 1 AND sign = 1) * 100.) / countIf(sign = 1), 2) AS bounce_rate',
|
||||||
])
|
])
|
||||||
.from(TABLE_NAMES.sessions, true)
|
.from(TABLE_NAMES.sessions, true)
|
||||||
.where('sign', '=', 1)
|
.where('sign', '=', 1)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.rawWhere(where)
|
.rawWhere(where)
|
||||||
.groupBy(['date'])
|
.groupBy(['date'])
|
||||||
@@ -178,7 +189,7 @@ export class OverviewService {
|
|||||||
.orderBy('date', 'ASC');
|
.orderBy('date', 'ASC');
|
||||||
|
|
||||||
// Overall unique visitors
|
// Overall unique visitors
|
||||||
const overallUniqueVisitorsQuery = clix(this.client)
|
const overallUniqueVisitorsQuery = clix(this.client, timezone)
|
||||||
.select([
|
.select([
|
||||||
'uniq(profile_id) AS unique_visitors',
|
'uniq(profile_id) AS unique_visitors',
|
||||||
'uniq(session_id) AS total_sessions',
|
'uniq(session_id) AS total_sessions',
|
||||||
@@ -187,23 +198,23 @@ export class OverviewService {
|
|||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('name', '=', 'screen_view')
|
.where('name', '=', 'screen_view')
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.rawWhere(this.getRawWhereClause('events', filters));
|
.rawWhere(this.getRawWhereClause('events', filters));
|
||||||
|
|
||||||
return clix(this.client)
|
return clix(this.client, timezone)
|
||||||
.with('session_agg', sessionAggQuery)
|
.with('session_agg', sessionAggQuery)
|
||||||
.with(
|
.with(
|
||||||
'overall_bounce_rate',
|
'overall_bounce_rate',
|
||||||
clix(this.client)
|
clix(this.client, timezone)
|
||||||
.select(['bounce_rate'])
|
.select(['bounce_rate'])
|
||||||
.from('session_agg')
|
.from('session_agg')
|
||||||
.where('date', '=', clix.exp("'1970-01-01 00:00:00'")),
|
.where('date', '=', clix.exp("'1970-01-01 00:00:00'")),
|
||||||
)
|
)
|
||||||
.with(
|
.with(
|
||||||
'daily_stats',
|
'daily_stats',
|
||||||
clix(this.client)
|
clix(this.client, timezone)
|
||||||
.select(['date', 'bounce_rate'])
|
.select(['date', 'bounce_rate'])
|
||||||
.from('session_agg')
|
.from('session_agg')
|
||||||
.where('date', '!=', clix.exp("'1970-01-01 00:00:00'")),
|
.where('date', '!=', clix.exp("'1970-01-01 00:00:00'")),
|
||||||
@@ -221,7 +232,7 @@ export class OverviewService {
|
|||||||
overall_total_sessions: number;
|
overall_total_sessions: number;
|
||||||
overall_bounce_rate: number;
|
overall_bounce_rate: number;
|
||||||
}>([
|
}>([
|
||||||
`${clix.toStartOfInterval('e.created_at', interval, startDate)} AS date`,
|
`${clix.toInterval('e.created_at', interval)} AS date`,
|
||||||
'ds.bounce_rate as bounce_rate',
|
'ds.bounce_rate as bounce_rate',
|
||||||
'uniq(e.profile_id) AS unique_visitors',
|
'uniq(e.profile_id) AS unique_visitors',
|
||||||
'uniq(e.session_id) AS total_sessions',
|
'uniq(e.session_id) AS total_sessions',
|
||||||
@@ -236,20 +247,29 @@ export class OverviewService {
|
|||||||
.from(`${TABLE_NAMES.events} AS e`)
|
.from(`${TABLE_NAMES.events} AS e`)
|
||||||
.leftJoin(
|
.leftJoin(
|
||||||
'daily_stats AS ds',
|
'daily_stats AS ds',
|
||||||
`${clix.toStartOfInterval('e.created_at', interval, startDate)} = ds.date`,
|
`${clix.toInterval('e.created_at', interval)} = ds.date`,
|
||||||
)
|
)
|
||||||
.where('e.project_id', '=', projectId)
|
.where('e.project_id', '=', projectId)
|
||||||
.where('e.name', '=', 'screen_view')
|
.where('e.name', '=', 'screen_view')
|
||||||
.where('e.created_at', 'BETWEEN', [
|
.where('e.created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.rawWhere(this.getRawWhereClause('events', filters))
|
.rawWhere(this.getRawWhereClause('events', filters))
|
||||||
.groupBy(['date', 'ds.bounce_rate'])
|
.groupBy(['date', 'ds.bounce_rate'])
|
||||||
.orderBy('date', 'ASC')
|
.orderBy('date', 'ASC')
|
||||||
.fill(
|
.fill(
|
||||||
clix.toStartOfInterval(clix.datetime(startDate), interval, startDate),
|
clix.toStartOf(
|
||||||
clix.toStartOfInterval(clix.datetime(endDate), interval, startDate),
|
clix.datetime(
|
||||||
|
startDate,
|
||||||
|
['month', 'week'].includes(interval) ? 'toDate' : 'toDateTime',
|
||||||
|
),
|
||||||
|
interval,
|
||||||
|
),
|
||||||
|
clix.datetime(
|
||||||
|
endDate,
|
||||||
|
['month', 'week'].includes(interval) ? 'toDate' : 'toDateTime',
|
||||||
|
),
|
||||||
clix.toInterval('1', interval),
|
clix.toInterval('1', interval),
|
||||||
)
|
)
|
||||||
.transform({
|
.transform({
|
||||||
@@ -289,7 +309,7 @@ export class OverviewService {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const query = clix(this.client)
|
const query = clix(this.client, timezone)
|
||||||
.select<{
|
.select<{
|
||||||
date: string;
|
date: string;
|
||||||
bounce_rate: number;
|
bounce_rate: number;
|
||||||
@@ -299,7 +319,7 @@ export class OverviewService {
|
|||||||
total_screen_views: number;
|
total_screen_views: number;
|
||||||
views_per_session: number;
|
views_per_session: number;
|
||||||
}>([
|
}>([
|
||||||
`${clix.toStartOfInterval('created_at', interval, startDate)} AS date`,
|
`${clix.toStartOf('created_at', interval, timezone)} AS date`,
|
||||||
'round(sum(sign * is_bounce) * 100.0 / sum(sign), 2) as bounce_rate',
|
'round(sum(sign * is_bounce) * 100.0 / sum(sign), 2) as bounce_rate',
|
||||||
'uniqIf(profile_id, sign > 0) AS unique_visitors',
|
'uniqIf(profile_id, sign > 0) AS unique_visitors',
|
||||||
'sum(sign) AS total_sessions',
|
'sum(sign) AS total_sessions',
|
||||||
@@ -310,8 +330,8 @@ export class OverviewService {
|
|||||||
])
|
])
|
||||||
.from('sessions')
|
.from('sessions')
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.rawWhere(where)
|
.rawWhere(where)
|
||||||
@@ -320,8 +340,17 @@ export class OverviewService {
|
|||||||
.rollup()
|
.rollup()
|
||||||
.orderBy('date', 'ASC')
|
.orderBy('date', 'ASC')
|
||||||
.fill(
|
.fill(
|
||||||
clix.toStartOfInterval(clix.datetime(startDate), interval, startDate),
|
clix.toStartOf(
|
||||||
clix.toStartOfInterval(clix.datetime(endDate), interval, startDate),
|
clix.datetime(
|
||||||
|
startDate,
|
||||||
|
['month', 'week'].includes(interval) ? 'toDate' : 'toDateTime',
|
||||||
|
),
|
||||||
|
interval,
|
||||||
|
),
|
||||||
|
clix.datetime(
|
||||||
|
endDate,
|
||||||
|
['month', 'week'].includes(interval) ? 'toDate' : 'toDateTime',
|
||||||
|
),
|
||||||
clix.toInterval('1', interval),
|
clix.toInterval('1', interval),
|
||||||
)
|
)
|
||||||
.transform({
|
.transform({
|
||||||
@@ -384,8 +413,9 @@ export class OverviewService {
|
|||||||
endDate,
|
endDate,
|
||||||
cursor = 1,
|
cursor = 1,
|
||||||
limit = 10,
|
limit = 10,
|
||||||
|
timezone,
|
||||||
}: IGetTopPagesInput) {
|
}: IGetTopPagesInput) {
|
||||||
const pageStatsQuery = clix(this.client)
|
const pageStatsQuery = clix(this.client, timezone)
|
||||||
.select([
|
.select([
|
||||||
'origin',
|
'origin',
|
||||||
'path',
|
'path',
|
||||||
@@ -398,15 +428,15 @@ export class OverviewService {
|
|||||||
.where('name', '=', 'screen_view')
|
.where('name', '=', 'screen_view')
|
||||||
.where('path', '!=', '')
|
.where('path', '!=', '')
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.groupBy(['origin', 'path'])
|
.groupBy(['origin', 'path'])
|
||||||
.orderBy('count', 'DESC')
|
.orderBy('count', 'DESC')
|
||||||
.limit(limit)
|
.limit(limit)
|
||||||
.offset((cursor - 1) * limit);
|
.offset((cursor - 1) * limit);
|
||||||
|
|
||||||
const bounceStatsQuery = clix(this.client)
|
const bounceStatsQuery = clix(this.client, timezone)
|
||||||
.select([
|
.select([
|
||||||
'entry_path',
|
'entry_path',
|
||||||
'entry_origin',
|
'entry_origin',
|
||||||
@@ -416,15 +446,15 @@ export class OverviewService {
|
|||||||
.where('sign', '=', 1)
|
.where('sign', '=', 1)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.groupBy(['entry_path', 'entry_origin']);
|
.groupBy(['entry_path', 'entry_origin']);
|
||||||
|
|
||||||
pageStatsQuery.rawWhere(this.getRawWhereClause('events', filters));
|
pageStatsQuery.rawWhere(this.getRawWhereClause('events', filters));
|
||||||
bounceStatsQuery.rawWhere(this.getRawWhereClause('sessions', filters));
|
bounceStatsQuery.rawWhere(this.getRawWhereClause('sessions', filters));
|
||||||
|
|
||||||
const mainQuery = clix(this.client)
|
const mainQuery = clix(this.client, timezone)
|
||||||
.with('page_stats', pageStatsQuery)
|
.with('page_stats', pageStatsQuery)
|
||||||
.with('bounce_stats', bounceStatsQuery)
|
.with('bounce_stats', bounceStatsQuery)
|
||||||
.select<{
|
.select<{
|
||||||
@@ -455,6 +485,7 @@ export class OverviewService {
|
|||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
filters,
|
filters,
|
||||||
|
timezone,
|
||||||
});
|
});
|
||||||
|
|
||||||
return mainQuery.execute();
|
return mainQuery.execute();
|
||||||
@@ -468,6 +499,7 @@ export class OverviewService {
|
|||||||
mode,
|
mode,
|
||||||
cursor = 1,
|
cursor = 1,
|
||||||
limit = 10,
|
limit = 10,
|
||||||
|
timezone,
|
||||||
}: IGetTopEntryExitInput) {
|
}: IGetTopEntryExitInput) {
|
||||||
const where = this.getRawWhereClause('sessions', filters);
|
const where = this.getRawWhereClause('sessions', filters);
|
||||||
|
|
||||||
@@ -476,11 +508,12 @@ export class OverviewService {
|
|||||||
filters,
|
filters,
|
||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
|
timezone,
|
||||||
});
|
});
|
||||||
|
|
||||||
const offset = (cursor - 1) * limit;
|
const offset = (cursor - 1) * limit;
|
||||||
|
|
||||||
const query = clix(this.client)
|
const query = clix(this.client, timezone)
|
||||||
.select<{
|
.select<{
|
||||||
origin: string;
|
origin: string;
|
||||||
path: string;
|
path: string;
|
||||||
@@ -497,8 +530,8 @@ export class OverviewService {
|
|||||||
.from(TABLE_NAMES.sessions, true)
|
.from(TABLE_NAMES.sessions, true)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.rawWhere(where)
|
.rawWhere(where)
|
||||||
.groupBy([`${mode}_origin`, `${mode}_path`])
|
.groupBy([`${mode}_origin`, `${mode}_path`])
|
||||||
@@ -510,7 +543,7 @@ export class OverviewService {
|
|||||||
let mainQuery = query;
|
let mainQuery = query;
|
||||||
|
|
||||||
if (this.isPageFilter(filters)) {
|
if (this.isPageFilter(filters)) {
|
||||||
mainQuery = clix(this.client)
|
mainQuery = clix(this.client, timezone)
|
||||||
.with('distinct_sessions', distinctSessionQuery)
|
.with('distinct_sessions', distinctSessionQuery)
|
||||||
.merge(query)
|
.merge(query)
|
||||||
.where(
|
.where(
|
||||||
@@ -525,6 +558,7 @@ export class OverviewService {
|
|||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
filters,
|
filters,
|
||||||
|
timezone,
|
||||||
});
|
});
|
||||||
|
|
||||||
return mainQuery.execute();
|
return mainQuery.execute();
|
||||||
@@ -535,19 +569,21 @@ export class OverviewService {
|
|||||||
filters,
|
filters,
|
||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
|
timezone,
|
||||||
}: {
|
}: {
|
||||||
projectId: string;
|
projectId: string;
|
||||||
filters: IChartEventFilter[];
|
filters: IChartEventFilter[];
|
||||||
startDate: string;
|
startDate: string;
|
||||||
endDate: string;
|
endDate: string;
|
||||||
|
timezone: string;
|
||||||
}) {
|
}) {
|
||||||
return clix(this.client)
|
return clix(this.client, timezone)
|
||||||
.select(['DISTINCT session_id'])
|
.select(['DISTINCT session_id'])
|
||||||
.from(TABLE_NAMES.events)
|
.from(TABLE_NAMES.events)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.rawWhere(this.getRawWhereClause('events', filters));
|
.rawWhere(this.getRawWhereClause('events', filters));
|
||||||
}
|
}
|
||||||
@@ -560,12 +596,14 @@ export class OverviewService {
|
|||||||
column,
|
column,
|
||||||
cursor = 1,
|
cursor = 1,
|
||||||
limit = 10,
|
limit = 10,
|
||||||
|
timezone,
|
||||||
}: IGetTopGenericInput) {
|
}: IGetTopGenericInput) {
|
||||||
const distinctSessionQuery = this.getDistinctSessions({
|
const distinctSessionQuery = this.getDistinctSessions({
|
||||||
projectId,
|
projectId,
|
||||||
filters,
|
filters,
|
||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
|
timezone,
|
||||||
});
|
});
|
||||||
|
|
||||||
const prefixColumn = (() => {
|
const prefixColumn = (() => {
|
||||||
@@ -584,7 +622,7 @@ export class OverviewService {
|
|||||||
|
|
||||||
const offset = (cursor - 1) * limit;
|
const offset = (cursor - 1) * limit;
|
||||||
|
|
||||||
const query = clix(this.client)
|
const query = clix(this.client, timezone)
|
||||||
.select<{
|
.select<{
|
||||||
prefix?: string;
|
prefix?: string;
|
||||||
name: string;
|
name: string;
|
||||||
@@ -601,8 +639,8 @@ export class OverviewService {
|
|||||||
.from(TABLE_NAMES.sessions, true)
|
.from(TABLE_NAMES.sessions, true)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
.where('created_at', 'BETWEEN', [
|
.where('created_at', 'BETWEEN', [
|
||||||
clix.datetime(startDate),
|
clix.datetime(startDate, 'toDateTime'),
|
||||||
clix.datetime(endDate),
|
clix.datetime(endDate, 'toDateTime'),
|
||||||
])
|
])
|
||||||
.groupBy([prefixColumn, column].filter(Boolean))
|
.groupBy([prefixColumn, column].filter(Boolean))
|
||||||
.having('sum(sign)', '>', 0)
|
.having('sum(sign)', '>', 0)
|
||||||
@@ -613,7 +651,7 @@ export class OverviewService {
|
|||||||
let mainQuery = query;
|
let mainQuery = query;
|
||||||
|
|
||||||
if (this.isPageFilter(filters)) {
|
if (this.isPageFilter(filters)) {
|
||||||
mainQuery = clix(this.client)
|
mainQuery = clix(this.client, timezone)
|
||||||
.with('distinct_sessions', distinctSessionQuery)
|
.with('distinct_sessions', distinctSessionQuery)
|
||||||
.merge(query)
|
.merge(query)
|
||||||
.where(
|
.where(
|
||||||
@@ -632,6 +670,7 @@ export class OverviewService {
|
|||||||
startDate,
|
startDate,
|
||||||
endDate,
|
endDate,
|
||||||
filters,
|
filters,
|
||||||
|
timezone,
|
||||||
}),
|
}),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ export interface SqlBuilderObject {
|
|||||||
joins: Record<string, string>;
|
joins: Record<string, string>;
|
||||||
limit: number | undefined;
|
limit: number | undefined;
|
||||||
offset: number | undefined;
|
offset: number | undefined;
|
||||||
|
fill: string | undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function createSqlBuilder() {
|
export function createSqlBuilder() {
|
||||||
@@ -26,6 +27,7 @@ export function createSqlBuilder() {
|
|||||||
joins: {},
|
joins: {},
|
||||||
limit: undefined,
|
limit: undefined,
|
||||||
offset: undefined,
|
offset: undefined,
|
||||||
|
fill: undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
const getWhere = () =>
|
const getWhere = () =>
|
||||||
@@ -43,6 +45,7 @@ export function createSqlBuilder() {
|
|||||||
const getOffset = () => (sb.offset ? `OFFSET ${sb.offset}` : '');
|
const getOffset = () => (sb.offset ? `OFFSET ${sb.offset}` : '');
|
||||||
const getJoins = () =>
|
const getJoins = () =>
|
||||||
Object.keys(sb.joins).length ? join(sb.joins, ' ') : '';
|
Object.keys(sb.joins).length ? join(sb.joins, ' ') : '';
|
||||||
|
const getFill = () => (sb.fill ? `WITH FILL ${sb.fill}` : '');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
sb,
|
sb,
|
||||||
@@ -54,6 +57,7 @@ export function createSqlBuilder() {
|
|||||||
getOrderBy,
|
getOrderBy,
|
||||||
getHaving,
|
getHaving,
|
||||||
getJoins,
|
getJoins,
|
||||||
|
getFill,
|
||||||
getSql: () => {
|
getSql: () => {
|
||||||
const sql = [
|
const sql = [
|
||||||
getSelect(),
|
getSelect(),
|
||||||
@@ -65,6 +69,7 @@ export function createSqlBuilder() {
|
|||||||
getOrderBy(),
|
getOrderBy(),
|
||||||
getLimit(),
|
getLimit(),
|
||||||
getOffset(),
|
getOffset(),
|
||||||
|
getFill(),
|
||||||
]
|
]
|
||||||
.filter(Boolean)
|
.filter(Boolean)
|
||||||
.join(' ');
|
.join(' ');
|
||||||
|
|||||||
@@ -1,45 +1,29 @@
|
|||||||
import {
|
|
||||||
differenceInMilliseconds,
|
|
||||||
endOfMonth,
|
|
||||||
endOfYear,
|
|
||||||
formatISO,
|
|
||||||
startOfDay,
|
|
||||||
startOfMonth,
|
|
||||||
startOfYear,
|
|
||||||
subDays,
|
|
||||||
subMilliseconds,
|
|
||||||
subMinutes,
|
|
||||||
subMonths,
|
|
||||||
subYears,
|
|
||||||
} from 'date-fns';
|
|
||||||
import * as mathjs from 'mathjs';
|
import * as mathjs from 'mathjs';
|
||||||
import { last, pluck, repeat, reverse, uniq } from 'ramda';
|
import { last, pluck, reverse, uniq } from 'ramda';
|
||||||
import { escape } from 'sqlstring';
|
import { escape } from 'sqlstring';
|
||||||
|
|
||||||
|
import type { ISerieDataItem } from '@openpanel/common';
|
||||||
import {
|
import {
|
||||||
|
DateTime,
|
||||||
average,
|
average,
|
||||||
completeSerie,
|
|
||||||
getPreviousMetric,
|
getPreviousMetric,
|
||||||
|
groupByLabels,
|
||||||
max,
|
max,
|
||||||
min,
|
min,
|
||||||
round,
|
round,
|
||||||
slug,
|
slug,
|
||||||
sum,
|
sum,
|
||||||
} from '@openpanel/common';
|
} from '@openpanel/common';
|
||||||
import type { ISerieDataItem } from '@openpanel/common';
|
|
||||||
import { alphabetIds } from '@openpanel/constants';
|
import { alphabetIds } from '@openpanel/constants';
|
||||||
import {
|
import {
|
||||||
TABLE_NAMES,
|
TABLE_NAMES,
|
||||||
chQuery,
|
chQuery,
|
||||||
createSqlBuilder,
|
createSqlBuilder,
|
||||||
db,
|
|
||||||
formatClickhouseDate,
|
formatClickhouseDate,
|
||||||
getChartSql,
|
getChartSql,
|
||||||
getEventFiltersWhereClause,
|
getEventFiltersWhereClause,
|
||||||
getOrganizationByProjectId,
|
|
||||||
getOrganizationByProjectIdCached,
|
|
||||||
getOrganizationSubscriptionChartEndDate,
|
getOrganizationSubscriptionChartEndDate,
|
||||||
getProfiles,
|
getSettingsForProject,
|
||||||
} from '@openpanel/db';
|
} from '@openpanel/db';
|
||||||
import type {
|
import type {
|
||||||
FinalChart,
|
FinalChart,
|
||||||
@@ -48,9 +32,7 @@ import type {
|
|||||||
IChartInputWithDates,
|
IChartInputWithDates,
|
||||||
IChartRange,
|
IChartRange,
|
||||||
IGetChartDataInput,
|
IGetChartDataInput,
|
||||||
IInterval,
|
|
||||||
} from '@openpanel/validation';
|
} from '@openpanel/validation';
|
||||||
import { TRPCNotFoundError } from '../errors';
|
|
||||||
|
|
||||||
function getEventLegend(event: IChartEvent) {
|
function getEventLegend(event: IChartEvent) {
|
||||||
return event.displayName || event.name;
|
return event.displayName || event.name;
|
||||||
@@ -134,115 +116,190 @@ export function withFormula(
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
const toDynamicISODateWithTZ = (
|
export function getDatesFromRange(range: IChartRange, timezone: string) {
|
||||||
date: string,
|
|
||||||
blueprint: string,
|
|
||||||
interval: IInterval,
|
|
||||||
) => {
|
|
||||||
// If we have a space in the date we know it's a date with time
|
|
||||||
if (date.includes(' ')) {
|
|
||||||
// If interval is minutes we need to convert the timezone to what timezone is used (either on client or the server)
|
|
||||||
// - We use timezone from server if its a predefined range (yearToDate, lastYear, etc.)
|
|
||||||
// - We use timezone from client if its a custom range
|
|
||||||
if (interval === 'minute' || interval === 'hour') {
|
|
||||||
return (
|
|
||||||
date.replace(' ', 'T') +
|
|
||||||
// Only append timezone if it's not UTC (Z)
|
|
||||||
(blueprint.match(/[+-]\d{2}:\d{2}/) ? blueprint.slice(-6) : 'Z')
|
|
||||||
);
|
|
||||||
}
|
|
||||||
// Otherwise we just return without the timezone
|
|
||||||
// It will be converted to the correct timezone on the client
|
|
||||||
return date.replace(' ', 'T');
|
|
||||||
}
|
|
||||||
return `${date}T00:00:00Z`;
|
|
||||||
};
|
|
||||||
|
|
||||||
export function getDatesFromRange(range: IChartRange) {
|
|
||||||
if (range === '30min' || range === 'lastHour') {
|
if (range === '30min' || range === 'lastHour') {
|
||||||
const minutes = range === '30min' ? 30 : 60;
|
const minutes = range === '30min' ? 30 : 60;
|
||||||
const startDate = formatISO(subMinutes(new Date(), minutes));
|
const startDate = DateTime.now()
|
||||||
const endDate = formatISO(new Date());
|
.minus({ minute: minutes })
|
||||||
|
.startOf('minute')
|
||||||
|
.setZone(timezone)
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('minute')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (range === 'today') {
|
if (range === 'today') {
|
||||||
// This is last 24 hours instead
|
const startDate = DateTime.now()
|
||||||
// Makes it easier to handle timezones
|
.setZone(timezone)
|
||||||
// const startDate = startOfDay(new Date());
|
.startOf('day')
|
||||||
// const endDate = endOfDay(new Date());
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
const startDate = subDays(new Date(), 1);
|
const endDate = DateTime.now()
|
||||||
const endDate = new Date();
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate: formatISO(startDate),
|
startDate: startDate,
|
||||||
endDate: formatISO(endDate),
|
endDate: endDate,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (range === 'yesterday') {
|
||||||
|
const startDate = DateTime.now()
|
||||||
|
.minus({ day: 1 })
|
||||||
|
.setZone(timezone)
|
||||||
|
.startOf('day')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.minus({ day: 1 })
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
return {
|
||||||
|
startDate: startDate,
|
||||||
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (range === '7d') {
|
if (range === '7d') {
|
||||||
const startDate = formatISO(startOfDay(subDays(new Date(), 7)));
|
const startDate = DateTime.now()
|
||||||
const endDate = formatISO(new Date());
|
.minus({ day: 7 })
|
||||||
|
.setZone(timezone)
|
||||||
|
.startOf('day')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (range === '6m') {
|
||||||
|
const startDate = DateTime.now()
|
||||||
|
.minus({ month: 6 })
|
||||||
|
.setZone(timezone)
|
||||||
|
.startOf('day')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
|
return {
|
||||||
|
startDate: startDate,
|
||||||
|
endDate: endDate,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (range === '12m') {
|
||||||
|
const startDate = DateTime.now()
|
||||||
|
.minus({ month: 12 })
|
||||||
|
.setZone(timezone)
|
||||||
|
.startOf('month')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('month')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
|
return {
|
||||||
|
startDate: startDate,
|
||||||
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (range === 'monthToDate') {
|
if (range === 'monthToDate') {
|
||||||
const startDate = formatISO(startOfMonth(new Date()));
|
const startDate = DateTime.now()
|
||||||
const endDate = formatISO(new Date());
|
.setZone(timezone)
|
||||||
|
.startOf('month')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (range === 'lastMonth') {
|
if (range === 'lastMonth') {
|
||||||
const month = subMonths(new Date(), 1);
|
const month = DateTime.now()
|
||||||
const startDate = formatISO(startOfMonth(month));
|
.minus({ month: 1 })
|
||||||
const endDate = formatISO(endOfMonth(month));
|
.setZone(timezone)
|
||||||
|
.startOf('month');
|
||||||
|
|
||||||
|
const startDate = month.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = month
|
||||||
|
.endOf('month')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (range === 'yearToDate') {
|
if (range === 'yearToDate') {
|
||||||
const startDate = formatISO(startOfYear(new Date()));
|
const startDate = DateTime.now()
|
||||||
const endDate = formatISO(new Date());
|
.setZone(timezone)
|
||||||
|
.startOf('year')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (range === 'lastYear') {
|
if (range === 'lastYear') {
|
||||||
const year = subYears(new Date(), 1);
|
const year = DateTime.now().minus({ year: 1 }).setZone(timezone);
|
||||||
const startDate = formatISO(startOfYear(year));
|
const startDate = year.startOf('year').toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
const endDate = formatISO(endOfYear(year));
|
const endDate = year.endOf('year').toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// range === '30d'
|
// range === '30d'
|
||||||
const startDate = formatISO(startOfDay(subDays(new Date(), 30)));
|
const startDate = DateTime.now()
|
||||||
const endDate = formatISO(new Date());
|
.minus({ day: 30 })
|
||||||
|
.setZone(timezone)
|
||||||
|
.startOf('day')
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
const endDate = DateTime.now()
|
||||||
|
.setZone(timezone)
|
||||||
|
.endOf('day')
|
||||||
|
.plus({ millisecond: 1 })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate,
|
startDate: startDate,
|
||||||
endDate,
|
endDate: endDate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -268,12 +325,15 @@ function fillFunnel(funnel: { level: number; count: number }[], steps: number) {
|
|||||||
return filled.reverse();
|
return filled.reverse();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getChartStartEndDate({
|
export function getChartStartEndDate(
|
||||||
startDate,
|
{
|
||||||
endDate,
|
startDate,
|
||||||
range,
|
endDate,
|
||||||
}: Pick<IChartInput, 'endDate' | 'startDate' | 'range'>) {
|
range,
|
||||||
const ranges = getDatesFromRange(range);
|
}: Pick<IChartInput, 'endDate' | 'startDate' | 'range'>,
|
||||||
|
timezone: string,
|
||||||
|
) {
|
||||||
|
const ranges = getDatesFromRange(range, timezone);
|
||||||
|
|
||||||
if (startDate && endDate) {
|
if (startDate && endDate) {
|
||||||
return { startDate: startDate, endDate: endDate };
|
return { startDate: startDate, endDate: endDate };
|
||||||
@@ -293,10 +353,25 @@ export function getChartPrevStartEndDate({
|
|||||||
startDate: string;
|
startDate: string;
|
||||||
endDate: string;
|
endDate: string;
|
||||||
}) {
|
}) {
|
||||||
const diff = differenceInMilliseconds(new Date(endDate), new Date(startDate));
|
let diff = DateTime.fromFormat(endDate, 'yyyy-MM-dd HH:mm:ss').diff(
|
||||||
|
DateTime.fromFormat(startDate, 'yyyy-MM-dd HH:mm:ss'),
|
||||||
|
);
|
||||||
|
|
||||||
|
// this will make sure our start and end date's are correct
|
||||||
|
// otherwise if a day ends with 23:59:59.999 and starts with 00:00:00.000
|
||||||
|
// the diff will be 23:59:59.999 and that will make the start date wrong
|
||||||
|
// so we add 1 millisecond to the diff
|
||||||
|
if ((diff.milliseconds / 1000) % 2 !== 0) {
|
||||||
|
diff = diff.plus({ millisecond: 1 });
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
startDate: formatISO(subMilliseconds(new Date(startDate), diff + 1000)),
|
startDate: DateTime.fromFormat(startDate, 'yyyy-MM-dd HH:mm:ss')
|
||||||
endDate: formatISO(subMilliseconds(new Date(endDate), diff + 1000)),
|
.minus({ millisecond: diff.milliseconds })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss'),
|
||||||
|
endDate: DateTime.fromFormat(endDate, 'yyyy-MM-dd HH:mm:ss')
|
||||||
|
.minus({ millisecond: diff.milliseconds })
|
||||||
|
.toFormat('yyyy-MM-dd HH:mm:ss'),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -386,118 +461,60 @@ export async function getFunnelData({
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getFunnelStep({
|
export async function getChartSerie(
|
||||||
projectId,
|
payload: IGetChartDataInput,
|
||||||
startDate,
|
timezone: string,
|
||||||
endDate,
|
) {
|
||||||
step,
|
|
||||||
...payload
|
|
||||||
}: IChartInput & {
|
|
||||||
step: number;
|
|
||||||
}) {
|
|
||||||
throw new Error('not implemented');
|
|
||||||
// if (!startDate || !endDate) {
|
|
||||||
// throw new Error('startDate and endDate are required');
|
|
||||||
// }
|
|
||||||
|
|
||||||
// if (payload.events.length === 0) {
|
|
||||||
// throw new Error('no events selected');
|
|
||||||
// }
|
|
||||||
|
|
||||||
// const funnels = payload.events.map((event) => {
|
|
||||||
// const { sb, getWhere } = createSqlBuilder();
|
|
||||||
// sb.where = getEventFiltersWhereClause(event.filters);
|
|
||||||
// sb.where.name = `name = ${escape(event.name)}`;
|
|
||||||
// return getWhere().replace('WHERE ', '');
|
|
||||||
// });
|
|
||||||
|
|
||||||
// const innerSql = `SELECT
|
|
||||||
// session_id,
|
|
||||||
// windowFunnel(${ONE_DAY_IN_SECONDS})(toUnixTimestamp(created_at), ${funnels.join(', ')}) AS level
|
|
||||||
// FROM ${TABLE_NAMES.events}
|
|
||||||
// WHERE
|
|
||||||
// project_id = ${escape(projectId)} AND
|
|
||||||
// created_at >= '${formatClickhouseDate(startDate)}' AND
|
|
||||||
// created_at <= '${formatClickhouseDate(endDate)}' AND
|
|
||||||
// name IN (${payload.events.map((event) => escape(event.name)).join(', ')})
|
|
||||||
// GROUP BY session_id`;
|
|
||||||
|
|
||||||
// const profileIdsQuery = `WITH sessions AS (${innerSql})
|
|
||||||
// SELECT
|
|
||||||
// DISTINCT e.profile_id as id
|
|
||||||
// FROM sessions s
|
|
||||||
// JOIN ${TABLE_NAMES.events} e ON s.session_id = e.session_id
|
|
||||||
// WHERE
|
|
||||||
// s.level = ${step} AND
|
|
||||||
// e.project_id = ${escape(projectId)} AND
|
|
||||||
// e.created_at >= '${formatClickhouseDate(startDate)}' AND
|
|
||||||
// e.created_at <= '${formatClickhouseDate(endDate)}' AND
|
|
||||||
// name IN (${payload.events.map((event) => escape(event.name)).join(', ')})
|
|
||||||
// ORDER BY e.created_at DESC
|
|
||||||
// LIMIT 500
|
|
||||||
// `;
|
|
||||||
|
|
||||||
// const res = await chQuery<{
|
|
||||||
// id: string;
|
|
||||||
// }>(profileIdsQuery);
|
|
||||||
|
|
||||||
// return getProfiles(
|
|
||||||
// res.map((r) => r.id),
|
|
||||||
// projectId,
|
|
||||||
// );
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getChartSerie(payload: IGetChartDataInput) {
|
|
||||||
async function getSeries() {
|
async function getSeries() {
|
||||||
const result = await chQuery<ISerieDataItem>(getChartSql(payload));
|
const result = await chQuery<ISerieDataItem>(
|
||||||
|
getChartSql({ ...payload, timezone }),
|
||||||
|
{
|
||||||
|
session_timezone: timezone,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
if (result.length === 0 && payload.breakdowns.length > 0) {
|
if (result.length === 0 && payload.breakdowns.length > 0) {
|
||||||
return await chQuery<ISerieDataItem>(
|
return await chQuery<ISerieDataItem>(
|
||||||
getChartSql({
|
getChartSql({
|
||||||
...payload,
|
...payload,
|
||||||
breakdowns: [],
|
breakdowns: [],
|
||||||
|
timezone,
|
||||||
}),
|
}),
|
||||||
|
{
|
||||||
|
session_timezone: timezone,
|
||||||
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
return getSeries()
|
return getSeries()
|
||||||
.then((data) =>
|
.then(groupByLabels)
|
||||||
completeSerie(data, payload.startDate, payload.endDate, payload.interval),
|
|
||||||
)
|
|
||||||
.then((series) => {
|
.then((series) => {
|
||||||
return Object.keys(series).map((key) => {
|
return series.map((serie) => {
|
||||||
const firstDataItem = series[key]![0]!;
|
|
||||||
const isBreakdown =
|
|
||||||
payload.breakdowns.length && firstDataItem.labels.length;
|
|
||||||
const serieLabel = isBreakdown
|
|
||||||
? firstDataItem.labels
|
|
||||||
: [getEventLegend(payload.event)];
|
|
||||||
return {
|
return {
|
||||||
name: serieLabel,
|
...serie,
|
||||||
event: payload.event,
|
event: payload.event,
|
||||||
data: series[key]!.map((item) => ({
|
|
||||||
...item,
|
|
||||||
date: toDynamicISODateWithTZ(
|
|
||||||
item.date,
|
|
||||||
payload.startDate,
|
|
||||||
payload.interval,
|
|
||||||
),
|
|
||||||
})),
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export type IGetChartSerie = Awaited<ReturnType<typeof getChartSeries>>[number];
|
export type IGetChartSerie = Awaited<ReturnType<typeof getChartSeries>>[number];
|
||||||
export async function getChartSeries(input: IChartInputWithDates) {
|
export async function getChartSeries(
|
||||||
|
input: IChartInputWithDates,
|
||||||
|
timezone: string,
|
||||||
|
) {
|
||||||
const series = (
|
const series = (
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
input.events.map(async (event) =>
|
input.events.map(async (event) =>
|
||||||
getChartSerie({
|
getChartSerie(
|
||||||
...input,
|
{
|
||||||
event,
|
...input,
|
||||||
}),
|
event,
|
||||||
|
},
|
||||||
|
timezone,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
).flat();
|
).flat();
|
||||||
@@ -510,7 +527,8 @@ export async function getChartSeries(input: IChartInputWithDates) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function getChart(input: IChartInput) {
|
export async function getChart(input: IChartInput) {
|
||||||
const currentPeriod = getChartStartEndDate(input);
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
|
const currentPeriod = getChartStartEndDate(input, timezone);
|
||||||
const previousPeriod = getChartPrevStartEndDate(currentPeriod);
|
const previousPeriod = getChartPrevStartEndDate(currentPeriod);
|
||||||
|
|
||||||
const endDate = await getOrganizationSubscriptionChartEndDate(
|
const endDate = await getOrganizationSubscriptionChartEndDate(
|
||||||
@@ -522,14 +540,17 @@ export async function getChart(input: IChartInput) {
|
|||||||
currentPeriod.endDate = endDate;
|
currentPeriod.endDate = endDate;
|
||||||
}
|
}
|
||||||
|
|
||||||
const promises = [getChartSeries({ ...input, ...currentPeriod })];
|
const promises = [getChartSeries({ ...input, ...currentPeriod }, timezone)];
|
||||||
|
|
||||||
if (input.previous) {
|
if (input.previous) {
|
||||||
promises.push(
|
promises.push(
|
||||||
getChartSeries({
|
getChartSeries(
|
||||||
...input,
|
{
|
||||||
...previousPeriod,
|
...input,
|
||||||
}),
|
...previousPeriod,
|
||||||
|
},
|
||||||
|
timezone,
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import {
|
|||||||
db,
|
db,
|
||||||
funnelService,
|
funnelService,
|
||||||
getSelectPropertyKey,
|
getSelectPropertyKey,
|
||||||
|
getSettingsForProject,
|
||||||
toDate,
|
toDate,
|
||||||
} from '@openpanel/db';
|
} from '@openpanel/db';
|
||||||
import {
|
import {
|
||||||
@@ -80,7 +81,7 @@ export const chartRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.query(async ({ input: { projectId, event } }) => {
|
.query(async ({ input: { projectId, event } }) => {
|
||||||
const profiles = await clix(ch)
|
const profiles = await clix(ch, 'UTC')
|
||||||
.select<Pick<IServiceProfile, 'properties'>>(['properties'])
|
.select<Pick<IServiceProfile, 'properties'>>(['properties'])
|
||||||
.from(TABLE_NAMES.profiles)
|
.from(TABLE_NAMES.profiles)
|
||||||
.where('project_id', '=', projectId)
|
.where('project_id', '=', projectId)
|
||||||
@@ -214,7 +215,8 @@ export const chartRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
funnel: protectedProcedure.input(zChartInput).query(async ({ input }) => {
|
funnel: protectedProcedure.input(zChartInput).query(async ({ input }) => {
|
||||||
const currentPeriod = getChartStartEndDate(input);
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
|
const currentPeriod = getChartStartEndDate(input, timezone);
|
||||||
const previousPeriod = getChartPrevStartEndDate(currentPeriod);
|
const previousPeriod = getChartPrevStartEndDate(currentPeriod);
|
||||||
|
|
||||||
const [current, previous] = await Promise.all([
|
const [current, previous] = await Promise.all([
|
||||||
@@ -231,7 +233,8 @@ export const chartRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
conversion: protectedProcedure.input(zChartInput).query(async ({ input }) => {
|
conversion: protectedProcedure.input(zChartInput).query(async ({ input }) => {
|
||||||
const currentPeriod = getChartStartEndDate(input);
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
|
const currentPeriod = getChartStartEndDate(input, timezone);
|
||||||
const previousPeriod = getChartPrevStartEndDate(currentPeriod);
|
const previousPeriod = getChartPrevStartEndDate(currentPeriod);
|
||||||
|
|
||||||
const [current, previous] = await Promise.all([
|
const [current, previous] = await Promise.all([
|
||||||
@@ -254,7 +257,7 @@ export const chartRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
chart: publicProcedure
|
chart: publicProcedure
|
||||||
.use(cacher)
|
// .use(cacher)
|
||||||
.input(zChartInput)
|
.input(zChartInput)
|
||||||
.query(async ({ input, ctx }) => {
|
.query(async ({ input, ctx }) => {
|
||||||
if (ctx.session.userId) {
|
if (ctx.session.userId) {
|
||||||
@@ -301,8 +304,9 @@ export const chartRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.query(async ({ input }) => {
|
.query(async ({ input }) => {
|
||||||
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
const { projectId, firstEvent, secondEvent } = input;
|
const { projectId, firstEvent, secondEvent } = input;
|
||||||
const dates = getChartStartEndDate(input);
|
const dates = getChartStartEndDate(input, timezone);
|
||||||
const diffInterval = {
|
const diffInterval = {
|
||||||
minute: () => differenceInDays(dates.endDate, dates.startDate),
|
minute: () => differenceInDays(dates.endDate, dates.startDate),
|
||||||
hour: () => differenceInDays(dates.endDate, dates.startDate),
|
hour: () => differenceInDays(dates.endDate, dates.startDate),
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import {
|
|||||||
formatClickhouseDate,
|
formatClickhouseDate,
|
||||||
getEventList,
|
getEventList,
|
||||||
getEvents,
|
getEvents,
|
||||||
|
getSettingsForProject,
|
||||||
overviewService,
|
overviewService,
|
||||||
sessionService,
|
sessionService,
|
||||||
} from '@openpanel/db';
|
} from '@openpanel/db';
|
||||||
@@ -275,7 +276,8 @@ export const eventRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.query(async ({ input }) => {
|
.query(async ({ input }) => {
|
||||||
const { startDate, endDate } = getChartStartEndDate(input);
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
|
const { startDate, endDate } = getChartStartEndDate(input, timezone);
|
||||||
if (input.search) {
|
if (input.search) {
|
||||||
input.filters.push({
|
input.filters.push({
|
||||||
id: 'path',
|
id: 'path',
|
||||||
@@ -292,6 +294,7 @@ export const eventRouter = createTRPCRouter({
|
|||||||
interval: input.interval,
|
interval: input.interval,
|
||||||
cursor: input.cursor || 1,
|
cursor: input.cursor || 1,
|
||||||
limit: input.take,
|
limit: input.take,
|
||||||
|
timezone,
|
||||||
});
|
});
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import crypto from 'node:crypto';
|
|||||||
import type { z } from 'zod';
|
import type { z } from 'zod';
|
||||||
|
|
||||||
import { stripTrailingSlash } from '@openpanel/common';
|
import { stripTrailingSlash } from '@openpanel/common';
|
||||||
import { db, getId, getOrganizationBySlug, getUserById } from '@openpanel/db';
|
import { db, getId, getOrganizationById, getUserById } from '@openpanel/db';
|
||||||
import type { IServiceUser, ProjectType } from '@openpanel/db';
|
import type { IServiceUser, ProjectType } from '@openpanel/db';
|
||||||
import { zOnboardingProject } from '@openpanel/validation';
|
import { zOnboardingProject } from '@openpanel/validation';
|
||||||
|
|
||||||
@@ -16,7 +16,7 @@ async function createOrGetOrganization(
|
|||||||
user: IServiceUser,
|
user: IServiceUser,
|
||||||
) {
|
) {
|
||||||
if (input.organizationId) {
|
if (input.organizationId) {
|
||||||
return await getOrganizationBySlug(input.organizationId);
|
return await getOrganizationById(input.organizationId);
|
||||||
}
|
}
|
||||||
|
|
||||||
const TRIAL_DURATION_IN_DAYS = 30;
|
const TRIAL_DURATION_IN_DAYS = 30;
|
||||||
@@ -29,6 +29,7 @@ async function createOrGetOrganization(
|
|||||||
createdByUserId: user.id,
|
createdByUserId: user.id,
|
||||||
subscriptionEndsAt: addDays(new Date(), TRIAL_DURATION_IN_DAYS),
|
subscriptionEndsAt: addDays(new Date(), TRIAL_DURATION_IN_DAYS),
|
||||||
subscriptionStatus: 'trialing',
|
subscriptionStatus: 'trialing',
|
||||||
|
timezone: input.timezone,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
import { connectUserToOrganization, db } from '@openpanel/db';
|
import { connectUserToOrganization, db } from '@openpanel/db';
|
||||||
import { zInviteUser } from '@openpanel/validation';
|
import { zEditOrganization, zInviteUser } from '@openpanel/validation';
|
||||||
|
|
||||||
import { generateSecureId } from '@openpanel/common/server/id';
|
import { generateSecureId } from '@openpanel/common/server/id';
|
||||||
import { sendEmail } from '@openpanel/email';
|
import { sendEmail } from '@openpanel/email';
|
||||||
@@ -12,12 +12,7 @@ import { createTRPCRouter, protectedProcedure } from '../trpc';
|
|||||||
|
|
||||||
export const organizationRouter = createTRPCRouter({
|
export const organizationRouter = createTRPCRouter({
|
||||||
update: protectedProcedure
|
update: protectedProcedure
|
||||||
.input(
|
.input(zEditOrganization)
|
||||||
z.object({
|
|
||||||
id: z.string(),
|
|
||||||
name: z.string(),
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
const access = await getOrganizationAccess({
|
const access = await getOrganizationAccess({
|
||||||
userId: ctx.session.userId,
|
userId: ctx.session.userId,
|
||||||
@@ -34,6 +29,7 @@ export const organizationRouter = createTRPCRouter({
|
|||||||
},
|
},
|
||||||
data: {
|
data: {
|
||||||
name: input.name,
|
name: input.name,
|
||||||
|
timezone: input.timezone,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
import {
|
import {
|
||||||
getOrganizationByProjectIdCached,
|
|
||||||
getOrganizationSubscriptionChartEndDate,
|
getOrganizationSubscriptionChartEndDate,
|
||||||
|
getSettingsForProject,
|
||||||
overviewService,
|
overviewService,
|
||||||
zGetMetricsInput,
|
zGetMetricsInput,
|
||||||
zGetTopGenericInput,
|
zGetTopGenericInput,
|
||||||
zGetTopPagesInput,
|
zGetTopPagesInput,
|
||||||
} from '@openpanel/db';
|
} from '@openpanel/db';
|
||||||
import { type IChartRange, zRange } from '@openpanel/validation';
|
import { type IChartRange, zRange } from '@openpanel/validation';
|
||||||
import { TRPCError } from '@trpc/server';
|
import { format } from 'date-fns';
|
||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
import { cacheMiddleware, createTRPCRouter, publicProcedure } from '../trpc';
|
import { cacheMiddleware, createTRPCRouter, publicProcedure } from '../trpc';
|
||||||
import {
|
import {
|
||||||
@@ -34,8 +34,8 @@ function getCurrentAndPrevious<
|
|||||||
range: IChartRange;
|
range: IChartRange;
|
||||||
projectId: string;
|
projectId: string;
|
||||||
},
|
},
|
||||||
>(input: T, fetchPrevious = false) {
|
>(input: T, fetchPrevious: boolean, timezone: string) {
|
||||||
const current = getChartStartEndDate(input);
|
const current = getChartStartEndDate(input, timezone);
|
||||||
const previous = getChartPrevStartEndDate(current);
|
const previous = getChartPrevStartEndDate(current);
|
||||||
|
|
||||||
return async <R>(
|
return async <R>(
|
||||||
@@ -88,9 +88,11 @@ export const overviewRouter = createTRPCRouter({
|
|||||||
)
|
)
|
||||||
.use(cacher)
|
.use(cacher)
|
||||||
.query(async ({ ctx, input }) => {
|
.query(async ({ ctx, input }) => {
|
||||||
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
const { current, previous } = await getCurrentAndPrevious(
|
const { current, previous } = await getCurrentAndPrevious(
|
||||||
input,
|
{ ...input, timezone },
|
||||||
true,
|
true,
|
||||||
|
timezone,
|
||||||
)(overviewService.getMetrics.bind(overviewService));
|
)(overviewService.getMetrics.bind(overviewService));
|
||||||
return {
|
return {
|
||||||
metrics: {
|
metrics: {
|
||||||
@@ -107,6 +109,7 @@ export const overviewRouter = createTRPCRouter({
|
|||||||
const prev = previous?.series[index];
|
const prev = previous?.series[index];
|
||||||
return {
|
return {
|
||||||
...item,
|
...item,
|
||||||
|
date: format(item.date, 'yyyy-MM-dd HH:mm:ss'),
|
||||||
prev_bounce_rate: prev?.bounce_rate,
|
prev_bounce_rate: prev?.bounce_rate,
|
||||||
prev_unique_visitors: prev?.unique_visitors,
|
prev_unique_visitors: prev?.unique_visitors,
|
||||||
prev_total_screen_views: prev?.total_screen_views,
|
prev_total_screen_views: prev?.total_screen_views,
|
||||||
@@ -129,12 +132,14 @@ export const overviewRouter = createTRPCRouter({
|
|||||||
)
|
)
|
||||||
.use(cacher)
|
.use(cacher)
|
||||||
.query(async ({ input }) => {
|
.query(async ({ input }) => {
|
||||||
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
const { current } = await getCurrentAndPrevious(
|
const { current } = await getCurrentAndPrevious(
|
||||||
input,
|
{ ...input },
|
||||||
false,
|
false,
|
||||||
|
timezone,
|
||||||
)(async (input) => {
|
)(async (input) => {
|
||||||
if (input.mode === 'page') {
|
if (input.mode === 'page') {
|
||||||
return overviewService.getTopPages(input);
|
return overviewService.getTopPages({ ...input, timezone });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (input.mode === 'bot') {
|
if (input.mode === 'bot') {
|
||||||
@@ -144,6 +149,7 @@ export const overviewRouter = createTRPCRouter({
|
|||||||
return overviewService.getTopEntryExit({
|
return overviewService.getTopEntryExit({
|
||||||
...input,
|
...input,
|
||||||
mode: input.mode,
|
mode: input.mode,
|
||||||
|
timezone,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -160,9 +166,11 @@ export const overviewRouter = createTRPCRouter({
|
|||||||
)
|
)
|
||||||
.use(cacher)
|
.use(cacher)
|
||||||
.query(async ({ input }) => {
|
.query(async ({ input }) => {
|
||||||
|
const { timezone } = await getSettingsForProject(input.projectId);
|
||||||
const { current } = await getCurrentAndPrevious(
|
const { current } = await getCurrentAndPrevious(
|
||||||
input,
|
{ ...input, timezone },
|
||||||
false,
|
false,
|
||||||
|
timezone,
|
||||||
)(overviewService.getTopGeneric.bind(overviewService));
|
)(overviewService.getTopGeneric.bind(overviewService));
|
||||||
|
|
||||||
return current;
|
return current;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
import { db, getReferences } from '@openpanel/db';
|
import { db, getReferences, getSettingsForProject } from '@openpanel/db';
|
||||||
import { zCreateReference, zRange } from '@openpanel/validation';
|
import { zCreateReference, zRange } from '@openpanel/validation';
|
||||||
|
|
||||||
import { getProjectAccess } from '../access';
|
import { getProjectAccess } from '../access';
|
||||||
@@ -56,8 +56,9 @@ export const referenceRouter = createTRPCRouter({
|
|||||||
range: zRange,
|
range: zRange,
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.query(({ input: { projectId, ...input } }) => {
|
.query(async ({ input: { projectId, ...input } }) => {
|
||||||
const { startDate, endDate } = getChartStartEndDate(input);
|
const { timezone } = await getSettingsForProject(projectId);
|
||||||
|
const { startDate, endDate } = getChartStartEndDate(input, timezone);
|
||||||
return getReferences({
|
return getReferences({
|
||||||
where: {
|
where: {
|
||||||
projectId,
|
projectId,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import {
|
import {
|
||||||
db,
|
db,
|
||||||
getOrganizationBillingEventsCountSerieCached,
|
getOrganizationBillingEventsCountSerieCached,
|
||||||
getOrganizationBySlug,
|
getOrganizationById,
|
||||||
} from '@openpanel/db';
|
} from '@openpanel/db';
|
||||||
import {
|
import {
|
||||||
cancelSubscription,
|
cancelSubscription,
|
||||||
@@ -24,7 +24,7 @@ export const subscriptionRouter = createTRPCRouter({
|
|||||||
getCurrent: protectedProcedure
|
getCurrent: protectedProcedure
|
||||||
.input(z.object({ organizationId: z.string() }))
|
.input(z.object({ organizationId: z.string() }))
|
||||||
.query(async ({ input }) => {
|
.query(async ({ input }) => {
|
||||||
const organization = await getOrganizationBySlug(input.organizationId);
|
const organization = await getOrganizationById(input.organizationId);
|
||||||
|
|
||||||
if (!organization.subscriptionProductId) {
|
if (!organization.subscriptionProductId) {
|
||||||
return null;
|
return null;
|
||||||
@@ -150,7 +150,7 @@ export const subscriptionRouter = createTRPCRouter({
|
|||||||
cancelSubscription: protectedProcedure
|
cancelSubscription: protectedProcedure
|
||||||
.input(z.object({ organizationId: z.string() }))
|
.input(z.object({ organizationId: z.string() }))
|
||||||
.mutation(async ({ input }) => {
|
.mutation(async ({ input }) => {
|
||||||
const organization = await getOrganizationBySlug(input.organizationId);
|
const organization = await getOrganizationById(input.organizationId);
|
||||||
if (!organization.subscriptionId) {
|
if (!organization.subscriptionId) {
|
||||||
throw TRPCBadRequestError('Organization has no subscription');
|
throw TRPCBadRequestError('Organization has no subscription');
|
||||||
}
|
}
|
||||||
@@ -163,7 +163,7 @@ export const subscriptionRouter = createTRPCRouter({
|
|||||||
portal: protectedProcedure
|
portal: protectedProcedure
|
||||||
.input(z.object({ organizationId: z.string() }))
|
.input(z.object({ organizationId: z.string() }))
|
||||||
.mutation(async ({ input }) => {
|
.mutation(async ({ input }) => {
|
||||||
const organization = await getOrganizationBySlug(input.organizationId);
|
const organization = await getOrganizationById(input.organizationId);
|
||||||
if (!organization.subscriptionCustomerId) {
|
if (!organization.subscriptionCustomerId) {
|
||||||
throw TRPCBadRequestError('Organization has no subscription');
|
throw TRPCBadRequestError('Organization has no subscription');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -203,6 +203,7 @@ export const zOnboardingProject = z
|
|||||||
website: z.boolean(),
|
website: z.boolean(),
|
||||||
app: z.boolean(),
|
app: z.boolean(),
|
||||||
backend: z.boolean(),
|
backend: z.boolean(),
|
||||||
|
timezone: z.string().optional(),
|
||||||
})
|
})
|
||||||
.superRefine((data, ctx) => {
|
.superRefine((data, ctx) => {
|
||||||
if (!data.organization && !data.organizationId) {
|
if (!data.organization && !data.organizationId) {
|
||||||
@@ -434,3 +435,9 @@ export const zCheckout = z.object({
|
|||||||
productId: z.string(),
|
productId: z.string(),
|
||||||
});
|
});
|
||||||
export type ICheckout = z.infer<typeof zCheckout>;
|
export type ICheckout = z.infer<typeof zCheckout>;
|
||||||
|
|
||||||
|
export const zEditOrganization = z.object({
|
||||||
|
id: z.string().min(2),
|
||||||
|
name: z.string().min(2),
|
||||||
|
timezone: z.string().min(1),
|
||||||
|
});
|
||||||
|
|||||||
23
pnpm-lock.yaml
generated
23
pnpm-lock.yaml
generated
@@ -4,12 +4,6 @@ settings:
|
|||||||
autoInstallPeers: true
|
autoInstallPeers: true
|
||||||
excludeLinksFromLockfile: false
|
excludeLinksFromLockfile: false
|
||||||
|
|
||||||
catalogs:
|
|
||||||
default:
|
|
||||||
zod:
|
|
||||||
specifier: ^3.24.2
|
|
||||||
version: 3.24.2
|
|
||||||
|
|
||||||
importers:
|
importers:
|
||||||
|
|
||||||
.:
|
.:
|
||||||
@@ -891,6 +885,9 @@ importers:
|
|||||||
date-fns:
|
date-fns:
|
||||||
specifier: ^3.3.1
|
specifier: ^3.3.1
|
||||||
version: 3.3.1
|
version: 3.3.1
|
||||||
|
luxon:
|
||||||
|
specifier: ^3.6.1
|
||||||
|
version: 3.6.1
|
||||||
mathjs:
|
mathjs:
|
||||||
specifier: ^12.3.2
|
specifier: ^12.3.2
|
||||||
version: 12.3.2
|
version: 12.3.2
|
||||||
@@ -919,6 +916,9 @@ importers:
|
|||||||
'@openpanel/validation':
|
'@openpanel/validation':
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../validation
|
version: link:../validation
|
||||||
|
'@types/luxon':
|
||||||
|
specifier: ^3.6.2
|
||||||
|
version: 3.6.2
|
||||||
'@types/node':
|
'@types/node':
|
||||||
specifier: 20.14.8
|
specifier: 20.14.8
|
||||||
version: 20.14.8
|
version: 20.14.8
|
||||||
@@ -6435,6 +6435,9 @@ packages:
|
|||||||
'@types/lodash@4.14.202':
|
'@types/lodash@4.14.202':
|
||||||
resolution: {integrity: sha512-OvlIYQK9tNneDlS0VN54LLd5uiPCBOp7gS5Z0f1mjoJYBrtStzgmJBxONW3U6OZqdtNzZPmn9BS/7WI7BFFcFQ==}
|
resolution: {integrity: sha512-OvlIYQK9tNneDlS0VN54LLd5uiPCBOp7gS5Z0f1mjoJYBrtStzgmJBxONW3U6OZqdtNzZPmn9BS/7WI7BFFcFQ==}
|
||||||
|
|
||||||
|
'@types/luxon@3.6.2':
|
||||||
|
resolution: {integrity: sha512-R/BdP7OxEMc44l2Ex5lSXHoIXTB2JLNa3y2QISIbr58U/YcsffyQrYW//hZSdrfxrjRZj3GcUoxMPGdO8gSYuw==}
|
||||||
|
|
||||||
'@types/mdast@4.0.3':
|
'@types/mdast@4.0.3':
|
||||||
resolution: {integrity: sha512-LsjtqsyF+d2/yFOYaN22dHZI1Cpwkrj+g06G8+qtUKlhovPW89YhqSnfKtMbkgmEtYpH2gydRNULd6y8mciAFg==}
|
resolution: {integrity: sha512-LsjtqsyF+d2/yFOYaN22dHZI1Cpwkrj+g06G8+qtUKlhovPW89YhqSnfKtMbkgmEtYpH2gydRNULd6y8mciAFg==}
|
||||||
|
|
||||||
@@ -9597,6 +9600,10 @@ packages:
|
|||||||
resolution: {integrity: sha512-zobTr7akeGHnv7eBOXcRgMeCP6+uyYsczwmeRCauvpvaAltgNyTbLH/+VaEAPUeWBT+1GuNmz4wC/6jtQzbbVA==}
|
resolution: {integrity: sha512-zobTr7akeGHnv7eBOXcRgMeCP6+uyYsczwmeRCauvpvaAltgNyTbLH/+VaEAPUeWBT+1GuNmz4wC/6jtQzbbVA==}
|
||||||
engines: {node: '>=12'}
|
engines: {node: '>=12'}
|
||||||
|
|
||||||
|
luxon@3.6.1:
|
||||||
|
resolution: {integrity: sha512-tJLxrKJhO2ukZ5z0gyjY1zPh3Rh88Ej9P7jNrZiHMUXHae1yvI2imgOZtL1TO8TW6biMMKfTtAOoEJANgtWBMQ==}
|
||||||
|
engines: {node: '>=12'}
|
||||||
|
|
||||||
magic-string@0.30.17:
|
magic-string@0.30.17:
|
||||||
resolution: {integrity: sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==}
|
resolution: {integrity: sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==}
|
||||||
|
|
||||||
@@ -18633,6 +18640,8 @@ snapshots:
|
|||||||
|
|
||||||
'@types/lodash@4.14.202': {}
|
'@types/lodash@4.14.202': {}
|
||||||
|
|
||||||
|
'@types/luxon@3.6.2': {}
|
||||||
|
|
||||||
'@types/mdast@4.0.3':
|
'@types/mdast@4.0.3':
|
||||||
dependencies:
|
dependencies:
|
||||||
'@types/unist': 3.0.2
|
'@types/unist': 3.0.2
|
||||||
@@ -22514,6 +22523,8 @@ snapshots:
|
|||||||
|
|
||||||
luxon@3.4.4: {}
|
luxon@3.4.4: {}
|
||||||
|
|
||||||
|
luxon@3.6.1: {}
|
||||||
|
|
||||||
magic-string@0.30.17:
|
magic-string@0.30.17:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@jridgewell/sourcemap-codec': 1.5.0
|
'@jridgewell/sourcemap-codec': 1.5.0
|
||||||
|
|||||||
Reference in New Issue
Block a user