diff --git a/apps/server/src/v1/middleware.ts b/apps/server/src/v1/middleware.ts index 0b40518a04..d6619fcd5b 100644 --- a/apps/server/src/v1/middleware.ts +++ b/apps/server/src/v1/middleware.ts @@ -55,12 +55,12 @@ export function trackMiddleware(event: EventProps, eventProps?: string[]) { await next(); // REMINDER: only track the event if the request was successful - // REMINDER: use setTimeout to avoid blocking the response - if (c.finalized) { + if (!c.error) { // We have checked the request to be valid already const json = (await c.req.json()) as unknown; const additionalProps = parseInputToProps(json, eventProps); + // REMINDER: use setTimeout to avoid blocking the response setTimeout(async () => { const analytics = await setupAnalytics({ userId: `api_${c.get("workspaceId")}`, diff --git a/apps/web/content-collections.ts b/apps/web/content-collections.ts index 87f1e8c00a..6b3d7740c6 100644 --- a/apps/web/content-collections.ts +++ b/apps/web/content-collections.ts @@ -28,6 +28,8 @@ const prettyCode = [ dark: "github-dark-dimmed", light: "github-light", }, + grid: true, + keepBackground: false, // biome-ignore lint/suspicious/noExplicitAny: onVisitLine(node: any) { // Prevent lines from collapsing in `display: grid` mode, and @@ -36,14 +38,6 @@ const prettyCode = [ node.children = [{ type: "text", value: " " }]; } }, - // biome-ignore lint/suspicious/noExplicitAny: - onVisitHighlightedLine(node: any) { - node.properties.className.push("highlighted"); - }, - // biome-ignore lint/suspicious/noExplicitAny: - onVisitHighlightedWord(node: any) { - node.properties.className = ["word"]; - }, }, ]; @@ -61,6 +55,7 @@ const posts = defineCollection({ url: z.string().optional(), avatar: z.string().optional(), }), + tag: z.enum(["company", "engineering", "education"]), }), transform: async (document, context) => { const mdx = await compileMDX(context, document, { diff --git a/apps/web/package.json b/apps/web/package.json index 8b65e61baa..9a8596b7d7 100644 --- a/apps/web/package.json +++ b/apps/web/package.json @@ -97,7 +97,6 @@ "postcss": "8.4.38", "rehype-autolink-headings": "7.1.0", "rehype-slug": "5.1.0", - "remark-gfm": "3.0.1", "tailwindcss": "3.4.3", "typescript": "5.6.2", "unified": "10.1.2" diff --git a/apps/web/public/assets/posts/event-analytics-implementation/event-analytics-implementation.png b/apps/web/public/assets/posts/event-analytics-implementation/event-analytics-implementation.png new file mode 100644 index 0000000000..f81be4a1fd Binary files /dev/null and b/apps/web/public/assets/posts/event-analytics-implementation/event-analytics-implementation.png differ diff --git a/apps/web/src/components/content/article.tsx b/apps/web/src/components/content/article.tsx index 01eecc463d..c1e81021b8 100644 --- a/apps/web/src/components/content/article.tsx +++ b/apps/web/src/components/content/article.tsx @@ -2,7 +2,7 @@ import type { Post } from "content-collections"; import Image from "next/image"; import Link from "next/link"; -import { Avatar, AvatarFallback, AvatarImage } from "@openstatus/ui"; +import { Avatar, AvatarFallback, AvatarImage, Badge } from "@openstatus/ui"; import { Mdx } from "@/components/content/mdx"; import { formatDate } from "@/lib/utils"; @@ -40,11 +40,15 @@ export function Article({ post }: { post: Post }) { > {post.author.name} -

- {formatDate(post.publishedAt)} - - {post.readingTime} -

+
+ + + {post.readingTime} + + + {post.tag} + +
diff --git a/apps/web/src/components/content/mdx.tsx b/apps/web/src/components/content/mdx.tsx index 8ebbc9218c..acf0602b54 100644 --- a/apps/web/src/components/content/mdx.tsx +++ b/apps/web/src/components/content/mdx.tsx @@ -13,7 +13,7 @@ export function Mdx({ code, className }: MdxProps) { // FIXME: weird behaviour when `prose-headings:font-cal` and on mouse movement font gets bigger
diff --git a/apps/web/src/components/content/pre.tsx b/apps/web/src/components/content/pre.tsx index a0ff9e4b02..e2021627f6 100644 --- a/apps/web/src/components/content/pre.tsx +++ b/apps/web/src/components/content/pre.tsx @@ -1,11 +1,12 @@ "use client"; +import { cn } from "@/lib/utils"; import { Button } from "@openstatus/ui/src/components/button"; import { Clipboard, ClipboardCopy } from "lucide-react"; import React from "react"; export interface PreProps extends React.HTMLAttributes {} -export default function Pre({ children, ...props }: PreProps) { +export default function Pre({ children, className, ...props }: PreProps) { const [copied, setCopied] = React.useState(false); const ref = React.useRef(null); @@ -43,7 +44,7 @@ export default function Pre({ children, ...props }: PreProps) { )} -
+      
         {children}
       
diff --git a/apps/web/src/content/posts/2023-year-review.mdx b/apps/web/src/content/posts/2023-year-review.mdx index ebcec053d8..25219465ba 100644 --- a/apps/web/src/content/posts/2023-year-review.mdx +++ b/apps/web/src/content/posts/2023-year-review.mdx @@ -6,8 +6,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2023-12-29 image: /assets/posts/2023-year-review/title.png +tag: company --- It has been a wild six months for us at OpenStatus. In late June, we began diff --git a/apps/web/src/content/posts/dynamic-breadcrumb-nextjs.mdx b/apps/web/src/content/posts/dynamic-breadcrumb-nextjs.mdx index 375387a0a4..abab0a0cf1 100644 --- a/apps/web/src/content/posts/dynamic-breadcrumb-nextjs.mdx +++ b/apps/web/src/content/posts/dynamic-breadcrumb-nextjs.mdx @@ -5,8 +5,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-08-19 image: /assets/posts/dynamic-breadcrumb-nextjs/breadcrumb.png +tag: engineering --- In this post, we'll dive into the process of creating dynamic breadcrumbs in Next.js using parallel routes. Our goal is to build a breadcrumb component that automatically updates based on the current page and its hierarchy, all while leveraging server-side rendering for optimal performance. diff --git a/apps/web/src/content/posts/event-analytics-implementation.mdx b/apps/web/src/content/posts/event-analytics-implementation.mdx new file mode 100644 index 0000000000..04e08fe9da --- /dev/null +++ b/apps/web/src/content/posts/event-analytics-implementation.mdx @@ -0,0 +1,195 @@ +--- +title: How We Implemented Event Analytics with OpenPanel +description: + Leveraging Hono OpenAPI middleware and tRPC metadata + middleware to implement event analytics easily. +author: + name: Maximilian Kaske + url: https://x.com/mxkaske + avatar: /assets/authors/max.png +publishedAt: 2024-12-27 +image: /assets/posts/event-analytics-implementation/event-analytics-implementation.png +tag: engineering +--- + +We had never really tracked events properly. We had some basic tracking in place, but it was not very useful. It is time to improve that with [OpenPanel](https://openpanel.dev?ref=openstatus). + +After some research, we finally settled on leveraging tRPC and Hono middlewares. Shoutout to [Midday](https://midday.ai?ref=openstatus) for the (tRPC) inspiration. They use a similar approach with [next-safe-action](https://next-safe-action.dev?ref=openstatus) for their server actions. + +This post is not a step-by-step guide but instead presents the core concepts and ideas behind the implementation. Please refer to the [Hono](https://hono.dev?ref=openstatus) or [tRPC](https://trpc.io?ref=openstatus) documentation for more detailed information and our [GitHub](https://openstatus.dev/github) repository for the full implementation. + +--- + +First, let's start with the basics. We need to define the events we want to track, like `page_created`, `user_created`, etc. + +```ts +// packages/analytics/src/events.ts +export type EventProps = { + name: string; + channel: string; +}; + +export const Events = { + CreatePage: { + name: "page_created", + channel: "page", + }, + UpdatePage: { + name: "page_upated", + channel: "page", + }, + // ... add more events +} as const satisfies Record; +``` + +Next, we need to initialize OpenPanel (see [installation](https://openpanel.dev/docs/sdks/javascript)) and set up the analytics in our application. + +```ts +// packages/analytics/src/index.ts +import { + OpenPanel, + type PostEventPayload, + type IdentifyPayload, +} from "@openpanel/sdk"; +import { type EventProps } from "@openstatus/analytics"; + +const op = new OpenPanel({ + clientId: process.env.OPENPANEL_CLIENT_ID, + clientSecret: process.env.OPENPANEL_CLIENT_SECRET, +}); + +export async function setupAnalytics(props: Partial) { + if (props.profileId) { + await op.identify(props): + } + + return { + track: (opts: EventProps & PostEventPayload["properties"]) => { + const { name, ...rest } = opts; + return op.track(name, rest); + }, + }; +} +``` + +Now that we have the basic setup in place, we can start implementing the tracking in our application. We will use the tRPC middleware and metadata to track events. Below, we define the `trackEvent` middleware that will track the event after the procedure has been executed. An `enforceUserSession` middleware can be added to include the user's ID for tracking. + +```ts {8,12,35} /trackEvent/ +// packages/trpc/src/index.ts +import { after } from "next/server"; +import { initTRPC } from "@trpc/server"; +import { setupAnalytics, type EventProps } from "@openstatus/analytics"; +import type { User } from "@openstatus/auth"; + +type Context = { user?: User }; +type Meta = { track?: EventProps }; + +export const t = initTRPC + .context() + .meta() + .create({ /* ... */ }); + + +const trackEvent = t.middleware(async opts => { + const result = await opts.next(opts.ctx); + + if (!result.ok) return result; + + if (opts.meta.track) { + after(async () => { + const identify = opts.ctx.user ? { userId: opts.ctx.user.id } : {}; + const analytics = await setupAnalytics(identify); + await analytics.track(opts.meta.track); + }) + } + return result; +}); + +const enforceUserSession = t.middleware(async opts => { + // ... set user to ctx +}); + +export const protectedProcedure = t.procedure + .use(enforceUserSession) + .use(trackEvent); +``` + +The `after` function (similar to `waitUntil`) will execute the tracking after the procedure has been executed and won't block the response. + +The `next()` return value has an `ok` boolean property to check if the procedure was successful. If not, we don't want to track the event. + +How will we use it in a procedure? Adding a `meta` property will allow us to track the event by defining the event we want to track. + +```ts {6} +// packages/trpc/src/router/page.ts +import { Events } from '@openstatus/analytics'; +import { insertPageSchema } from "@openstatus/db"; +import { createTRPCRouter, protectedProcedure } from "../trpc"; + +export const pageRouter = createTRPCRouter({ + create: protectedProcedure + .meta({ track: Events.CreatePage }) + .input(insertPageSchema) + .mutation(async (opts) => { /* ... */ }) +}); +``` + +Voilà! Each time you want to add tracking to a new procedure, you only need to add the `meta` property with the event you want to track. The middleware handles the rest. + +--- + +Now, how do we track the events within our API? Let's start by adding the `trackMiddleware` function and only track the event if the response has been finalized. + +```ts /trackMiddleware/ +// app/server/src/middleware.ts +import { setupAnalytics, type EventProps } from "@openstatus/analytics"; +import type { Context, Next } from "hono"; +import type { User } from "@openstatus/auth"; + +export function trackMiddleware(event: EventProps) { + return async (c: Context<{ Variables: { user?: User } }, "/*">, next: Next) => { + await next(); + + if (!c.error) { + setTimeout(async () => { + const analytics = await setupAnalytics({ + profileId: c.get("user")?.id, + }); + await analytics.track(event); + }, 0); + } + }; +} +``` + +Depending on where you are running the server, you might want to replace `setTimeout` with `waitUntil` (cf workers, Vercel) or other functions that extend the lifetime of the request without blocking the response. + +And again, we check if there was an `error` before tracking the event. We don't want to track unsuccessful events. + +The [`@hono/zod-openapi`](https://github.com/honojs/middleware/tree/main/packages/zod-openapi) routes have a `middleware` property that allows you to add middleware to the route. This is where we will add the tracking middleware. + +```ts {11} +// apps/web/src/pages/post.ts +import { createRoute } from "@hono/zod-openapi"; +import { Events } from "@openstatus/analytics"; +import { trackMiddleware } from "../middleware"; + +const postRoute = createRoute({ + method: "post", + tags: ["page"], + description: "Create a new Page", + path: "/", + middleware: [trackMiddleware(Events.CreatePage)], + request: { /* ... */ }, + responses: { /* ... */}, +}); + +// ... +``` + +--- + +And that's it. With minimal code changes and the help of middlewares, we have implemented event tracking in our application. You can swap [OpenPanel](https://openpanel.dev?ref=openstatus) with any other analytics provider like PostHog, but give it a try, it's an amazing tool! + +By the way, this approach can be used for audit log tracking for example as well. + +Check out our [GitHub](https://openstatus.dev/github) repository for the full implementation and don't forget to leave a star if you found this helpful. \ No newline at end of file diff --git a/apps/web/src/content/posts/migration-auth-clerk-to-next-auth.mdx b/apps/web/src/content/posts/migration-auth-clerk-to-next-auth.mdx index 4d21346842..7a2e6c0ed0 100644 --- a/apps/web/src/content/posts/migration-auth-clerk-to-next-auth.mdx +++ b/apps/web/src/content/posts/migration-auth-clerk-to-next-auth.mdx @@ -5,8 +5,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-05-15 image: /assets/posts/migration-auth-clerk-to-next-auth/authjs.png +tag: engineering --- We recently switched from [Clerk](https://clerk.com) to NextAuth diff --git a/apps/web/src/content/posts/migration-backend-from-vercel-to-fly.mdx b/apps/web/src/content/posts/migration-backend-from-vercel-to-fly.mdx index 9952224ca9..ff41f11df1 100644 --- a/apps/web/src/content/posts/migration-backend-from-vercel-to-fly.mdx +++ b/apps/web/src/content/posts/migration-backend-from-vercel-to-fly.mdx @@ -6,8 +6,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2023-10-29 image: /assets/posts/migration-backend-from-vercel-to-fly/fly.png +tag: company --- In this article, we are going to see the reasons that made us change our backend diff --git a/apps/web/src/content/posts/migration-planetscale-to-turso.mdx b/apps/web/src/content/posts/migration-planetscale-to-turso.mdx index 16955f50e2..de7aa9bdef 100644 --- a/apps/web/src/content/posts/migration-planetscale-to-turso.mdx +++ b/apps/web/src/content/posts/migration-planetscale-to-turso.mdx @@ -6,7 +6,9 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2023-08-20 +tag: company --- ## What are we building ? 🏗️ diff --git a/apps/web/src/content/posts/monitoring-latency-cf-workers-fly-koyeb-raylway-render.mdx b/apps/web/src/content/posts/monitoring-latency-cf-workers-fly-koyeb-raylway-render.mdx index 3e15466a7e..3906c9e654 100644 --- a/apps/web/src/content/posts/monitoring-latency-cf-workers-fly-koyeb-raylway-render.mdx +++ b/apps/web/src/content/posts/monitoring-latency-cf-workers-fly-koyeb-raylway-render.mdx @@ -7,8 +7,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-02-19 image: /assets/posts/monitoring-latency/all-hosting-providers.png +tag: education --- > ⚠️ We are using the default settings for each provider and conducting diff --git a/apps/web/src/content/posts/monitoring-latency-vercel-edge-vs-serverless.mdx b/apps/web/src/content/posts/monitoring-latency-vercel-edge-vs-serverless.mdx index 38755594a2..060122cd1e 100644 --- a/apps/web/src/content/posts/monitoring-latency-vercel-edge-vs-serverless.mdx +++ b/apps/web/src/content/posts/monitoring-latency-vercel-edge-vs-serverless.mdx @@ -6,8 +6,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-03-14 image: /assets/posts/monitoring-vercel/serverless-vs-edge.png +tag: education --- In our previous diff --git a/apps/web/src/content/posts/our-new-pricing-explained.mdx b/apps/web/src/content/posts/our-new-pricing-explained.mdx index 48a8dbde79..b4ec6ac9fe 100644 --- a/apps/web/src/content/posts/our-new-pricing-explained.mdx +++ b/apps/web/src/content/posts/our-new-pricing-explained.mdx @@ -6,8 +6,10 @@ description: author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-01-15 image: /assets/posts/our-new-pricing-explained/pricing-hard.png +tag: company --- We began the new year with two goals: to increase profitability and attract more diff --git a/apps/web/src/content/posts/pricing-update-july-2024.mdx b/apps/web/src/content/posts/pricing-update-july-2024.mdx index 6bacef27eb..ddb98a1b00 100644 --- a/apps/web/src/content/posts/pricing-update-july-2024.mdx +++ b/apps/web/src/content/posts/pricing-update-july-2024.mdx @@ -4,8 +4,10 @@ description: We have update our pricing again, let's deep dive into it. author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-07-24 image: /assets/posts/pricing-update-july-2024/pricing.png +tag: company --- diff --git a/apps/web/src/content/posts/q1-2024-update.mdx b/apps/web/src/content/posts/q1-2024-update.mdx index d56eea2a63..ab10ff3927 100644 --- a/apps/web/src/content/posts/q1-2024-update.mdx +++ b/apps/web/src/content/posts/q1-2024-update.mdx @@ -4,8 +4,10 @@ description: Let's review our first quarter of 2024. author: name: Thibault Le Ouay Ducasse url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-04-29 image: /assets/posts/q1-2024-update/q1-2024-update.png +tag: company --- A lot has happened in the first quarter of 2024. We have been working hard to diff --git a/apps/web/src/content/posts/reflecting-1-year-building-openstatus.mdx b/apps/web/src/content/posts/reflecting-1-year-building-openstatus.mdx index d8da090804..9688b5ed60 100644 --- a/apps/web/src/content/posts/reflecting-1-year-building-openstatus.mdx +++ b/apps/web/src/content/posts/reflecting-1-year-building-openstatus.mdx @@ -5,8 +5,10 @@ description: Reflecting on our first year of building OpenStatus, an open source author: name: Thibault Le Ouay url: https://bsky.app/profile/thibaultleouay.dev + avatar: /assets/authors/thibault.jpeg publishedAt: 2024-09-26 image: /assets/posts/reflecting-1-year-building-openstatus/first-year.png +tag: company --- In one year, we've grown our open-source synthetic monitoring platform to more than 6,000 GitHub stars and 5,000 users. This article will reflect on our first year of building OpenStatus. diff --git a/apps/web/src/content/posts/secure-api-with-unkey.mdx b/apps/web/src/content/posts/secure-api-with-unkey.mdx index c5d51acd4c..7c8ec66bb1 100644 --- a/apps/web/src/content/posts/secure-api-with-unkey.mdx +++ b/apps/web/src/content/posts/secure-api-with-unkey.mdx @@ -4,8 +4,10 @@ description: The simplest way to secure your API routes within seconds. author: name: Maximilian Kaske url: https://twitter.com/mxkaske + avatar: /assets/authors/max.png publishedAt: 2023-10-01 image: /assets/posts/secure-api-with-unkey/unkey.png +tag: engineering --- ## Introduction diff --git a/apps/web/src/content/posts/the-first-48-hours.mdx b/apps/web/src/content/posts/the-first-48-hours.mdx index bebf22571a..222cc0e41a 100644 --- a/apps/web/src/content/posts/the-first-48-hours.mdx +++ b/apps/web/src/content/posts/the-first-48-hours.mdx @@ -5,7 +5,9 @@ description: The numbers, limits we faced and consequences we have taken. author: name: Maximilian Kaske url: https://twitter.com/mxkaske + avatar: /assets/authors/max.png publishedAt: 2023-08-02 +tag: company --- ## 48 hours of Rollercoaster 🎢 diff --git a/apps/web/src/styles/globals.css b/apps/web/src/styles/globals.css index 6a9f3a0b05..cacf0fa5fd 100644 --- a/apps/web/src/styles/globals.css +++ b/apps/web/src/styles/globals.css @@ -102,4 +102,12 @@ body { @apply bg-background text-foreground; } + + [data-highlighted-chars] { + @apply bg-muted rounded; + } + + [data-highlighted-line] { + @apply bg-muted rounded; + } } diff --git a/packages/api/src/trpc.ts b/packages/api/src/trpc.ts index 64e5adddbb..48127ff6c5 100644 --- a/packages/api/src/trpc.ts +++ b/packages/api/src/trpc.ts @@ -194,6 +194,11 @@ const enforceUserIsAuthed = t.middleware(async (opts) => { return result; } + // REMINDER: We only track the event if the request was successful + if (!result.ok) { + return result; + } + // REMINDER: We only track the event if the request was successful // REMINDER: We are not blocking the request after(async () => { diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 385c1d48b2..36104bd5a2 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -459,9 +459,6 @@ importers: rehype-slug: specifier: 5.1.0 version: 5.1.0 - remark-gfm: - specifier: 3.0.1 - version: 3.0.1 tailwindcss: specifier: 3.4.3 version: 3.4.3(ts-node@10.9.2(@types/node@20.14.8)(typescript@5.6.2)) @@ -7636,9 +7633,6 @@ packages: mdast-util-directive@3.0.0: resolution: {integrity: sha512-JUpYOqKI4mM3sZcNxmF/ox04XYFFkNwr0CFlrQIkCwbvH0xzMCqkMqAde9wRd80VAhaUrwFwKm2nxretdT1h7Q==} - mdast-util-find-and-replace@2.2.2: - resolution: {integrity: sha512-MTtdFRz/eMDHXzeK6W3dO7mXUlF82Gom4y0oOgvHhh/HXZAGvIQDUvQ0SuUx+j2tv44b8xTHOm8K/9OoRFnXKw==} - mdast-util-find-and-replace@3.0.1: resolution: {integrity: sha512-SG21kZHGC3XRTSUhtofZkBzZTJNM5ecCi0SK2IMKmSXR8vO3peL+kb1O0z7Zl83jKtutG4k5Wv/W7V3/YHvzPA==} @@ -7651,39 +7645,21 @@ packages: mdast-util-frontmatter@2.0.1: resolution: {integrity: sha512-LRqI9+wdgC25P0URIJY9vwocIzCcksduHQ9OF2joxQoyTNVduwLAFUzjoopuRJbJAReaKrNQKAZKL3uCMugWJA==} - mdast-util-gfm-autolink-literal@1.0.3: - resolution: {integrity: sha512-My8KJ57FYEy2W2LyNom4n3E7hKTuQk/0SES0u16tjA9Z3oFkF4RrC/hPAPgjlSpezsOvI8ObcXcElo92wn5IGA==} - mdast-util-gfm-autolink-literal@2.0.0: resolution: {integrity: sha512-FyzMsduZZHSc3i0Px3PQcBT4WJY/X/RCtEJKuybiC6sjPqLv7h1yqAkmILZtuxMSsUyaLUWNp71+vQH2zqp5cg==} - mdast-util-gfm-footnote@1.0.2: - resolution: {integrity: sha512-56D19KOGbE00uKVj3sgIykpwKL179QsVFwx/DCW0u/0+URsryacI4MAdNJl0dh+u2PSsD9FtxPFbHCzJ78qJFQ==} - mdast-util-gfm-footnote@2.0.0: resolution: {integrity: sha512-5jOT2boTSVkMnQ7LTrd6n/18kqwjmuYqo7JUPe+tRCY6O7dAuTFMtTPauYYrMPpox9hlN0uOx/FL8XvEfG9/mQ==} - mdast-util-gfm-strikethrough@1.0.3: - resolution: {integrity: sha512-DAPhYzTYrRcXdMjUtUjKvW9z/FNAMTdU0ORyMcbmkwYNbKocDpdk+PX1L1dQgOID/+vVs1uBQ7ElrBQfZ0cuiQ==} - mdast-util-gfm-strikethrough@2.0.0: resolution: {integrity: sha512-mKKb915TF+OC5ptj5bJ7WFRPdYtuHv0yTRxK2tJvi+BDqbkiG7h7u/9SI89nRAYcmap2xHQL9D+QG/6wSrTtXg==} - mdast-util-gfm-table@1.0.7: - resolution: {integrity: sha512-jjcpmNnQvrmN5Vx7y7lEc2iIOEytYv7rTvu+MeyAsSHTASGCCRA79Igg2uKssgOs1i1po8s3plW0sTu1wkkLGg==} - mdast-util-gfm-table@2.0.0: resolution: {integrity: sha512-78UEvebzz/rJIxLvE7ZtDd/vIQ0RHv+3Mh5DR96p7cS7HsBhYIICDBCu8csTNWNO6tBWfqXPWekRuj2FNOGOZg==} - mdast-util-gfm-task-list-item@1.0.2: - resolution: {integrity: sha512-PFTA1gzfp1B1UaiJVyhJZA1rm0+Tzn690frc/L8vNX1Jop4STZgOE6bxUhnzdVSB+vm2GU1tIsuQcA9bxTQpMQ==} - mdast-util-gfm-task-list-item@2.0.0: resolution: {integrity: sha512-IrtvNvjxC1o06taBAVJznEnkiHxLFTzgonUdy8hzFVeDun0uTjxxrRGVaNFqkU1wJR3RBPEfsxmU6jDWPofrTQ==} - mdast-util-gfm@2.0.2: - resolution: {integrity: sha512-qvZ608nBppZ4icQlhQQIAdc6S3Ffj9RGmzwUKUWuEICFnd1LVkN3EktF7ZHAgfcEdvZB5owU9tQgt99e2TlLjg==} - mdast-util-gfm@3.0.0: resolution: {integrity: sha512-dgQEX5Amaq+DuUqf26jJqSK9qgixgd6rYDHAv4aTBuA92cTknZlKpPfa86Z/s8Dj8xsAQpFfBmPUHWJBWqS4Bw==} @@ -7699,9 +7675,6 @@ packages: mdast-util-mdxjs-esm@2.0.1: resolution: {integrity: sha512-EcmOpxsZ96CvlP03NghtH1EsLtr0n9Tm4lPUJUBccV9RwUOneqSycg19n5HGzCf+10LozMRSObtVr3ee1WoHtg==} - mdast-util-phrasing@3.0.1: - resolution: {integrity: sha512-WmI1gTXUBJo4/ZmSk79Wcb2HcjPJBzM1nlI/OUWA8yk2X9ik3ffNbBGsU+09BFmXaL1IBb9fiuvq6/KMiNycSg==} - mdast-util-phrasing@4.1.0: resolution: {integrity: sha512-TqICwyvJJpBwvGAMZjj4J2n0X8QWp21b9l0o7eXyVJ25YNWYbJDVIyD1bZXE6WtV6RmKJVYmQAKWa0zWOABz2w==} @@ -7711,9 +7684,6 @@ packages: mdast-util-to-hast@13.2.0: resolution: {integrity: sha512-QGYKEuUsYT9ykKBCMOEDLsU5JRObWQusAolFMeko/tYPufNkRffBAQjIE+99jbA87xv6FgmjLtwjh9wBWajwAA==} - mdast-util-to-markdown@1.5.0: - resolution: {integrity: sha512-bbv7TPv/WC49thZPg3jXuqzuvI45IL2EVAr/KxF0BSdHsU0ceFHOmwQn6evxAh1GaoK/6GQ1wp4R4oW2+LFL/A==} - mdast-util-to-markdown@2.1.0: resolution: {integrity: sha512-SR2VnIEdVNCJbP6y7kVTJgPLifdr8WEU440fQec7qHoHOUz/oJ2jmNRqdDQ3rbiStOXb2mCDGTuwsK5OPUgYlQ==} @@ -7748,45 +7718,24 @@ packages: micromark-extension-frontmatter@2.0.0: resolution: {integrity: sha512-C4AkuM3dA58cgZha7zVnuVxBhDsbttIMiytjgsM2XbHAB2faRVaHRle40558FBN+DJcrLNCoqG5mlrpdU4cRtg==} - micromark-extension-gfm-autolink-literal@1.0.5: - resolution: {integrity: sha512-z3wJSLrDf8kRDOh2qBtoTRD53vJ+CWIyo7uyZuxf/JAbNJjiHsOpG1y5wxk8drtv3ETAHutCu6N3thkOOgueWg==} - micromark-extension-gfm-autolink-literal@2.1.0: resolution: {integrity: sha512-oOg7knzhicgQ3t4QCjCWgTmfNhvQbDDnJeVu9v81r7NltNCVmhPy1fJRX27pISafdjL+SVc4d3l48Gb6pbRypw==} - micromark-extension-gfm-footnote@1.1.2: - resolution: {integrity: sha512-Yxn7z7SxgyGWRNa4wzf8AhYYWNrwl5q1Z8ii+CSTTIqVkmGZF1CElX2JI8g5yGoM3GAman9/PVCUFUSJ0kB/8Q==} - micromark-extension-gfm-footnote@2.1.0: resolution: {integrity: sha512-/yPhxI1ntnDNsiHtzLKYnE3vf9JZ6cAisqVDauhp4CEHxlb4uoOTxOCJ+9s51bIB8U1N1FJ1RXOKTIlD5B/gqw==} - micromark-extension-gfm-strikethrough@1.0.7: - resolution: {integrity: sha512-sX0FawVE1o3abGk3vRjOH50L5TTLr3b5XMqnP9YDRb34M0v5OoZhG+OHFz1OffZ9dlwgpTBKaT4XW/AsUVnSDw==} - micromark-extension-gfm-strikethrough@2.1.0: resolution: {integrity: sha512-ADVjpOOkjz1hhkZLlBiYA9cR2Anf8F4HqZUO6e5eDcPQd0Txw5fxLzzxnEkSkfnD0wziSGiv7sYhk/ktvbf1uw==} - micromark-extension-gfm-table@1.0.7: - resolution: {integrity: sha512-3ZORTHtcSnMQEKtAOsBQ9/oHp9096pI/UvdPtN7ehKvrmZZ2+bbWhi0ln+I9drmwXMt5boocn6OlwQzNXeVeqw==} - micromark-extension-gfm-table@2.1.0: resolution: {integrity: sha512-Ub2ncQv+fwD70/l4ou27b4YzfNaCJOvyX4HxXU15m7mpYY+rjuWzsLIPZHJL253Z643RpbcP1oeIJlQ/SKW67g==} - micromark-extension-gfm-tagfilter@1.0.2: - resolution: {integrity: sha512-5XWB9GbAUSHTn8VPU8/1DBXMuKYT5uOgEjJb8gN3mW0PNW5OPHpSdojoqf+iq1xo7vWzw/P8bAHY0n6ijpXF7g==} - micromark-extension-gfm-tagfilter@2.0.0: resolution: {integrity: sha512-xHlTOmuCSotIA8TW1mDIM6X2O1SiX5P9IuDtqGonFhEK0qgRI4yeC6vMxEV2dgyr2TiD+2PQ10o+cOhdVAcwfg==} - micromark-extension-gfm-task-list-item@1.0.5: - resolution: {integrity: sha512-RMFXl2uQ0pNQy6Lun2YBYT9g9INXtWJULgbt01D/x8/6yJ2qpKyzdZD3pi6UIkzF++Da49xAelVKUeUMqd5eIQ==} - micromark-extension-gfm-task-list-item@2.1.0: resolution: {integrity: sha512-qIBZhqxqI6fjLDYFTBIa4eivDMnP+OZqsNwmQ3xNLE4Cxwc+zfQEfbs6tzAo2Hjq+bh6q5F+Z8/cksrLFYWQQw==} - micromark-extension-gfm@2.0.3: - resolution: {integrity: sha512-vb9OoHqrhCmbRidQv/2+Bc6pkP0FrtlhurxZofvOEy5o8RtuuvTq+RQ1Vw5ZDNrVraQZu3HixESqbG+0iKk/MQ==} - micromark-extension-gfm@3.0.0: resolution: {integrity: sha512-vsKArQsicm7t0z2GugkCKtZehqUm31oeGBV/KVSorWSy8ZlNAv7ytjFhvaryUiCUJYqs+NoE6AFhpQvBTM6Q4w==} @@ -8954,9 +8903,6 @@ packages: remark-frontmatter@5.0.0: resolution: {integrity: sha512-XTFYvNASMe5iPN0719nPrdItC9aU0ssC4v14mH1BCi1u0n1gAocqcujWUrByftZTbLhRtiKRyjYTSIOcr69UVQ==} - remark-gfm@3.0.1: - resolution: {integrity: sha512-lEFDoi2PICJyNrACFOfDD3JlLkuSbOa5Wd8EPt06HUdptv8Gn0bxYTdbU/XXQ3swAPkEaGxxPN9cbnMHvVu1Ig==} - remark-gfm@4.0.0: resolution: {integrity: sha512-U92vJgBPkbw4Zfu/IiW2oTZLSL3Zpv+uI7My2eq8JxKgqraFdU8YUGicEJCEgSbeaG+QDFqIcwwfMTOEelPxuA==} @@ -9245,8 +9191,8 @@ packages: react: ^18.0.0 || ^19.0.0 || ^19.0.0-rc react-dom: ^18.0.0 || ^19.0.0 || ^19.0.0-rc - sort-keys@5.0.0: - resolution: {integrity: sha512-Pdz01AvCAottHTPQGzndktFNdbRA75BgOfeT1hH+AMnJFv8lynkPi42rfeEhpx1saTEI3YNMWxfqu0sFD1G8pw==} + sort-keys@5.1.0: + resolution: {integrity: sha512-aSbHV0DaBcr7u0PVHXzM6NbZNAtrr9sF6+Qfs9UUVG7Ll3jQ6hHi8F/xqIIcn2rvIVbr0v/2zyjSdwSV47AgLQ==} engines: {node: '>=12'} source-map-js@1.2.0: @@ -13854,7 +13800,7 @@ snapshots: nanoid: 5.0.7 postcss-nested: 6.0.1(postcss@8.4.49) unhead: 1.9.15 - unified: 11.0.4 + unified: 11.0.5 vue: 3.4.31(typescript@5.7.2) transitivePeerDependencies: - '@jest/globals' @@ -17152,7 +17098,7 @@ snapshots: hash-obj@4.0.0: dependencies: is-obj: 3.0.0 - sort-keys: 5.0.0 + sort-keys: 5.1.0 type-fest: 1.4.0 hasown@2.0.2: @@ -18013,13 +17959,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-find-and-replace@2.2.2: - dependencies: - '@types/mdast': 3.0.14 - escape-string-regexp: 5.0.0 - unist-util-is: 5.2.1 - unist-util-visit-parents: 5.1.3 - mdast-util-find-and-replace@3.0.1: dependencies: '@types/mdast': 4.0.4 @@ -18072,13 +18011,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-gfm-autolink-literal@1.0.3: - dependencies: - '@types/mdast': 3.0.14 - ccount: 2.0.1 - mdast-util-find-and-replace: 2.2.2 - micromark-util-character: 1.2.0 - mdast-util-gfm-autolink-literal@2.0.0: dependencies: '@types/mdast': 4.0.4 @@ -18087,12 +18019,6 @@ snapshots: mdast-util-find-and-replace: 3.0.1 micromark-util-character: 2.1.0 - mdast-util-gfm-footnote@1.0.2: - dependencies: - '@types/mdast': 3.0.14 - mdast-util-to-markdown: 1.5.0 - micromark-util-normalize-identifier: 1.1.0 - mdast-util-gfm-footnote@2.0.0: dependencies: '@types/mdast': 4.0.4 @@ -18103,11 +18029,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-gfm-strikethrough@1.0.3: - dependencies: - '@types/mdast': 3.0.14 - mdast-util-to-markdown: 1.5.0 - mdast-util-gfm-strikethrough@2.0.0: dependencies: '@types/mdast': 4.0.4 @@ -18116,15 +18037,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-gfm-table@1.0.7: - dependencies: - '@types/mdast': 3.0.14 - markdown-table: 3.0.3 - mdast-util-from-markdown: 1.3.1 - mdast-util-to-markdown: 1.5.0 - transitivePeerDependencies: - - supports-color - mdast-util-gfm-table@2.0.0: dependencies: '@types/mdast': 4.0.4 @@ -18135,11 +18047,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-gfm-task-list-item@1.0.2: - dependencies: - '@types/mdast': 3.0.14 - mdast-util-to-markdown: 1.5.0 - mdast-util-gfm-task-list-item@2.0.0: dependencies: '@types/mdast': 4.0.4 @@ -18149,18 +18056,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-gfm@2.0.2: - dependencies: - mdast-util-from-markdown: 1.3.1 - mdast-util-gfm-autolink-literal: 1.0.3 - mdast-util-gfm-footnote: 1.0.2 - mdast-util-gfm-strikethrough: 1.0.3 - mdast-util-gfm-table: 1.0.7 - mdast-util-gfm-task-list-item: 1.0.2 - mdast-util-to-markdown: 1.5.0 - transitivePeerDependencies: - - supports-color - mdast-util-gfm@3.0.0: dependencies: mdast-util-from-markdown: 2.0.1 @@ -18222,11 +18117,6 @@ snapshots: transitivePeerDependencies: - supports-color - mdast-util-phrasing@3.0.1: - dependencies: - '@types/mdast': 3.0.14 - unist-util-is: 5.2.1 - mdast-util-phrasing@4.1.0: dependencies: '@types/mdast': 4.0.4 @@ -18255,17 +18145,6 @@ snapshots: unist-util-visit: 5.0.0 vfile: 6.0.3 - mdast-util-to-markdown@1.5.0: - dependencies: - '@types/mdast': 3.0.14 - '@types/unist': 2.0.9 - longest-streak: 3.1.0 - mdast-util-phrasing: 3.0.1 - mdast-util-to-string: 3.2.0 - micromark-util-decode-string: 1.1.0 - unist-util-visit: 4.1.2 - zwitch: 2.0.4 - mdast-util-to-markdown@2.1.0: dependencies: '@types/mdast': 4.0.4 @@ -18360,13 +18239,6 @@ snapshots: micromark-util-symbol: 2.0.0 micromark-util-types: 2.0.0 - micromark-extension-gfm-autolink-literal@1.0.5: - dependencies: - micromark-util-character: 1.2.0 - micromark-util-sanitize-uri: 1.2.0 - micromark-util-symbol: 1.1.0 - micromark-util-types: 1.1.0 - micromark-extension-gfm-autolink-literal@2.1.0: dependencies: micromark-util-character: 2.1.0 @@ -18374,17 +18246,6 @@ snapshots: micromark-util-symbol: 2.0.0 micromark-util-types: 2.0.0 - micromark-extension-gfm-footnote@1.1.2: - dependencies: - micromark-core-commonmark: 1.1.0 - micromark-factory-space: 1.1.0 - micromark-util-character: 1.2.0 - micromark-util-normalize-identifier: 1.1.0 - micromark-util-sanitize-uri: 1.2.0 - micromark-util-symbol: 1.1.0 - micromark-util-types: 1.1.0 - uvu: 0.5.6 - micromark-extension-gfm-footnote@2.1.0: dependencies: devlop: 1.1.0 @@ -18396,15 +18257,6 @@ snapshots: micromark-util-symbol: 2.0.0 micromark-util-types: 2.0.0 - micromark-extension-gfm-strikethrough@1.0.7: - dependencies: - micromark-util-chunked: 1.1.0 - micromark-util-classify-character: 1.1.0 - micromark-util-resolve-all: 1.1.0 - micromark-util-symbol: 1.1.0 - micromark-util-types: 1.1.0 - uvu: 0.5.6 - micromark-extension-gfm-strikethrough@2.1.0: dependencies: devlop: 1.1.0 @@ -18414,14 +18266,6 @@ snapshots: micromark-util-symbol: 2.0.0 micromark-util-types: 2.0.0 - micromark-extension-gfm-table@1.0.7: - dependencies: - micromark-factory-space: 1.1.0 - micromark-util-character: 1.2.0 - micromark-util-symbol: 1.1.0 - micromark-util-types: 1.1.0 - uvu: 0.5.6 - micromark-extension-gfm-table@2.1.0: dependencies: devlop: 1.1.0 @@ -18430,22 +18274,10 @@ snapshots: micromark-util-symbol: 2.0.0 micromark-util-types: 2.0.0 - micromark-extension-gfm-tagfilter@1.0.2: - dependencies: - micromark-util-types: 1.1.0 - micromark-extension-gfm-tagfilter@2.0.0: dependencies: micromark-util-types: 2.0.0 - micromark-extension-gfm-task-list-item@1.0.5: - dependencies: - micromark-factory-space: 1.1.0 - micromark-util-character: 1.2.0 - micromark-util-symbol: 1.1.0 - micromark-util-types: 1.1.0 - uvu: 0.5.6 - micromark-extension-gfm-task-list-item@2.1.0: dependencies: devlop: 1.1.0 @@ -18454,17 +18286,6 @@ snapshots: micromark-util-symbol: 2.0.0 micromark-util-types: 2.0.0 - micromark-extension-gfm@2.0.3: - dependencies: - micromark-extension-gfm-autolink-literal: 1.0.5 - micromark-extension-gfm-footnote: 1.1.2 - micromark-extension-gfm-strikethrough: 1.0.7 - micromark-extension-gfm-table: 1.0.7 - micromark-extension-gfm-tagfilter: 1.0.2 - micromark-extension-gfm-task-list-item: 1.0.5 - micromark-util-combine-extensions: 1.1.0 - micromark-util-types: 1.1.0 - micromark-extension-gfm@3.0.0: dependencies: micromark-extension-gfm-autolink-literal: 2.1.0 @@ -19920,7 +19741,7 @@ snapshots: rehype-parse@9.0.0: dependencies: '@types/hast': 3.0.4 - hast-util-from-html: 2.0.1 + hast-util-from-html: 2.0.3 unified: 11.0.5 rehype-pretty-code@0.10.0(shiki@0.14.4): @@ -20005,15 +19826,6 @@ snapshots: transitivePeerDependencies: - supports-color - remark-gfm@3.0.1: - dependencies: - '@types/mdast': 3.0.14 - mdast-util-gfm: 2.0.2 - micromark-extension-gfm: 2.0.3 - unified: 10.1.2 - transitivePeerDependencies: - - supports-color - remark-gfm@4.0.0: dependencies: '@types/mdast': 4.0.4 @@ -20456,7 +20268,7 @@ snapshots: react: 19.0.0 react-dom: 19.0.0(react@19.0.0) - sort-keys@5.0.0: + sort-keys@5.1.0: dependencies: is-plain-obj: 4.1.0