From 6eafdabb3f7541173c54c56de75e74da997c241f Mon Sep 17 00:00:00 2001 From: Esteban Dalel R Date: Mon, 14 Aug 2023 11:31:02 -0500 Subject: [PATCH 1/2] Feature/posthog frontend (#242) * Create param validator * Move discord * Move user settings getter to app router * Use standard var * move getAllPublicUserData to app router * Better error handling * Delete getAllData.ts * Delete discord * Throw errors * Handle error throwing * Move updateSettings to app router * Update confluence.svg * Configure jest * Create settings.test.ts * Create updateSettings.test.ts * Add getting test * Use generalized validator * Move Stripe to app router * Move Sendgrid to app router * Move vscode to airtable Analytics to app router * Frontend analytics * Change export * Fix export * Fix export --- .../analytics/vsmarketplace/update/route.ts | 18 +++++++--- app/api/sendgrid/sendTeammateInvite/route.ts | 28 +++++++++++++++ app/api/sendgrid/sendWelcome/route.ts | 18 ++++++++++ .../api/stripe/createSubscription/route.ts | 22 +++++++++--- app/layout.tsx | 26 ++++++++------ app/providers.tsx | 35 +++++++++++++++++++ pages/api/extension/getContext.ts | 28 +++++++-------- pages/api/hover/getHoverData.ts | 26 +++++++------- pages/api/sendgrid/sendTeammateInvite.ts | 17 --------- pages/api/sendgrid/sendWelcome.ts | 12 ------- utils/sendgrid/sendTeammateInvite.ts | 6 ++-- 11 files changed, 157 insertions(+), 79 deletions(-) rename pages/api/analytics/vsmarketplace/update.ts => app/api/analytics/vsmarketplace/update/route.ts (77%) create mode 100644 app/api/sendgrid/sendTeammateInvite/route.ts create mode 100644 app/api/sendgrid/sendWelcome/route.ts rename pages/api/stripe/createSubscription.ts => app/api/stripe/createSubscription/route.ts (70%) create mode 100644 app/providers.tsx delete mode 100644 pages/api/sendgrid/sendTeammateInvite.ts delete mode 100644 pages/api/sendgrid/sendWelcome.ts diff --git a/pages/api/analytics/vsmarketplace/update.ts b/app/api/analytics/vsmarketplace/update/route.ts similarity index 77% rename from pages/api/analytics/vsmarketplace/update.ts rename to app/api/analytics/vsmarketplace/update/route.ts index 9be51aeee..5389328bd 100644 --- a/pages/api/analytics/vsmarketplace/update.ts +++ b/app/api/analytics/vsmarketplace/update/route.ts @@ -1,3 +1,5 @@ +import { NextResponse } from "next/server"; +import validateParams from "../../../../../utils/api/validateParams"; import Airtable from "airtable"; Airtable.configure({ endpointUrl: "https://api.airtable.com", @@ -24,13 +26,21 @@ function toRecords(infoArray) { } return records; } -export default async function handler(req, res) { - const { dailyStats } = req.body; +export async function POST(request: Request) { + const req = await request.json(); + const { missingParams } = validateParams(req, ["dailyStats"]); + + if (missingParams.length > 0) { + return NextResponse.json({ + error: `Missing parameters: ${missingParams.join(", ")}`, + }); + } + const { dailyStats } = req; let records; if (dailyStats.length < 10) { records = toRecords(dailyStats.slice(0, 10)); let createdRecord = await base("vscmarketplace").create(records); - res.status(200).json(createdRecord); + return NextResponse.json(createdRecord); } else { records = toRecords(dailyStats); // now take the records and chunk them into groups of 10 @@ -53,6 +63,6 @@ export default async function handler(req, res) { // @ts-ignore createdRecords.push(createdRecord); } - res.status(200).json(createdRecords); + return NextResponse.json(createdRecords); } } diff --git a/app/api/sendgrid/sendTeammateInvite/route.ts b/app/api/sendgrid/sendTeammateInvite/route.ts new file mode 100644 index 000000000..9ef9686ae --- /dev/null +++ b/app/api/sendgrid/sendTeammateInvite/route.ts @@ -0,0 +1,28 @@ +import { NextResponse } from "next/server"; +import validateParams from "../../../../utils/api/validateParams"; +import sendTeammateInvite from "../../../../utils/sendgrid/sendTeammateInvite"; + +export async function POST(request: Request) { + const req = await request.json(); + const { missingParams } = validateParams(req, [ + "sender", + "email", + "inviteUrl", + "teamName", + ]); + + if (missingParams.length > 0) { + return NextResponse.json({ + error: `Missing parameters: ${missingParams.join(", ")}`, + }); + } + const { sender, email, inviteUrl, teamName } = req; + + let emailSent = await sendTeammateInvite({ + sender, + teammateEmail: email, + inviteUrl, + teamName, + }); + return NextResponse.json(emailSent); +} diff --git a/app/api/sendgrid/sendWelcome/route.ts b/app/api/sendgrid/sendWelcome/route.ts new file mode 100644 index 000000000..71ae598b9 --- /dev/null +++ b/app/api/sendgrid/sendWelcome/route.ts @@ -0,0 +1,18 @@ +import { NextResponse } from "next/server"; +import validateParams from "../../../../utils/api/validateParams"; +import sendWelcome from "../../../../utils/sendgrid/sendWelcome"; + +export async function POST(request: Request) { + const req = await request.json(); + const { missingParams } = validateParams(req, ["sender", "emails"]); + + if (missingParams.length > 0) { + return NextResponse.json({ + error: `Missing parameters: ${missingParams.join(", ")}`, + }); + } + const { sender, emails } = req; + + let emailSent = await sendWelcome({ sender, emails }); + return NextResponse.json(emailSent); +} diff --git a/pages/api/stripe/createSubscription.ts b/app/api/stripe/createSubscription/route.ts similarity index 70% rename from pages/api/stripe/createSubscription.ts rename to app/api/stripe/createSubscription/route.ts index 5722c395a..4614f8278 100644 --- a/pages/api/stripe/createSubscription.ts +++ b/app/api/stripe/createSubscription/route.ts @@ -1,15 +1,25 @@ +import { NextResponse } from "next/server"; import Stripe from "stripe"; +import validateParams from "../../../../utils/api/validateParams"; const stripe = new Stripe(process.env.NEXT_PUBLIC_STRIPE_SECRET_KEY!, { apiVersion: "2022-08-01", }); -export default async function handler(req, res) { +export async function POST(request: Request) { + const req = await request.json(); + const { missingParams } = validateParams(req, ["email"]); + + if (missingParams.length > 0) { + return NextResponse.json({ + error: `Missing parameters: ${missingParams.join(", ")}`, + }); + } try { // create a stripe customer and get its id const customerId = await stripe.customers .create({ - email: req.body.email, + email: req.email, }) .then((customer) => { return customer.id; @@ -23,7 +33,7 @@ export default async function handler(req, res) { items: [ { price: priceId, - quantity: req.body.quantity, + quantity: req.quantity, }, ], payment_behavior: "default_incomplete", @@ -31,7 +41,7 @@ export default async function handler(req, res) { expand: ["latest_invoice.payment_intent"], }); - res.send({ + return NextResponse.json({ subscriptionId: subscription.id, // We use Stripe's Expand functionality to get the latest invoice and its payment intent // So we can pass it to the front end to confirm the payment @@ -42,6 +52,8 @@ export default async function handler(req, res) { } catch (error) { // get error code const errorCode = error.raw?.code; - return res.status(errorCode).send({ error: { message: error.message } }); + return NextResponse.json({ + error: { message: error.message, errorCode }, + }); } } diff --git a/app/layout.tsx b/app/layout.tsx index 96c14ae6b..e81c211ca 100644 --- a/app/layout.tsx +++ b/app/layout.tsx @@ -5,6 +5,7 @@ import { getServerSession } from "next-auth"; import Navbar from "../components/Navbar"; import Header from "../components/Header"; import LogInBtn from "../components/login-btn"; +import { PHProvider, PostHogPageview } from "./providers"; import AuthProvider from "../lib/auth/AuthProvider"; @@ -30,17 +31,22 @@ export default async function RootLayout({ return ( + + + - {userEmail ? ( - <> -
- - {children} - - - ) : ( - - )} + + {userEmail ? ( + <> +
+ + {children} + + + ) : ( + + )} + ); diff --git a/app/providers.tsx b/app/providers.tsx new file mode 100644 index 000000000..2677c5400 --- /dev/null +++ b/app/providers.tsx @@ -0,0 +1,35 @@ +// app/providers.tsx +"use client"; +import posthog from "posthog-js"; +import { PostHogProvider } from "posthog-js/react"; +import { usePathname, useSearchParams } from "next/navigation"; +import { useEffect } from "react"; + +if (typeof window !== "undefined") { + posthog.init(process.env.NEXT_PUBLIC_POSTHOG_KEY!, { + api_host: process.env.NEXT_PUBLIC_POSTHOG_HOST, + }); +} + +export function PostHogPageview(): JSX.Element { + const pathname = usePathname(); + const searchParams = useSearchParams(); + + useEffect(() => { + if (pathname) { + let url = window.origin + pathname; + if (searchParams && searchParams.toString()) { + url = url + `?${searchParams.toString()}`; + } + posthog.capture("$pageview", { + $current_url: url, + }); + } + }, [pathname, searchParams]); + + return <>; +} + +export function PHProvider({ children }: { children: React.ReactNode }) { + return {children}; +} diff --git a/pages/api/extension/getContext.ts b/pages/api/extension/getContext.ts index c5c35c357..6ac0dbb69 100644 --- a/pages/api/extension/getContext.ts +++ b/pages/api/extension/getContext.ts @@ -4,6 +4,7 @@ import getUserTokens from "../../../utils/db/user/getUserTokens"; import updateTokensFromJira from "../../../utils/jira/updateTokens"; import updateTokens from "../../../utils/db/jira/updateTokens"; import searchMessageByText from "../../../utils/slack/searchMessageByText"; +import validateParams from "../../../utils/api/validateParams"; function replaceSpecialChars(inputString) { const specialChars = /[!"#$%&/()=?_"{}¨*]/g; // Edit this list to include or exclude characters return inputString.toLowerCase().replace(specialChars, " "); @@ -155,21 +156,20 @@ export default async function handler(req, res) { properties: { user, repo, owner, gitSystem }, }); - if (!user) { - return res.send({ error: "no user" }); - } - if (!repo) { - return res.send({ error: "no repo" }); - } - if (!owner) { - return res.send({ error: "no owner" }); - } - if (!commitList) { - return res.send({ error: "no commitList" }); - } - if (!gitSystem) { - return res.send({ error: "no gitSystem" }); + const { missingParams } = validateParams(req.body, [ + "user", + "repo", + "owner", + "gitSystem", + "commitList", + ]); + + if (missingParams.length > 0) { + return res.json({ + error: `Missing parameters: ${missingParams.join(", ")}`, + }); } + let userTokens; try { userTokens = await getUserTokens({ email: user }); diff --git a/pages/api/hover/getHoverData.ts b/pages/api/hover/getHoverData.ts index e264363ef..bd824c209 100644 --- a/pages/api/hover/getHoverData.ts +++ b/pages/api/hover/getHoverData.ts @@ -4,6 +4,7 @@ import { trackEvent } from "../../../utils/analytics/azureAppInsights"; import updateTokensFromJira from "../../../utils/jira/updateTokens"; import updateTokens from "../../../utils/db/jira/updateTokens"; import searchMessageByText from "../../../utils/slack/searchMessageByText"; +import validateParams from "../../../utils/api/validateParams"; function replaceSpecialChars(inputString) { const specialChars = /[!"#$%&/()=?_"{}¨*]/g; // Edit this list to include or exclude characters return inputString.toLowerCase().replace(specialChars, " "); @@ -14,21 +15,18 @@ function handleRejection(reason) { } export default async function handler(req, res) { const { user, gitSystem, repo, owner, commitTitle } = req.body; + const { missingParams } = validateParams(req.body, [ + "user", + "repo", + "owner", + "gitSystem", + "commitTitle", + ]); - if (!user) { - return res.send({ error: "no user" }); - } - if (!repo) { - return res.send({ error: "no repo" }); - } - if (!owner) { - return res.send({ error: "no owner" }); - } - if (!gitSystem) { - return res.send({ error: "no gitSystem" }); - } - if (!commitTitle) { - return res.send({ error: "no commitTitle" }); + if (missingParams.length > 0) { + return res.json({ + error: `Missing parameters: ${missingParams.join(", ")}`, + }); } let userTokens; diff --git a/pages/api/sendgrid/sendTeammateInvite.ts b/pages/api/sendgrid/sendTeammateInvite.ts deleted file mode 100644 index feeb23ccb..000000000 --- a/pages/api/sendgrid/sendTeammateInvite.ts +++ /dev/null @@ -1,17 +0,0 @@ -import sendTeammateInvite from "../../../utils/sendgrid/sendTeammateInvite"; -export default async function handler(req, res) { - let { sender, email, inviteUrl, teamName } = req.body; - if (!email) { - return res.send({ error: "no email" }); - } - if (!sender) { - return res.send({ error: "no sender" }); - } - let emailSent = await sendTeammateInvite({ - sender, - email, - inviteUrl, - teamName, - }); - return res.send(emailSent); -} diff --git a/pages/api/sendgrid/sendWelcome.ts b/pages/api/sendgrid/sendWelcome.ts deleted file mode 100644 index fde808a43..000000000 --- a/pages/api/sendgrid/sendWelcome.ts +++ /dev/null @@ -1,12 +0,0 @@ -import sendWelcome from "../../../utils/sendgrid/sendWelcome"; -export default async function handler(req, res) { - let { emails, sender } = req.body; - if (!emails) { - return res.send({ error: "no email" }); - } - if (!sender) { - return res.send({ error: "no sender" }); - } - let emailSent = await sendWelcome({ sender, emails }); - return res.send(emailSent); -} diff --git a/utils/sendgrid/sendTeammateInvite.ts b/utils/sendgrid/sendTeammateInvite.ts index 330c64fc9..de41fe871 100644 --- a/utils/sendgrid/sendTeammateInvite.ts +++ b/utils/sendgrid/sendTeammateInvite.ts @@ -1,10 +1,10 @@ export default async function sendTeammateInvite({ - email, + teammateEmail, sender, inviteUrl, teamName, }: { - email: string; + teammateEmail: string; sender: string; inviteUrl: string; teamName: string; @@ -12,7 +12,7 @@ export default async function sendTeammateInvite({ const sgMail = require("@sendgrid/mail"); sgMail.setApiKey(process.env.SENDGRID_API_KEY); const msg = { - to: email, + to: teammateEmail, from: "info@watermelon.tools", templateId: "d-dd5c729f0be5439daac6b1faaf0431d6", dynamic_template_data: { From 4e720635700b8f4d4ec5b8d154abc6f9e28d9203 Mon Sep 17 00:00:00 2001 From: Esteban Dalel R Date: Mon, 14 Aug 2023 15:37:00 -0500 Subject: [PATCH 2/2] Feature/posthog backend (#243) * Create param validator * Move discord * Move user settings getter to app router * Use standard var * move getAllPublicUserData to app router * Better error handling * Delete getAllData.ts * Delete discord * Throw errors * Handle error throwing * Move updateSettings to app router * Update confluence.svg * Configure jest * Create settings.test.ts * Create updateSettings.test.ts * Add getting test * Use generalized validator * Move Stripe to app router * Move Sendgrid to app router * Move vscode to airtable Analytics to app router * Create posthog.ts * Add types, move to utils * Capture posthog event * Change export * Fix export * Fix export * Check dev env --- app/api/user/settings/route.ts | 9 ++++++++- utils/posthog/posthog.ts | 36 ++++++++++++++++++++++++++++++++++ 2 files changed, 44 insertions(+), 1 deletion(-) create mode 100644 utils/posthog/posthog.ts diff --git a/app/api/user/settings/route.ts b/app/api/user/settings/route.ts index d9db2ea32..a0f47e8d1 100644 --- a/app/api/user/settings/route.ts +++ b/app/api/user/settings/route.ts @@ -1,6 +1,7 @@ import { NextResponse } from "next/server"; import getUserSettings from "../../../../utils/db/user/settings"; import validateParams from "../../../../utils/api/validateParams"; +import posthog from "../../../../utils/posthog/posthog"; export async function POST(request: Request) { const req = await request.json(); @@ -11,7 +12,13 @@ export async function POST(request: Request) { error: `Missing parameters: ${missingParams.join(", ")}`, }); } - + posthog.capture({ + distinctId: req.email, + event: "user_settings_viewed", + properties: { + email: req.email, + }, + }); try { let dbResponse = await getUserSettings({ email: req.email }); return NextResponse.json(dbResponse); diff --git a/utils/posthog/posthog.ts b/utils/posthog/posthog.ts new file mode 100644 index 000000000..72c9b750d --- /dev/null +++ b/utils/posthog/posthog.ts @@ -0,0 +1,36 @@ +import { PostHog } from "posthog-node"; +interface PostHogEvent { + event: string; + distinctId?: string; + properties?: Record; + groups?: Record; +} + +function PostHogClient(apiKey: string) { + const posthogClient = new PostHog(apiKey, { + host: process.env.NEXT_PUBLIC_POSTHOG_HOST, + }); + + return { + capture: ({ event, distinctId, properties, groups }: PostHogEvent) => { + posthogClient.capture({ + distinctId: distinctId || "unknown_user", + event, + properties, + groups, + }); + console.log("posthog event", event, properties); + if (process.env.NODE_ENV === "development") { + console.log( + `PostHog event: ${event} with properties: ${JSON.stringify( + properties + )}` + ); + } + }, + }; +} + +const posthog = PostHogClient(process.env.NEXT_PUBLIC_POSTHOG_KEY!); + +export default posthog;