mirror of
https://github.com/LukeHagar/baton.git
synced 2025-12-06 04:19:20 +00:00
Saving initial state
This commit is contained in:
21
.gitignore
vendored
Normal file
21
.gitignore
vendored
Normal file
@@ -0,0 +1,21 @@
|
||||
node_modules
|
||||
|
||||
# Output
|
||||
.output
|
||||
.vercel
|
||||
/.svelte-kit
|
||||
/build
|
||||
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Env
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
!.env.test
|
||||
|
||||
# Vite
|
||||
vite.config.js.timestamp-*
|
||||
vite.config.ts.timestamp-*
|
||||
4
.prettierignore
Normal file
4
.prettierignore
Normal file
@@ -0,0 +1,4 @@
|
||||
# Package Managers
|
||||
package-lock.json
|
||||
pnpm-lock.yaml
|
||||
yarn.lock
|
||||
8
.prettierrc
Normal file
8
.prettierrc
Normal file
@@ -0,0 +1,8 @@
|
||||
{
|
||||
"useTabs": true,
|
||||
"singleQuote": true,
|
||||
"trailingComma": "none",
|
||||
"printWidth": 100,
|
||||
"plugins": ["prettier-plugin-svelte"],
|
||||
"overrides": [{ "files": "*.svelte", "options": { "parser": "svelte" } }]
|
||||
}
|
||||
38
README.md
Normal file
38
README.md
Normal file
@@ -0,0 +1,38 @@
|
||||
# create-svelte
|
||||
|
||||
Everything you need to build a Svelte project, powered by [`create-svelte`](https://github.com/sveltejs/kit/tree/main/packages/create-svelte).
|
||||
|
||||
## Creating a project
|
||||
|
||||
If you're seeing this, you've probably already done this step. Congrats!
|
||||
|
||||
```bash
|
||||
# create a new project in the current directory
|
||||
npm create svelte@latest
|
||||
|
||||
# create a new project in my-app
|
||||
npm create svelte@latest my-app
|
||||
```
|
||||
|
||||
## Developing
|
||||
|
||||
Once you've created a project and installed dependencies with `npm install` (or `pnpm install` or `yarn`), start a development server:
|
||||
|
||||
```bash
|
||||
npm run dev
|
||||
|
||||
# or start the server and open the app in a new browser tab
|
||||
npm run dev -- --open
|
||||
```
|
||||
|
||||
## Building
|
||||
|
||||
To create a production version of your app:
|
||||
|
||||
```bash
|
||||
npm run build
|
||||
```
|
||||
|
||||
You can preview the production build with `npm run preview`.
|
||||
|
||||
> To deploy your app, you may need to install an [adapter](https://kit.svelte.dev/docs/adapters) for your target environment.
|
||||
32
eslint.config.js
Normal file
32
eslint.config.js
Normal file
@@ -0,0 +1,32 @@
|
||||
import eslint from '@eslint/js';
|
||||
import prettier from 'eslint-config-prettier';
|
||||
import svelte from 'eslint-plugin-svelte';
|
||||
import globals from 'globals';
|
||||
import tseslint from 'typescript-eslint';
|
||||
|
||||
export default tseslint.config(
|
||||
eslint.configs.recommended,
|
||||
...tseslint.configs.recommended,
|
||||
...svelte.configs['flat/recommended'],
|
||||
prettier,
|
||||
...svelte.configs['flat/prettier'],
|
||||
{
|
||||
languageOptions: {
|
||||
globals: {
|
||||
...globals.browser,
|
||||
...globals.node
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
files: ['**/*.svelte'],
|
||||
languageOptions: {
|
||||
parserOptions: {
|
||||
parser: tseslint.parser
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
ignores: ['build/', '.svelte-kit/', 'dist/']
|
||||
}
|
||||
);
|
||||
53
package.json
Normal file
53
package.json
Normal file
@@ -0,0 +1,53 @@
|
||||
{
|
||||
"name": "hooky",
|
||||
"version": "0.0.1",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"dev": "vite dev",
|
||||
"build": "vite build",
|
||||
"preview": "vite preview",
|
||||
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
|
||||
"check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch",
|
||||
"test": "vitest",
|
||||
"lint": "prettier --check . && eslint .",
|
||||
"format": "prettier --write ."
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ngrok/ngrok": "^1.4.1",
|
||||
"@skeletonlabs/skeleton": "^2.10.2",
|
||||
"@skeletonlabs/tw-plugin": "^0.4.0",
|
||||
"@sveltejs/adapter-auto": "^3.0.0",
|
||||
"@sveltejs/kit": "^2.0.0",
|
||||
"@sveltejs/vite-plugin-svelte": "^3.0.0",
|
||||
"@types/eslint": "^9.6.0",
|
||||
"@types/node": "^22.7.5",
|
||||
"autoprefixer": "^10.4.20",
|
||||
"axios": "^1.7.7",
|
||||
"eslint": "^9.0.0",
|
||||
"eslint-config-prettier": "^9.1.0",
|
||||
"eslint-plugin-svelte": "^2.36.0",
|
||||
"globals": "^15.0.0",
|
||||
"highlight.js": "^11.10.0",
|
||||
"paneforge": "^0.0.6",
|
||||
"postcss": "^8.4.47",
|
||||
"prettier": "^3.1.1",
|
||||
"prettier-plugin-svelte": "^3.1.2",
|
||||
"prisma": "^5.20.0",
|
||||
"svelte": "^4.2.7",
|
||||
"svelte-check": "^4.0.0",
|
||||
"svelte-french-toast": "^1.2.0",
|
||||
"svelte-highlight": "^7.7.0",
|
||||
"svelte-persisted-store": "^0.11.0",
|
||||
"tailwindcss": "^3.4.13",
|
||||
"typescript": "^5.0.0",
|
||||
"typescript-eslint": "^8.0.0",
|
||||
"vite": "^5.0.3",
|
||||
"vitest": "^2.0.0"
|
||||
},
|
||||
"type": "module",
|
||||
"packageManager": "yarn@1.22.22+sha512.a6b2f7906b721bba3d67d4aff083df04dad64c399707841b7acf00f6b133b7ac24255f2652fa22ae3534329dc6180534e98d17432037ff6fd140556e2bb3137e",
|
||||
"dependencies": {
|
||||
"@prisma/client": "5.20.0",
|
||||
"lucide-svelte": "^0.453.0"
|
||||
}
|
||||
}
|
||||
6
postcss.config.js
Normal file
6
postcss.config.js
Normal file
@@ -0,0 +1,6 @@
|
||||
export default {
|
||||
plugins: {
|
||||
tailwindcss: {},
|
||||
autoprefixer: {},
|
||||
},
|
||||
}
|
||||
BIN
prisma/dev.db
Normal file
BIN
prisma/dev.db
Normal file
Binary file not shown.
BIN
prisma/dev.db-journal
Normal file
BIN
prisma/dev.db-journal
Normal file
Binary file not shown.
17
prisma/migrations/20241014010748_init/migration.sql
Normal file
17
prisma/migrations/20241014010748_init/migration.sql
Normal file
@@ -0,0 +1,17 @@
|
||||
-- CreateTable
|
||||
CREATE TABLE "User" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"email" TEXT NOT NULL,
|
||||
"name" TEXT
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "Webhook" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"content" TEXT,
|
||||
"ownerId" INTEGER NOT NULL,
|
||||
CONSTRAINT "Webhook_ownerId_fkey" FOREIGN KEY ("ownerId") REFERENCES "User" ("id") ON DELETE RESTRICT ON UPDATE CASCADE
|
||||
);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE UNIQUE INDEX "User_email_key" ON "User"("email");
|
||||
32
prisma/migrations/20241014020434_init/migration.sql
Normal file
32
prisma/migrations/20241014020434_init/migration.sql
Normal file
@@ -0,0 +1,32 @@
|
||||
/*
|
||||
Warnings:
|
||||
|
||||
- You are about to drop the `User` table. If the table is not empty, all the data it contains will be lost.
|
||||
- You are about to drop the column `content` on the `Webhook` table. All the data in the column will be lost.
|
||||
- You are about to drop the column `ownerId` on the `Webhook` table. All the data in the column will be lost.
|
||||
|
||||
*/
|
||||
-- DropIndex
|
||||
DROP INDEX "User_email_key";
|
||||
|
||||
-- DropTable
|
||||
PRAGMA foreign_keys=off;
|
||||
DROP TABLE "User";
|
||||
PRAGMA foreign_keys=on;
|
||||
|
||||
-- RedefineTables
|
||||
PRAGMA defer_foreign_keys=ON;
|
||||
PRAGMA foreign_keys=OFF;
|
||||
CREATE TABLE "new_Webhook" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"headers" TEXT,
|
||||
"method" TEXT,
|
||||
"path" TEXT,
|
||||
"body" TEXT,
|
||||
"createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
INSERT INTO "new_Webhook" ("id") SELECT "id" FROM "Webhook";
|
||||
DROP TABLE "Webhook";
|
||||
ALTER TABLE "new_Webhook" RENAME TO "Webhook";
|
||||
PRAGMA foreign_keys=ON;
|
||||
PRAGMA defer_foreign_keys=OFF;
|
||||
2
prisma/migrations/20241014140859_init/migration.sql
Normal file
2
prisma/migrations/20241014140859_init/migration.sql
Normal file
@@ -0,0 +1,2 @@
|
||||
-- AlterTable
|
||||
ALTER TABLE "Webhook" ADD COLUMN "query" TEXT;
|
||||
27
prisma/migrations/20241014191103_init/migration.sql
Normal file
27
prisma/migrations/20241014191103_init/migration.sql
Normal file
@@ -0,0 +1,27 @@
|
||||
/*
|
||||
Warnings:
|
||||
|
||||
- Made the column `body` on table `Webhook` required. This step will fail if there are existing NULL values in that column.
|
||||
- Made the column `headers` on table `Webhook` required. This step will fail if there are existing NULL values in that column.
|
||||
- Made the column `method` on table `Webhook` required. This step will fail if there are existing NULL values in that column.
|
||||
- Made the column `path` on table `Webhook` required. This step will fail if there are existing NULL values in that column.
|
||||
- Made the column `query` on table `Webhook` required. This step will fail if there are existing NULL values in that column.
|
||||
|
||||
*/
|
||||
-- RedefineTables
|
||||
PRAGMA defer_foreign_keys=ON;
|
||||
PRAGMA foreign_keys=OFF;
|
||||
CREATE TABLE "new_Webhook" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"headers" TEXT NOT NULL,
|
||||
"method" TEXT NOT NULL,
|
||||
"path" TEXT NOT NULL,
|
||||
"body" TEXT NOT NULL,
|
||||
"query" TEXT NOT NULL,
|
||||
"createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
INSERT INTO "new_Webhook" ("body", "createdAt", "headers", "id", "method", "path", "query") SELECT "body", "createdAt", "headers", "id", "method", "path", "query" FROM "Webhook";
|
||||
DROP TABLE "Webhook";
|
||||
ALTER TABLE "new_Webhook" RENAME TO "Webhook";
|
||||
PRAGMA foreign_keys=ON;
|
||||
PRAGMA defer_foreign_keys=OFF;
|
||||
7
prisma/migrations/20241017030724_init/migration.sql
Normal file
7
prisma/migrations/20241017030724_init/migration.sql
Normal file
@@ -0,0 +1,7 @@
|
||||
-- CreateTable
|
||||
CREATE TABLE "RelayTarget" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"name" TEXT NOT NULL,
|
||||
"url" TEXT NOT NULL,
|
||||
"createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
22
prisma/migrations/20241017143005_init/migration.sql
Normal file
22
prisma/migrations/20241017143005_init/migration.sql
Normal file
@@ -0,0 +1,22 @@
|
||||
/*
|
||||
Warnings:
|
||||
|
||||
- You are about to drop the column `query` on the `Webhook` table. All the data in the column will be lost.
|
||||
|
||||
*/
|
||||
-- RedefineTables
|
||||
PRAGMA defer_foreign_keys=ON;
|
||||
PRAGMA foreign_keys=OFF;
|
||||
CREATE TABLE "new_Webhook" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"headers" TEXT NOT NULL,
|
||||
"method" TEXT NOT NULL,
|
||||
"path" TEXT NOT NULL,
|
||||
"body" TEXT NOT NULL,
|
||||
"createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
INSERT INTO "new_Webhook" ("body", "createdAt", "headers", "id", "method", "path") SELECT "body", "createdAt", "headers", "id", "method", "path" FROM "Webhook";
|
||||
DROP TABLE "Webhook";
|
||||
ALTER TABLE "new_Webhook" RENAME TO "Webhook";
|
||||
PRAGMA foreign_keys=ON;
|
||||
PRAGMA defer_foreign_keys=OFF;
|
||||
15
prisma/migrations/20241018214949_init/migration.sql
Normal file
15
prisma/migrations/20241018214949_init/migration.sql
Normal file
@@ -0,0 +1,15 @@
|
||||
-- RedefineTables
|
||||
PRAGMA defer_foreign_keys=ON;
|
||||
PRAGMA foreign_keys=OFF;
|
||||
CREATE TABLE "new_RelayTarget" (
|
||||
"id" INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||
"name" TEXT NOT NULL,
|
||||
"url" TEXT NOT NULL,
|
||||
"active" BOOLEAN NOT NULL DEFAULT true,
|
||||
"createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
INSERT INTO "new_RelayTarget" ("createdAt", "id", "name", "url") SELECT "createdAt", "id", "name", "url" FROM "RelayTarget";
|
||||
DROP TABLE "RelayTarget";
|
||||
ALTER TABLE "new_RelayTarget" RENAME TO "RelayTarget";
|
||||
PRAGMA foreign_keys=ON;
|
||||
PRAGMA defer_foreign_keys=OFF;
|
||||
3
prisma/migrations/migration_lock.toml
Normal file
3
prisma/migrations/migration_lock.toml
Normal file
@@ -0,0 +1,3 @@
|
||||
# Please do not edit this file manually
|
||||
# It should be added in your version-control system (i.e. Git)
|
||||
provider = "sqlite"
|
||||
28
prisma/schema.prisma
Normal file
28
prisma/schema.prisma
Normal file
@@ -0,0 +1,28 @@
|
||||
// This is your Prisma schema file,
|
||||
// learn more about it in the docs: https://pris.ly/d/prisma-schema
|
||||
|
||||
generator client {
|
||||
provider = "prisma-client-js"
|
||||
}
|
||||
|
||||
datasource db {
|
||||
provider = "sqlite"
|
||||
url = env("DATABASE_URL")
|
||||
}
|
||||
|
||||
model Webhook {
|
||||
id Int @id @default(autoincrement())
|
||||
headers String
|
||||
method String
|
||||
path String
|
||||
body String
|
||||
createdAt DateTime @default(now())
|
||||
}
|
||||
|
||||
model RelayTarget {
|
||||
id Int @id @default(autoincrement())
|
||||
name String
|
||||
url String
|
||||
active Boolean @default(true)
|
||||
createdAt DateTime @default(now())
|
||||
}
|
||||
25
script.ts
Normal file
25
script.ts
Normal file
@@ -0,0 +1,25 @@
|
||||
import { PrismaClient } from '@prisma/client'
|
||||
|
||||
const prisma = new PrismaClient()
|
||||
|
||||
async function main() {
|
||||
const user = await prisma.webhook.create({
|
||||
data: {
|
||||
headers: '{"Content-Type":"application/json"}',
|
||||
method: 'POST',
|
||||
path: '/',
|
||||
body: '{"message":"Hello, world!"}'
|
||||
}
|
||||
})
|
||||
console.log(user)
|
||||
}
|
||||
|
||||
main()
|
||||
.then(async () => {
|
||||
await prisma.$disconnect()
|
||||
})
|
||||
.catch(async (e) => {
|
||||
console.error(e)
|
||||
await prisma.$disconnect()
|
||||
process.exit(1)
|
||||
})
|
||||
3
src/app.css
Normal file
3
src/app.css
Normal file
@@ -0,0 +1,3 @@
|
||||
@tailwind base;
|
||||
@tailwind components;
|
||||
@tailwind utilities;
|
||||
57
src/app.d.ts
vendored
Normal file
57
src/app.d.ts
vendored
Normal file
@@ -0,0 +1,57 @@
|
||||
// See https://kit.svelte.dev/docs/types#app
|
||||
// for information about these interfaces
|
||||
declare global {
|
||||
namespace App {
|
||||
|
||||
/**
|
||||
* Represents an error according to the RFC 7807 `application/problem+json` standard.
|
||||
* @see https://datatracker.ietf.org/doc/html/rfc7807
|
||||
*/
|
||||
interface Error {
|
||||
/**
|
||||
* A URI reference that identifies the problem type.
|
||||
* This URI should ideally point to human-readable documentation.
|
||||
* @example "https://example.com/probs/out-of-stock"
|
||||
*/
|
||||
type?: string;
|
||||
|
||||
/**
|
||||
* A short, human-readable summary of the problem.
|
||||
* @example "Out of Stock"
|
||||
*/
|
||||
title?: string;
|
||||
|
||||
/**
|
||||
* The HTTP status code associated with the problem.
|
||||
* @example 404
|
||||
*/
|
||||
status?: number;
|
||||
|
||||
/**
|
||||
* A detailed, human-readable explanation of the problem.
|
||||
* @example "The requested item is no longer available."
|
||||
*/
|
||||
detail?: string;
|
||||
|
||||
/**
|
||||
* A URI reference that identifies the specific occurrence of the problem.
|
||||
* This can be used for tracking and debugging.
|
||||
* @example "/products/12345"
|
||||
*/
|
||||
instance?: string;
|
||||
|
||||
/**
|
||||
* Additional fields providing extra information about the problem.
|
||||
* @remarks Use `unknown` to ensure type safety when accessing these fields.
|
||||
*/
|
||||
[key: string]: unknown;
|
||||
}
|
||||
|
||||
// interface Locals {}
|
||||
// interface PageData {}
|
||||
// interface PageState {}
|
||||
// interface Platform {}
|
||||
}
|
||||
}
|
||||
|
||||
export {};
|
||||
12
src/app.html
Normal file
12
src/app.html
Normal file
@@ -0,0 +1,12 @@
|
||||
<!doctype html>
|
||||
<html lang="en" dark>
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<link rel="icon" href="%sveltekit.assets%/favicon.png" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
%sveltekit.head%
|
||||
</head>
|
||||
<body data-sveltekit-preload-data="hover" data-theme="wintry">
|
||||
<div style="display: contents">%sveltekit.body%</div>
|
||||
</body>
|
||||
</html>
|
||||
7
src/index.test.ts
Normal file
7
src/index.test.ts
Normal file
@@ -0,0 +1,7 @@
|
||||
import { describe, it, expect } from 'vitest';
|
||||
|
||||
describe('sum test', () => {
|
||||
it('adds 1 + 2 to equal 3', () => {
|
||||
expect(1 + 2).toBe(3);
|
||||
});
|
||||
});
|
||||
46
src/lib/client/targets.ts
Normal file
46
src/lib/client/targets.ts
Normal file
@@ -0,0 +1,46 @@
|
||||
import axios from 'axios';
|
||||
import toast from 'svelte-french-toast';
|
||||
|
||||
export function deleteTarget(id: number) {
|
||||
toast.promise(
|
||||
axios.delete(`/targets/${id}`)
|
||||
.then(response => {
|
||||
if (response.status >= 200 && response.status < 300) {
|
||||
return 'Target deleted successfully';
|
||||
} else {
|
||||
throw new Error(`Failed to delete Target: ${response.statusText}`);
|
||||
}
|
||||
})
|
||||
.catch(error => {
|
||||
throw new Error(`Failed to delete Target: ${error?.response?.data?.message || error.message}`);
|
||||
}),
|
||||
{
|
||||
loading: 'Deleting Target...',
|
||||
success: 'Target deleted successfully',
|
||||
error: (error) => `Failed to delete Target: ${error.message}`
|
||||
},
|
||||
{ position: 'bottom-center' }
|
||||
);
|
||||
}
|
||||
|
||||
export function createTarget(name: string, url: string) {
|
||||
toast.promise(
|
||||
axios.post(`/targets`, { name, url })
|
||||
.then(response => {
|
||||
if (response.status >= 200 && response.status < 300) {
|
||||
return 'Target created successfully';
|
||||
} else {
|
||||
throw new Error(response.statusText);
|
||||
}
|
||||
})
|
||||
.catch(error => {
|
||||
throw new Error(error?.response?.data?.message || error.message);
|
||||
}),
|
||||
{
|
||||
loading: 'Creating Target...',
|
||||
success: 'Target created successfully',
|
||||
error: (error) => `Failed to create Target: ${error.message}`
|
||||
},
|
||||
{ position: 'bottom-center' }
|
||||
);
|
||||
}
|
||||
89
src/lib/client/webhooks.ts
Normal file
89
src/lib/client/webhooks.ts
Normal file
@@ -0,0 +1,89 @@
|
||||
import axios from 'axios';
|
||||
import type { Webhook } from '@prisma/client';
|
||||
import toast from 'svelte-french-toast';
|
||||
|
||||
export async function checkForNewWebhooks() {
|
||||
return await axios
|
||||
.head('/webhooks')
|
||||
.then((response) => {
|
||||
if (response.status == 200) {
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
})
|
||||
.catch((error) => {
|
||||
throw new Error(error?.response?.data?.message || error.message);
|
||||
});
|
||||
}
|
||||
|
||||
export function getWebhooks() {
|
||||
return toast.promise<Webhook[]>(
|
||||
axios
|
||||
.get('/webhooks')
|
||||
.then((response) => {
|
||||
if (response.status >= 200 && response.status < 300) {
|
||||
return response.data;
|
||||
} else {
|
||||
throw new Error(response.statusText);
|
||||
}
|
||||
})
|
||||
.catch((error) => {
|
||||
throw new Error(error?.response?.data?.message || error.message);
|
||||
}),
|
||||
{
|
||||
loading: 'Fetching webhooks...',
|
||||
success: 'updated Webhooks',
|
||||
error: (error) => `Failed to fetch webhooks: ${error.message}`
|
||||
},
|
||||
{ position: 'bottom-center' }
|
||||
);
|
||||
}
|
||||
|
||||
export function deleteWebhook(id: number) {
|
||||
toast.promise(
|
||||
axios
|
||||
.delete(`/webhooks/${id}`)
|
||||
.then((response) => {
|
||||
if (response.status >= 200 && response.status < 300) {
|
||||
return 'Webhook deleted successfully';
|
||||
} else {
|
||||
throw new Error(`Failed to delete webhook: ${response.statusText}`);
|
||||
}
|
||||
})
|
||||
.catch((error) => {
|
||||
throw new Error(
|
||||
`Failed to delete webhook: ${error?.response?.data?.message || error.message}`
|
||||
);
|
||||
}),
|
||||
{
|
||||
loading: 'Deleting webhook...',
|
||||
success: 'Webhook deleted successfully',
|
||||
error: (error) => `Failed to delete webhook: ${error.message}`
|
||||
},
|
||||
{ position: 'bottom-center' }
|
||||
);
|
||||
}
|
||||
|
||||
export function sendWebhook(webhookTarget: string, id: number) {
|
||||
toast.promise(
|
||||
axios
|
||||
.post(`/webhooks/${id}?webhookTarget=${webhookTarget}`)
|
||||
.then((response) => {
|
||||
if (response.status >= 200 && response.status < 300) {
|
||||
return 'Webhook sent successfully';
|
||||
} else {
|
||||
throw new Error(response.statusText);
|
||||
}
|
||||
})
|
||||
.catch((error) => {
|
||||
throw new Error(error?.response?.data?.message || error.message);
|
||||
}),
|
||||
{
|
||||
loading: 'Sending webhook...',
|
||||
success: 'Webhook sent successfully',
|
||||
error: (error) => `Failed to send webhook: ${error.message}`
|
||||
},
|
||||
{ position: 'bottom-center' }
|
||||
);
|
||||
}
|
||||
123
src/lib/components/AddUpdateModal.svelte
Normal file
123
src/lib/components/AddUpdateModal.svelte
Normal file
@@ -0,0 +1,123 @@
|
||||
<script lang="ts">
|
||||
import type { RelayTarget } from '@prisma/client';
|
||||
import { createEventDispatcher, onMount } from 'svelte';
|
||||
import { fade, fly } from 'svelte/transition';
|
||||
|
||||
export let target: RelayTarget | null = null;
|
||||
|
||||
const dispatch = createEventDispatcher();
|
||||
|
||||
let dialog: HTMLDialogElement;
|
||||
let nickname = '';
|
||||
let url = '';
|
||||
let method = 'POST';
|
||||
let headers = '';
|
||||
let isActive = true;
|
||||
|
||||
$: isEditMode = !!target;
|
||||
|
||||
$: if (target) {
|
||||
nickname = target.name;
|
||||
url = target.url;
|
||||
method = 'Post';
|
||||
headers = `{"Content-Type": "application/json"}`;
|
||||
isActive = true;
|
||||
}
|
||||
|
||||
onMount(() => {
|
||||
dialog.addEventListener('close', () => {
|
||||
resetForm();
|
||||
});
|
||||
});
|
||||
|
||||
function openModal() {
|
||||
dialog.showModal();
|
||||
}
|
||||
|
||||
function handleSubmit() {
|
||||
if (!target) {
|
||||
return;
|
||||
}
|
||||
const parsedHeaders = JSON.parse(headers);
|
||||
const payload = {
|
||||
nickname,
|
||||
url,
|
||||
method,
|
||||
headers: parsedHeaders,
|
||||
isActive
|
||||
};
|
||||
|
||||
if (isEditMode) {
|
||||
dispatch('update', { id: target.id, ...payload });
|
||||
} else {
|
||||
dispatch('create', payload);
|
||||
}
|
||||
|
||||
closeModal();
|
||||
}
|
||||
|
||||
function closeModal() {
|
||||
dialog.close();
|
||||
}
|
||||
|
||||
function resetForm() {
|
||||
if (!isEditMode) {
|
||||
nickname = '';
|
||||
url = '';
|
||||
method = 'POST';
|
||||
headers = '';
|
||||
isActive = true;
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<button on:click={openModal} class="btn variant-outline btn-sm">
|
||||
{isEditMode ? 'Update' : 'Add'}
|
||||
</button>
|
||||
|
||||
<div>
|
||||
<dialog bind:this={dialog} class="card text-on-surface-token">
|
||||
<div class="p-6" transition:fly={{ y: 20, duration: 300 }}>
|
||||
<h2 class="text-2xl font-bold mb-4">
|
||||
{isEditMode ? 'Update' : 'Create'} Webhook Relay Target
|
||||
</h2>
|
||||
<form on:submit|preventDefault={handleSubmit} class="space-y-4">
|
||||
<div>
|
||||
<label for="nickname" class="label">Nickname</label>
|
||||
<input class="input" type="text" id="nickname" bind:value={nickname} required />
|
||||
</div>
|
||||
<div>
|
||||
<label for="url" class="label">URL</label>
|
||||
<input type="url" id="url" bind:value={url} class="input" required />
|
||||
</div>
|
||||
|
||||
<label class="flex items-center space-x-2">
|
||||
<input class="checkbox" type="checkbox" checked />
|
||||
<p>Active</p>
|
||||
</label>
|
||||
|
||||
<div class="flex justify-end space-x-3">
|
||||
<button
|
||||
type="button"
|
||||
on:click={closeModal}
|
||||
class="px-4 py-2 border rounded-md text-sm font-medium hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"
|
||||
>
|
||||
Cancel
|
||||
</button>
|
||||
<button
|
||||
on:click={handleSubmit}
|
||||
class="px-4 py-2 border border-transparent rounded-md shadow-sm text-sm font-medium text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"
|
||||
>
|
||||
{isEditMode ? 'Update' : 'Create'}
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</dialog>
|
||||
</div>
|
||||
|
||||
<style>
|
||||
dialog::backdrop {
|
||||
background: rgba(0, 0, 0, 0.3);
|
||||
}
|
||||
</style>
|
||||
49
src/lib/components/CodeBlock.svelte
Normal file
49
src/lib/components/CodeBlock.svelte
Normal file
@@ -0,0 +1,49 @@
|
||||
<script lang="ts">
|
||||
import Highlight from 'svelte-highlight';
|
||||
import json from 'svelte-highlight/languages/json';
|
||||
import { modeCurrent } from '@skeletonlabs/skeleton';
|
||||
import { github, githubDark } from 'svelte-highlight/styles';
|
||||
import type { LanguageType } from 'svelte-highlight/languages';
|
||||
|
||||
export let code = '';
|
||||
export let language: LanguageType<string> = json
|
||||
export let label = '';
|
||||
|
||||
let header: HTMLElement
|
||||
|
||||
$: console.log($modeCurrent);
|
||||
</script>
|
||||
|
||||
<svelte:head>
|
||||
{#if $modeCurrent == true}
|
||||
{@html github}
|
||||
{:else}
|
||||
{@html githubDark}
|
||||
{/if}
|
||||
</svelte:head>
|
||||
|
||||
<div class="card w-full h-full overflow-hidden ">
|
||||
<header bind:this={header} class=" flex justify-between items-center p-2 bg-surface-200-700-token">
|
||||
<div class="flex items-center space-x-2 ">
|
||||
{#if label}
|
||||
<span class="badge variant-ghost">{label}</span>
|
||||
{/if}
|
||||
<!-- <span class="badge variant-ghost">{language.name.toUpperCase()}</span> -->
|
||||
</div>
|
||||
<button class="btn btn-sm variant-ghost" on:click={() => navigator.clipboard.writeText(code)}>
|
||||
Copy
|
||||
</button>
|
||||
</header>
|
||||
<section class="p-4 pb-14 h-full overflow-y-scroll overflow-x-auto">
|
||||
<Highlight {language} code={code} />
|
||||
<!-- <pre class="text-sm"><code class="hljs {language} font-mono">{@html highlighted}</code></pre> -->
|
||||
</section>
|
||||
</div>
|
||||
|
||||
<style>
|
||||
/* Ensure Highlight.js styles don't conflict with Skeleton */
|
||||
:global(.hljs) {
|
||||
background: transparent !important;
|
||||
padding: 0 !important;
|
||||
}
|
||||
</style>
|
||||
32
src/lib/components/RelayControl.svelte
Normal file
32
src/lib/components/RelayControl.svelte
Normal file
@@ -0,0 +1,32 @@
|
||||
<script lang="ts">
|
||||
import type { Writable } from "svelte/store";
|
||||
import AddUpdateModal from "./AddUpdateModal.svelte";
|
||||
|
||||
export let webhookTarget: Writable<string>
|
||||
export let intervalInSeconds: Writable<number>
|
||||
</script>
|
||||
|
||||
<div class="card p-2">
|
||||
<!-- Form for custom webhook URL and interval -->
|
||||
<div class="flex flex-row items-end space-x-2">
|
||||
<div class="flex-1">
|
||||
<label for="custom-url" class="sr-only">Custom Webhook URL</label>
|
||||
<input
|
||||
id="custom-url"
|
||||
class="w-full input text-sm"
|
||||
placeholder="Enter webhook URL"
|
||||
name="webhookTarget"
|
||||
bind:value={$webhookTarget}
|
||||
/>
|
||||
</div>
|
||||
<AddUpdateModal/>
|
||||
<!-- <select bind:value={$intervalInSeconds} class="select text-sm w-[100px]">
|
||||
<option value={1}>1s</option>
|
||||
<option value={10}>10s</option>
|
||||
<option value={30}>30s</option>
|
||||
<option value={60}>1m</option>
|
||||
<option value={300}>5m</option>
|
||||
<option value={600}>10m</option>
|
||||
</select> -->
|
||||
</div>
|
||||
</div>
|
||||
59
src/lib/components/WebhookInspect.svelte
Normal file
59
src/lib/components/WebhookInspect.svelte
Normal file
@@ -0,0 +1,59 @@
|
||||
<script lang="ts">
|
||||
import CodeBlock from '$lib/components/CodeBlock.svelte';
|
||||
import { RadioGroup, RadioItem } from '@skeletonlabs/skeleton';
|
||||
import { PaneGroup, Pane, PaneResizer } from 'paneforge';
|
||||
import type { Webhook } from '@prisma/client';
|
||||
import { type Writable } from 'svelte/store';
|
||||
import { GripVertical, GripHorizontal } from 'lucide-svelte';
|
||||
|
||||
export let selectedWebhook: Writable<Webhook | undefined>;
|
||||
|
||||
let showHeaders = true;
|
||||
let showBody = true;
|
||||
let value: number = 0;
|
||||
|
||||
$: console.log(value);
|
||||
</script>
|
||||
|
||||
{#if $selectedWebhook}
|
||||
|
||||
<div class="flex justify-end gap-2 items-center ">
|
||||
<!-- <RadioGroup>
|
||||
<RadioItem bind:group={value} name="justify" value={0}>(label)</RadioItem>
|
||||
<RadioItem bind:group={value} name="justify" value={1}>(label)</RadioItem>
|
||||
<RadioItem bind:group={value} name="justify" value={2}>(label)</RadioItem>
|
||||
</RadioGroup>
|
||||
<button class="btn btn-sm variant-outline" on:click={() => (showHeaders = !showHeaders)}>
|
||||
{(showHeaders ? 'Hide' : 'Show') + ' Headers'}
|
||||
</button>
|
||||
<button class="btn btn-sm variant-outline" on:click={() => (showBody = !showBody)}>
|
||||
{(showBody ? 'Hide' : 'Show') + ' Body'}
|
||||
</button> -->
|
||||
</div>
|
||||
<PaneGroup direction="vertical" class="w-full rounded-lg" autoSaveId="inspectPane">
|
||||
{#if showHeaders}
|
||||
<Pane defaultSize={25} class="rounded-lg bg-muted">
|
||||
<CodeBlock
|
||||
label="Headers"
|
||||
code={JSON.stringify(JSON.parse($selectedWebhook.headers), null, 2)}
|
||||
/>
|
||||
</Pane>
|
||||
{/if}
|
||||
{#if showHeaders && showBody}
|
||||
<PaneResizer
|
||||
class="relative flex h-2 my-4 items-center justify-center bg-surface-backdrop-token rounded-full"
|
||||
>
|
||||
<div
|
||||
class="z-10 flex h-5 w-7 items-center justify-center rounded-sm border bg-primary-backdrop-token bg-primary-hover-token"
|
||||
>
|
||||
<GripHorizontal class="size-4 text-black" />
|
||||
</div>
|
||||
</PaneResizer>
|
||||
{/if}
|
||||
{#if showBody}
|
||||
<Pane defaultSize={75} class="rounded-lg bg-muted">
|
||||
<CodeBlock label="Body" code={JSON.stringify(JSON.parse($selectedWebhook.body), null, 2)} />
|
||||
</Pane>
|
||||
{/if}
|
||||
</PaneGroup>
|
||||
{/if}
|
||||
3
src/lib/server/prisma.ts
Normal file
3
src/lib/server/prisma.ts
Normal file
@@ -0,0 +1,3 @@
|
||||
import { PrismaClient } from '@prisma/client';
|
||||
|
||||
export const prisma = new PrismaClient();
|
||||
32
src/lib/server/targets.ts
Normal file
32
src/lib/server/targets.ts
Normal file
@@ -0,0 +1,32 @@
|
||||
import { PrismaClient } from '@prisma/client';
|
||||
|
||||
const prisma = new PrismaClient();
|
||||
|
||||
export async function createTarget(name: string, url: string) {
|
||||
return await prisma.relayTarget.create({
|
||||
data: {
|
||||
name,
|
||||
url
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
export async function listTargets() {
|
||||
return await prisma.relayTarget.findMany();
|
||||
}
|
||||
|
||||
export async function deleteTarget(id: number) {
|
||||
return await prisma.relayTarget.delete({
|
||||
where: {
|
||||
id
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
export async function getTarget(id: number) {
|
||||
return await prisma.relayTarget.findUnique({
|
||||
where: {
|
||||
id
|
||||
}
|
||||
});
|
||||
}
|
||||
51
src/lib/server/webhooks.ts
Normal file
51
src/lib/server/webhooks.ts
Normal file
@@ -0,0 +1,51 @@
|
||||
import { writable } from "svelte/store";
|
||||
import { prisma } from "./prisma";
|
||||
|
||||
export const newWebhooks = writable<boolean>(true);
|
||||
|
||||
export async function handleWebhook(req: Request) {
|
||||
const headers = JSON.stringify(Object.fromEntries([...req.headers]));
|
||||
const url = new URL(req.url);
|
||||
|
||||
let cleanedPath = url.pathname.replace('/ingest', '');
|
||||
if (cleanedPath === '') {
|
||||
cleanedPath = '/';
|
||||
}
|
||||
console.log(cleanedPath);
|
||||
|
||||
const webhook = await prisma.webhook.create({
|
||||
data: {
|
||||
headers,
|
||||
method: req.method,
|
||||
path: cleanedPath + url.search,
|
||||
body: await req.text()
|
||||
}
|
||||
});
|
||||
|
||||
console.log(webhook);
|
||||
newWebhooks.set(true);
|
||||
}
|
||||
|
||||
export async function listWebhooks() {
|
||||
return await prisma.webhook.findMany();
|
||||
}
|
||||
|
||||
export async function deleteWebhook(id: number) {
|
||||
return await prisma.webhook.delete({
|
||||
where: {
|
||||
id
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
export async function getWebhook(id: number) {
|
||||
return await prisma.webhook.findUnique({
|
||||
where: {
|
||||
id
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
export async function clearWebhooks() {
|
||||
return await prisma.webhook.deleteMany();
|
||||
}
|
||||
30
src/routes/+layout.svelte
Normal file
30
src/routes/+layout.svelte
Normal file
@@ -0,0 +1,30 @@
|
||||
<script lang="ts">
|
||||
import { storeHighlightJs } from '@skeletonlabs/skeleton';
|
||||
import hljs from 'highlight.js/lib/core';
|
||||
import { Toaster } from 'svelte-french-toast';
|
||||
|
||||
import '../app.css';
|
||||
|
||||
// Import required language modules
|
||||
import css from 'highlight.js/lib/languages/css';
|
||||
import javascript from 'highlight.js/lib/languages/javascript';
|
||||
import json from 'highlight.js/lib/languages/json';
|
||||
import shell from 'highlight.js/lib/languages/shell';
|
||||
import typescript from 'highlight.js/lib/languages/typescript';
|
||||
import xml from 'highlight.js/lib/languages/xml';
|
||||
|
||||
// Register languages with highlight.js
|
||||
hljs.registerLanguage('xml', xml);
|
||||
hljs.registerLanguage('css', css);
|
||||
hljs.registerLanguage('json', json);
|
||||
hljs.registerLanguage('javascript', javascript);
|
||||
hljs.registerLanguage('typescript', typescript);
|
||||
hljs.registerLanguage('shell', shell);
|
||||
|
||||
storeHighlightJs.set(hljs);
|
||||
</script>
|
||||
|
||||
<Toaster />
|
||||
|
||||
|
||||
<slot />
|
||||
70
src/routes/+page.server.ts
Normal file
70
src/routes/+page.server.ts
Normal file
@@ -0,0 +1,70 @@
|
||||
import type { Actions } from './$types';
|
||||
import { createTarget } from '$lib/server/targets';
|
||||
import { getWebhook, sendWebhook, clearWebhooks } from '$lib/server/webhooks';
|
||||
import { fail } from '@sveltejs/kit';
|
||||
|
||||
export const actions = {
|
||||
create: async ({ request }) => {
|
||||
const data = await request.formData();
|
||||
|
||||
const name = data.get('relayName');
|
||||
const url = data.get('relayUrl');
|
||||
|
||||
if (!name) {
|
||||
return fail(400, { message: 'Unable to add relay target: missing name' });
|
||||
}
|
||||
|
||||
if (!url) {
|
||||
return fail(400, { message: 'Unable to add relay target: missing url' });
|
||||
}
|
||||
|
||||
try {
|
||||
const target = await createTarget(name.toString(), url.toString());
|
||||
return { success: true, type: 'relay', message: 'Relay target created successfully', target };
|
||||
} catch (error) {
|
||||
console.error(error);
|
||||
return fail(500, { message: 'Failed to create target' });
|
||||
}
|
||||
},
|
||||
|
||||
clear: async () => {
|
||||
await clearWebhooks();
|
||||
return { success: true, type: 'webhook', message: 'Webhooks cleared successfully' };
|
||||
},
|
||||
|
||||
send: async ({ request }) => {
|
||||
const data = await request.formData();
|
||||
|
||||
const type = "webhook"
|
||||
|
||||
const webhookId = data.get('webhookId');
|
||||
const webhookTarget = data.get('webhookTarget');
|
||||
|
||||
if (!webhookId) {
|
||||
return fail(400, { type,error: 'Missing webhookId' });
|
||||
}
|
||||
|
||||
if (!webhookTarget) {
|
||||
return fail(400, { type,error: 'Missing webhookTarget' });
|
||||
}
|
||||
|
||||
const webhook = await getWebhook(Number(webhookId));
|
||||
|
||||
if (!webhook) {
|
||||
return fail(404, { type,error: 'Webhook with id ' + webhookId + ' not found' });
|
||||
}
|
||||
|
||||
try {
|
||||
const webhookResp = await sendWebhook(webhookTarget.toString(), webhook);
|
||||
|
||||
if (webhookResp.status === 200) {
|
||||
return { success: true,type, message: 'Webhook sent successfully', webhook };
|
||||
} else {
|
||||
return fail(500, { type,error: 'Failed to send webhook' });
|
||||
}
|
||||
} catch (error) {
|
||||
console.error(error);
|
||||
return fail(500, { type,error: 'Failed to send webhook' });
|
||||
}
|
||||
}
|
||||
} satisfies Actions;
|
||||
131
src/routes/+page.svelte
Normal file
131
src/routes/+page.svelte
Normal file
@@ -0,0 +1,131 @@
|
||||
<script lang="ts">
|
||||
import { browser } from '$app/environment';
|
||||
import { deleteWebhook, getWebhooks, sendWebhook } from '$lib/client/webhooks';
|
||||
import RelayControl from '$lib/components/RelayControl.svelte';
|
||||
import WebhookInspect from '$lib/components/WebhookInspect.svelte';
|
||||
import type { Webhook } from '@prisma/client';
|
||||
import { AppBar, LightSwitch } from '@skeletonlabs/skeleton';
|
||||
import { RefreshCw, Trash2 } from 'lucide-svelte';
|
||||
import { onDestroy } from 'svelte';
|
||||
import { persisted } from 'svelte-persisted-store';
|
||||
import { writable, type Writable } from 'svelte/store';
|
||||
import { fly } from 'svelte/transition';
|
||||
|
||||
const webhooks: Writable<Webhook[]> = writable([]);
|
||||
|
||||
let refreshInterval: NodeJS.Timeout;
|
||||
|
||||
const intervalInSeconds: Writable<number> = persisted('intervalInSeconds', 1);
|
||||
const webhookTarget: Writable<string> = persisted('webhookTarget', '');
|
||||
const selectedWebhook: Writable<Webhook | undefined> = persisted('selectedWebhook', undefined);
|
||||
|
||||
function createInterval(intervalInSeconds: number) {
|
||||
clearInterval(refreshInterval);
|
||||
|
||||
if (browser) {
|
||||
// Fetch webhooks on load, and on interval update
|
||||
getWebhooks().then((data) => webhooks.set(data));
|
||||
}
|
||||
|
||||
// Start the interval
|
||||
refreshInterval = setInterval(async () => {
|
||||
console.log('Refreshing webhooks');
|
||||
webhooks.set(await (await fetch('/webhooks')).json());
|
||||
}, 1000 * intervalInSeconds);
|
||||
}
|
||||
|
||||
intervalInSeconds.subscribe((value) => createInterval(value));
|
||||
webhooks.subscribe((webhooks) => {
|
||||
if ($selectedWebhook == null && webhooks?.length > 0) {
|
||||
selectedWebhook.set(webhooks[0]);
|
||||
}
|
||||
});
|
||||
|
||||
onDestroy(() => {
|
||||
clearInterval(refreshInterval);
|
||||
});
|
||||
</script>
|
||||
|
||||
<div class="flex flex-col h-screen">
|
||||
<AppBar padding="p-2">
|
||||
<svelte:fragment slot="lead">Hooky</svelte:fragment>
|
||||
<svelte:fragment slot="trail">
|
||||
<LightSwitch />
|
||||
</svelte:fragment>
|
||||
</AppBar>
|
||||
|
||||
<div class="h-full overflow-hidden">
|
||||
<div class="h-full flex gap-4 p-4">
|
||||
<!-- Sidebar (Webhooks List) -->
|
||||
<div class="w-1/3 flex flex-col space-y-2">
|
||||
<RelayControl {webhookTarget} {intervalInSeconds} />
|
||||
<div class="flex-1 overflow-hidden">
|
||||
<div class="h-full overflow-y-auto px-1 space-y-2 custom-scrollbar">
|
||||
{#each $webhooks as webhook (webhook.id)}
|
||||
<button
|
||||
class="relative overflow-hidden bg-white bg-opacity-80 backdrop-blur-sm card shadow cursor-pointer hover:shadow-md transition-shadow duration-200 w-full"
|
||||
class:selected={$selectedWebhook?.id == webhook.id}
|
||||
on:click|preventDefault={() => selectedWebhook.set(webhook)}
|
||||
transition:fly|local={{ x: -200, duration: 200 }}
|
||||
>
|
||||
{#if $selectedWebhook?.id == webhook.id}
|
||||
<!-- Active border indicator -->
|
||||
<div
|
||||
class="absolute inset-0 border-2 border-blue-500 rounded-container-token pointer-events-none"
|
||||
/>
|
||||
{/if}
|
||||
<div class="p-2 relative">
|
||||
<div class="flex justify-between items-center mb-1">
|
||||
<div class="flex items-center space-x-2">
|
||||
<span class="px-1 py-0.5 rounded-token text-xs font-mono bg-gray-200">
|
||||
{webhook.method}
|
||||
</span>
|
||||
<span class="text-xs font-medium truncate" title={webhook.path}>
|
||||
{webhook.path}
|
||||
</span>
|
||||
</div>
|
||||
<span class="text-xs text-gray-500">
|
||||
{new Date(webhook.createdAt).toLocaleString()}
|
||||
</span>
|
||||
</div>
|
||||
<div class="flex justify-between items-center">
|
||||
<span class="text-xs text-gray-500">
|
||||
ID: <span class="font-mono" title={webhook.id.toString()}>
|
||||
{webhook.id}
|
||||
</span>
|
||||
</span>
|
||||
<span>
|
||||
<button
|
||||
type="submit"
|
||||
on:click|stopPropagation={() => sendWebhook($webhookTarget, webhook.id)}
|
||||
class="btn variant-outline btn-sm"
|
||||
aria-label="Resend webhook"
|
||||
>
|
||||
<RefreshCw class="h-3 w-3 mr-1" />
|
||||
Resend
|
||||
</button>
|
||||
<button
|
||||
on:click|stopPropagation={() => deleteWebhook(webhook.id)}
|
||||
type="submit"
|
||||
formaction="?/delete?webhookId={webhook.id}"
|
||||
class="btn variant-outline btn-sm"
|
||||
aria-label="Delete webhook"
|
||||
>
|
||||
<Trash2 class="h-3 w-3 mr-1" />
|
||||
Delete
|
||||
</button>
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</button>
|
||||
{/each}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<!-- Main Content (Headers and Request Body) -->
|
||||
<div class="w-2/3 flex flex-col space-y-2">
|
||||
<WebhookInspect {selectedWebhook} />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
11
src/routes/ingest/+server.ts
Normal file
11
src/routes/ingest/+server.ts
Normal file
@@ -0,0 +1,11 @@
|
||||
import { handleWebhook } from '$lib/server/webhooks';
|
||||
|
||||
export async function POST({ request }) {
|
||||
|
||||
await handleWebhook(request);
|
||||
|
||||
//return a simple 200 response
|
||||
return new Response(null, {
|
||||
status: 200,
|
||||
});
|
||||
}
|
||||
11
src/routes/ingest/[...path]/+server.ts
Normal file
11
src/routes/ingest/[...path]/+server.ts
Normal file
@@ -0,0 +1,11 @@
|
||||
import { handleWebhook } from '$lib/server/webhooks';
|
||||
|
||||
export async function POST({ request }) {
|
||||
|
||||
await handleWebhook(request);
|
||||
|
||||
//return a simple 200 response
|
||||
return new Response(null, {
|
||||
status: 200,
|
||||
});
|
||||
}
|
||||
23
src/routes/targets/+server.ts
Normal file
23
src/routes/targets/+server.ts
Normal file
@@ -0,0 +1,23 @@
|
||||
import { createTarget, listTargets } from '$lib/server/targets';
|
||||
import { error, json } from '@sveltejs/kit';
|
||||
|
||||
export async function GET() {
|
||||
return json(await listTargets());
|
||||
}
|
||||
|
||||
export async function POST({ request }) {
|
||||
const data = await request.json();
|
||||
if (!data.name) {
|
||||
return error(400, { title: 'Missing name', message: 'Relay Target Name is required' });
|
||||
}
|
||||
|
||||
if (!data.url) {
|
||||
return error(400, { title: 'Missing url', message: 'Relay Target URL is required' });
|
||||
}
|
||||
|
||||
await createTarget(data.name, data.url);
|
||||
|
||||
return new Response(null, {
|
||||
status: 201
|
||||
});
|
||||
}
|
||||
22
src/routes/targets/[id]/+server.ts
Normal file
22
src/routes/targets/[id]/+server.ts
Normal file
@@ -0,0 +1,22 @@
|
||||
import { deleteTarget, getTarget } from '$lib/server/targets';
|
||||
import { error, json } from '@sveltejs/kit';
|
||||
|
||||
export async function DELETE({ params }) {
|
||||
if (!params.id) {
|
||||
return error(400, { message: 'Missing targetId' });
|
||||
}
|
||||
|
||||
const Target = await getTarget(Number(params.id));
|
||||
|
||||
if (!Target) {
|
||||
return error(404, { message: `Target with id ${params.id} not found` });
|
||||
}
|
||||
|
||||
try {
|
||||
await deleteTarget(Target.id);
|
||||
return json({ message: 'Target deleted successfully' });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
return error(500, { message: `Failed to delete Target: ${err}` });
|
||||
}
|
||||
}
|
||||
12
src/routes/webhooks/+server.ts
Normal file
12
src/routes/webhooks/+server.ts
Normal file
@@ -0,0 +1,12 @@
|
||||
import { listWebhooks, newWebhooks } from '$lib/server/webhooks';
|
||||
import { json } from '@sveltejs/kit';
|
||||
import { get } from 'svelte/store';
|
||||
|
||||
export async function HEAD() {
|
||||
return get(newWebhooks) === true ? new Response(null, { status: 200 }) : new Response(null, { status: 204 });
|
||||
}
|
||||
|
||||
export async function GET() {
|
||||
newWebhooks.set(false);
|
||||
return json(await listWebhooks());
|
||||
}
|
||||
61
src/routes/webhooks/[id]/+server.ts
Normal file
61
src/routes/webhooks/[id]/+server.ts
Normal file
@@ -0,0 +1,61 @@
|
||||
import { deleteWebhook, getWebhook } from '$lib/server/webhooks';
|
||||
import { error, json } from '@sveltejs/kit';
|
||||
import axios, { AxiosError } from 'axios';
|
||||
|
||||
export async function DELETE({ params }) {
|
||||
if (!params.id) {
|
||||
return error(400, { message: 'Missing webhookId' });
|
||||
}
|
||||
|
||||
const webhook = await getWebhook(Number(params.id));
|
||||
|
||||
if (!webhook) {
|
||||
return error(404, { message: `Webhook with id ${params.id} not found` });
|
||||
}
|
||||
|
||||
try {
|
||||
await deleteWebhook(webhook.id);
|
||||
return json({ message: 'Webhook deleted successfully' });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
return error(500, { message: `Failed to delete webhook: ${err}` });
|
||||
}
|
||||
}
|
||||
|
||||
export async function POST({ params, url }) {
|
||||
if (!params.id) {
|
||||
return error(400, { message: 'Missing webhookId' });
|
||||
}
|
||||
|
||||
const webhookTarget = url.searchParams.get('webhookTarget');
|
||||
|
||||
if (!webhookTarget) {
|
||||
return error(400, { message: 'Missing webhookTarget' });
|
||||
}
|
||||
|
||||
const webhook = await getWebhook(Number(params.id));
|
||||
|
||||
if (!webhook) {
|
||||
return error(404, {
|
||||
title: 'Webhook not found',
|
||||
message: `Webhook with id ${params.id} not found`
|
||||
});
|
||||
}
|
||||
|
||||
try {
|
||||
await axios.post(webhookTarget + webhook.path, webhook.body, {
|
||||
headers: JSON.parse(webhook.headers)
|
||||
});
|
||||
return json({ message: 'Webhook sent successfully' });
|
||||
} catch (err) {
|
||||
const Error = err as Error | AxiosError;
|
||||
if (axios.isAxiosError(Error)) {
|
||||
return error(500, {
|
||||
title: 'Failed to send Webhook',
|
||||
message: Error.cause?.message || Error.message
|
||||
});
|
||||
} else {
|
||||
return error(500, { title: 'Failed to send Webhook', message: Error.message });
|
||||
}
|
||||
}
|
||||
}
|
||||
BIN
static/favicon.png
Normal file
BIN
static/favicon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.5 KiB |
18
svelte.config.js
Normal file
18
svelte.config.js
Normal file
@@ -0,0 +1,18 @@
|
||||
import adapter from '@sveltejs/adapter-auto';
|
||||
import { vitePreprocess } from '@sveltejs/vite-plugin-svelte';
|
||||
|
||||
/** @type {import('@sveltejs/kit').Config} */
|
||||
const config = {
|
||||
// Consult https://kit.svelte.dev/docs/integrations#preprocessors
|
||||
// for more information about preprocessors
|
||||
preprocess: vitePreprocess(),
|
||||
|
||||
kit: {
|
||||
// adapter-auto only supports some environments, see https://kit.svelte.dev/docs/adapter-auto for a list.
|
||||
// If your environment is not supported, or you settled on a specific environment, switch out the adapter.
|
||||
// See https://kit.svelte.dev/docs/adapters for more information about adapters.
|
||||
adapter: adapter()
|
||||
}
|
||||
};
|
||||
|
||||
export default config;
|
||||
31
tailwind.config.ts
Normal file
31
tailwind.config.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
|
||||
import { join } from 'path';
|
||||
import type { Config } from 'tailwindcss';
|
||||
|
||||
// 1. Import the Skeleton plugin
|
||||
import { skeleton } from '@skeletonlabs/tw-plugin';
|
||||
|
||||
const config = {
|
||||
// 2. Opt for dark mode to be handled via the class method
|
||||
darkMode: 'class',
|
||||
content: [
|
||||
'./src/**/*.{html,js,svelte,ts}',
|
||||
// 3. Append the path to the Skeleton package
|
||||
join(require.resolve(
|
||||
'@skeletonlabs/skeleton'),
|
||||
'../**/*.{html,js,svelte,ts}'
|
||||
)
|
||||
],
|
||||
theme: {
|
||||
extend: {},
|
||||
},
|
||||
plugins: [
|
||||
// 4. Append the Skeleton plugin (after other plugins)
|
||||
skeleton({
|
||||
themes: { preset: [ {name: "skeleton", enhancements: true},{name: "wintry", enhancements: true},{name: "hamlindigo", enhancements: true} ] }
|
||||
})
|
||||
]
|
||||
} satisfies Config;
|
||||
|
||||
export default config;
|
||||
|
||||
19
tsconfig.json
Normal file
19
tsconfig.json
Normal file
@@ -0,0 +1,19 @@
|
||||
{
|
||||
"extends": "./.svelte-kit/tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"allowJs": true,
|
||||
"checkJs": true,
|
||||
"esModuleInterop": true,
|
||||
"forceConsistentCasingInFileNames": true,
|
||||
"resolveJsonModule": true,
|
||||
"skipLibCheck": true,
|
||||
"sourceMap": true,
|
||||
"strict": true,
|
||||
"moduleResolution": "bundler"
|
||||
}
|
||||
// Path aliases are handled by https://kit.svelte.dev/docs/configuration#alias
|
||||
// except $lib which is handled by https://kit.svelte.dev/docs/configuration#files
|
||||
//
|
||||
// If you want to overwrite includes/excludes, make sure to copy over the relevant includes/excludes
|
||||
// from the referenced tsconfig.json - TypeScript does not merge them in
|
||||
}
|
||||
9
vite.config.ts
Normal file
9
vite.config.ts
Normal file
@@ -0,0 +1,9 @@
|
||||
import { sveltekit } from '@sveltejs/kit/vite';
|
||||
import { defineConfig } from 'vitest/config';
|
||||
|
||||
export default defineConfig({
|
||||
plugins: [sveltekit()],
|
||||
test: {
|
||||
include: ['src/**/*.{test,spec}.{js,ts}']
|
||||
}
|
||||
});
|
||||
Reference in New Issue
Block a user