mirror of
https://github.com/superschnups/Emy.git
synced 2026-06-22 03:13:10 +00:00
first commit
This commit is contained in:
parent
3f6417175e
commit
ddbe56763d
843 changed files with 112026 additions and 0 deletions
12
.gitignore
vendored
Normal file
12
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
*.jpg
|
||||||
|
*.jpeg
|
||||||
|
*.png
|
||||||
|
*.gif
|
||||||
|
*.webp
|
||||||
|
*.svg
|
||||||
|
# oder ganzen Ordner:
|
||||||
|
/images/
|
||||||
|
/img/
|
||||||
|
/assets/images/
|
||||||
|
public/
|
||||||
|
static/img
|
||||||
0
.hugo_build.lock
Normal file
0
.hugo_build.lock
Normal file
547
admin-server.js
Normal file
547
admin-server.js
Normal file
|
|
@ -0,0 +1,547 @@
|
||||||
|
const express = require('express');
|
||||||
|
const multer = require('multer');
|
||||||
|
const sharp = require('sharp');
|
||||||
|
const path = require('path');
|
||||||
|
const fs = require('fs');
|
||||||
|
const { exec } = require('child_process');
|
||||||
|
|
||||||
|
const app = express();
|
||||||
|
const PORT = 3001;
|
||||||
|
|
||||||
|
const IMAGES_DIR = path.join(__dirname, 'static/gallery/images');
|
||||||
|
const HERO_DIR = path.join(__dirname, 'static/hero');
|
||||||
|
const DATA_FILE = path.join(__dirname, 'data/gallery.json');
|
||||||
|
const HOMEPAGE_FILE = path.join(__dirname, 'data/homepage.json');
|
||||||
|
const CATEGORIES_FILE = path.join(__dirname, 'data/categories.json');
|
||||||
|
|
||||||
|
if (!fs.existsSync(HERO_DIR)) fs.mkdirSync(HERO_DIR, { recursive: true });
|
||||||
|
|
||||||
|
app.use(express.json());
|
||||||
|
app.use('/images', express.static(IMAGES_DIR));
|
||||||
|
|
||||||
|
// Multer: temporärer Speicher, dann sharp übernimmt
|
||||||
|
const upload = multer({
|
||||||
|
storage: multer.memoryStorage(),
|
||||||
|
limits: { fileSize: 20 * 1024 * 1024 },
|
||||||
|
fileFilter: (req, file, cb) => {
|
||||||
|
if (file.mimetype.startsWith('image/')) cb(null, true);
|
||||||
|
else cb(new Error('Nur Bilder erlaubt!'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function readGallery() {
|
||||||
|
return JSON.parse(fs.readFileSync(DATA_FILE, 'utf8'));
|
||||||
|
}
|
||||||
|
function writeGallery(data) {
|
||||||
|
fs.writeFileSync(DATA_FILE, JSON.stringify(data, null, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload
|
||||||
|
app.post('/api/upload', upload.array('photos', 20), async (req, res) => {
|
||||||
|
try {
|
||||||
|
const gallery = readGallery();
|
||||||
|
const added = [];
|
||||||
|
|
||||||
|
for (const file of req.files) {
|
||||||
|
const id = Date.now() + '-' + Math.random().toString(36).slice(2, 7);
|
||||||
|
const filename = id + '.webp';
|
||||||
|
const thumbname = id + '-thumb.webp';
|
||||||
|
|
||||||
|
// Vollbild (max 1200px)
|
||||||
|
await sharp(file.buffer)
|
||||||
|
.resize(1200, 1200, { fit: 'inside', withoutEnlargement: true })
|
||||||
|
.webp({ quality: 85 })
|
||||||
|
.toFile(path.join(IMAGES_DIR, filename));
|
||||||
|
|
||||||
|
// Thumbnail (400px)
|
||||||
|
await sharp(file.buffer)
|
||||||
|
.resize(400, 400, { fit: 'cover' })
|
||||||
|
.webp({ quality: 80 })
|
||||||
|
.toFile(path.join(IMAGES_DIR, thumbname));
|
||||||
|
|
||||||
|
const photo = {
|
||||||
|
id,
|
||||||
|
filename,
|
||||||
|
thumb: thumbname,
|
||||||
|
title: req.body.title || file.originalname.replace(/\.[^.]+$/, ''),
|
||||||
|
kategorie: req.body.kategorie || 'Training',
|
||||||
|
datum: new Date().toISOString().slice(0, 10)
|
||||||
|
};
|
||||||
|
gallery.photos.unshift(photo);
|
||||||
|
added.push(photo);
|
||||||
|
}
|
||||||
|
|
||||||
|
writeGallery(gallery);
|
||||||
|
res.json({ ok: true, added });
|
||||||
|
} catch (e) {
|
||||||
|
res.status(500).json({ ok: false, error: e.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Foto-Metadaten updaten
|
||||||
|
app.put('/api/photo/:id', (req, res) => {
|
||||||
|
const gallery = readGallery();
|
||||||
|
const photo = gallery.photos.find(p => p.id === req.params.id);
|
||||||
|
if (!photo) return res.status(404).json({ ok: false });
|
||||||
|
if (req.body.title !== undefined) photo.title = req.body.title;
|
||||||
|
if (req.body.kategorie !== undefined) photo.kategorie = req.body.kategorie;
|
||||||
|
writeGallery(gallery);
|
||||||
|
res.json({ ok: true, photo });
|
||||||
|
});
|
||||||
|
|
||||||
|
// Foto löschen
|
||||||
|
app.delete('/api/photo/:id', (req, res) => {
|
||||||
|
const gallery = readGallery();
|
||||||
|
const idx = gallery.photos.findIndex(p => p.id === req.params.id);
|
||||||
|
if (idx === -1) return res.status(404).json({ ok: false });
|
||||||
|
const [photo] = gallery.photos.splice(idx, 1);
|
||||||
|
[photo.filename, photo.thumb].forEach(f => {
|
||||||
|
const fp = path.join(IMAGES_DIR, f);
|
||||||
|
if (fs.existsSync(fp)) fs.unlinkSync(fp);
|
||||||
|
});
|
||||||
|
writeGallery(gallery);
|
||||||
|
res.json({ ok: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
// Alle Fotos
|
||||||
|
app.get('/api/photos', (req, res) => {
|
||||||
|
res.json(readGallery());
|
||||||
|
});
|
||||||
|
|
||||||
|
// ── Kategorien API ────────────────────────────────────────────
|
||||||
|
function readCategories() {
|
||||||
|
if (!fs.existsSync(CATEGORIES_FILE)) return ['Training', 'Wettkämpfe', 'Gürtelprüfungen'];
|
||||||
|
return JSON.parse(fs.readFileSync(CATEGORIES_FILE, 'utf8'));
|
||||||
|
}
|
||||||
|
function writeCategories(cats) {
|
||||||
|
fs.writeFileSync(CATEGORIES_FILE, JSON.stringify(cats, null, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
app.get('/api/categories', (req, res) => {
|
||||||
|
res.json(readCategories());
|
||||||
|
});
|
||||||
|
|
||||||
|
app.post('/api/categories', (req, res) => {
|
||||||
|
const name = (req.body.name || '').trim();
|
||||||
|
if (!name) return res.status(400).json({ ok: false, error: 'Name fehlt' });
|
||||||
|
const cats = readCategories();
|
||||||
|
if (cats.includes(name)) return res.status(409).json({ ok: false, error: 'Existiert bereits' });
|
||||||
|
cats.push(name);
|
||||||
|
writeCategories(cats);
|
||||||
|
res.json({ ok: true, categories: cats });
|
||||||
|
});
|
||||||
|
|
||||||
|
app.delete('/api/categories/:name', (req, res) => {
|
||||||
|
const cats = readCategories().filter(c => c !== req.params.name);
|
||||||
|
writeCategories(cats);
|
||||||
|
res.json({ ok: true, categories: cats });
|
||||||
|
});
|
||||||
|
|
||||||
|
// ── Homepage API ──────────────────────────────────────────────
|
||||||
|
function readHomepage() {
|
||||||
|
return JSON.parse(fs.readFileSync(HOMEPAGE_FILE, 'utf8'));
|
||||||
|
}
|
||||||
|
function writeHomepage(data) {
|
||||||
|
fs.writeFileSync(HOMEPAGE_FILE, JSON.stringify(data, null, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
app.get('/api/homepage', (req, res) => {
|
||||||
|
res.json(readHomepage());
|
||||||
|
});
|
||||||
|
|
||||||
|
function rebuildAndDeploy() {
|
||||||
|
const SSH_KEY = '/Users/jessi/.ssh/vpsserver/vpsserver';
|
||||||
|
const cmd = `cd /Users/jessi/karatehp && hugo --minify && SSH_ASKPASS_REQUIRE=never ssh-add ${SSH_KEY} <<< "bonzeikiller" 2>/dev/null; rsync -az --delete -e "ssh -o StrictHostKeyChecking=no -i ${SSH_KEY}" /Users/jessi/karatehp/public/ root@217.160.212.198:/var/www/emy.bonzeipunk.de/`;
|
||||||
|
exec(cmd, { shell: '/bin/bash' }, (err) => {
|
||||||
|
if (err) console.error('Deploy-Fehler:', err.message);
|
||||||
|
else console.log('✓ Deploy erfolgreich');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
app.put('/api/homepage', (req, res) => {
|
||||||
|
const hp = readHomepage();
|
||||||
|
if (req.body.badge !== undefined) hp.hero.badge = req.body.badge;
|
||||||
|
if (req.body.description !== undefined) hp.hero.description = req.body.description;
|
||||||
|
if (req.body.stats !== undefined) hp.stats = req.body.stats;
|
||||||
|
writeHomepage(hp);
|
||||||
|
res.json({ ok: true });
|
||||||
|
rebuildAndDeploy();
|
||||||
|
});
|
||||||
|
|
||||||
|
app.post('/api/homepage/image', upload.single('image'), async (req, res) => {
|
||||||
|
try {
|
||||||
|
const filename = 'hero.webp';
|
||||||
|
await sharp(req.file.buffer)
|
||||||
|
.resize(1200, 1200, { fit: 'inside', withoutEnlargement: true })
|
||||||
|
.webp({ quality: 88 })
|
||||||
|
.toFile(path.join(HERO_DIR, filename));
|
||||||
|
const hp = readHomepage();
|
||||||
|
hp.hero.image = filename;
|
||||||
|
writeHomepage(hp);
|
||||||
|
res.json({ ok: true, image: filename });
|
||||||
|
rebuildAndDeploy();
|
||||||
|
} catch (e) {
|
||||||
|
res.status(500).json({ ok: false, error: e.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
app.use('/hero', express.static(HERO_DIR));
|
||||||
|
|
||||||
|
// Admin-UI
|
||||||
|
app.get('/', (req, res) => {
|
||||||
|
res.sendFile(path.join(__dirname, 'admin.html'));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get('/__old', (req, res) => {
|
||||||
|
res.send(`<!DOCTYPE html>
|
||||||
|
<html lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||||
|
<title>Galerie verwalten 📸</title>
|
||||||
|
<script src="https://cdn.tailwindcss.com"></script>
|
||||||
|
<style>
|
||||||
|
* { font-family: 'Segoe UI', system-ui, sans-serif; }
|
||||||
|
.drop-zone { border: 3px dashed #e879a0; transition: all .2s; }
|
||||||
|
.drop-zone.drag-over { background: #fdf2f8; border-color: #b30065; transform: scale(1.01); }
|
||||||
|
.photo-card { transition: transform .2s, box-shadow .2s; }
|
||||||
|
.photo-card:hover { transform: translateY(-4px); box-shadow: 0 12px 24px rgba(179,0,101,.15); }
|
||||||
|
.kategorie-btn.active { background: #8930b0; color: white; }
|
||||||
|
.progress-bar { transition: width .3s; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="bg-pink-50 min-h-screen">
|
||||||
|
|
||||||
|
<!-- Header -->
|
||||||
|
<header class="bg-white shadow-md shadow-pink-100 sticky top-0 z-50">
|
||||||
|
<div class="max-w-5xl mx-auto px-6 py-4 flex items-center justify-between">
|
||||||
|
<div>
|
||||||
|
<h1 class="text-2xl font-black text-pink-600 italic uppercase tracking-tight">MiyaKarate</h1>
|
||||||
|
<p class="text-xs text-gray-400 font-medium">Galerie verwalten</p>
|
||||||
|
</div>
|
||||||
|
<a href="http://localhost:1313/galerie/" target="_blank"
|
||||||
|
class="flex items-center gap-2 bg-pink-600 text-white px-5 py-2.5 rounded-full font-bold text-sm hover:bg-pink-700 transition-colors">
|
||||||
|
🌐 Website ansehen
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main class="max-w-5xl mx-auto px-6 py-10 space-y-10">
|
||||||
|
|
||||||
|
<!-- Upload-Bereich -->
|
||||||
|
<section class="bg-white rounded-3xl p-8 shadow-sm">
|
||||||
|
<h2 class="text-xl font-black text-gray-800 mb-6 flex items-center gap-2">
|
||||||
|
<span class="text-3xl">📤</span> Fotos hochladen
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<!-- Drop-Zone -->
|
||||||
|
<div id="dropZone"
|
||||||
|
class="drop-zone rounded-2xl p-12 text-center cursor-pointer mb-6"
|
||||||
|
onclick="document.getElementById('fileInput').click()">
|
||||||
|
<div class="text-6xl mb-4">🖼️</div>
|
||||||
|
<p class="text-xl font-bold text-pink-600 mb-2">Fotos hier reinziehen</p>
|
||||||
|
<p class="text-gray-400 text-sm">oder hier klicken zum Auswählen</p>
|
||||||
|
<p class="text-gray-300 text-xs mt-2">JPG, PNG, HEIC • bis zu 20 MB</p>
|
||||||
|
</div>
|
||||||
|
<input type="file" id="fileInput" multiple accept="image/*" class="hidden"/>
|
||||||
|
|
||||||
|
<!-- Metadaten -->
|
||||||
|
<div class="grid grid-cols-1 sm:grid-cols-2 gap-4 mb-6">
|
||||||
|
<div>
|
||||||
|
<label class="block text-sm font-bold text-gray-600 mb-2">📝 Titel (optional)</label>
|
||||||
|
<input id="uploadTitle" type="text" placeholder="z.B. Landesmeisterschaft 2024"
|
||||||
|
class="w-full border-2 border-pink-100 rounded-xl px-4 py-3 focus:outline-none focus:border-pink-400 text-gray-700"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label class="block text-sm font-bold text-gray-600 mb-2">🏷️ Kategorie</label>
|
||||||
|
<div class="flex gap-2 flex-wrap pt-1">
|
||||||
|
<button onclick="setKat(this,'Training')" class="kategorie-btn active px-4 py-2 rounded-full border-2 border-purple-200 text-sm font-bold transition-colors">Training</button>
|
||||||
|
<button onclick="setKat(this,'Wettkämpfe')" class="kategorie-btn px-4 py-2 rounded-full border-2 border-purple-200 text-sm font-bold transition-colors bg-white text-gray-600">Wettkämpfe</button>
|
||||||
|
<button onclick="setKat(this,'Gürtelprüfungen')" class="kategorie-btn px-4 py-2 rounded-full border-2 border-purple-200 text-sm font-bold transition-colors bg-white text-gray-600">Gürtelprüfungen</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Vorschau der gewählten Dateien -->
|
||||||
|
<div id="previewList" class="grid grid-cols-3 sm:grid-cols-5 gap-3 mb-6 hidden"></div>
|
||||||
|
|
||||||
|
<!-- Fortschrittsbalken -->
|
||||||
|
<div id="progressWrap" class="hidden mb-4">
|
||||||
|
<div class="h-3 bg-pink-100 rounded-full overflow-hidden">
|
||||||
|
<div id="progressBar" class="progress-bar h-full bg-gradient-to-r from-pink-500 to-purple-500 rounded-full" style="width:0%"></div>
|
||||||
|
</div>
|
||||||
|
<p id="progressText" class="text-sm text-gray-500 mt-2 text-center">Wird hochgeladen…</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button id="uploadBtn" onclick="startUpload()" disabled
|
||||||
|
class="w-full py-4 rounded-2xl font-black text-xl bg-gradient-to-r from-pink-500 to-purple-600 text-white shadow-lg shadow-pink-200 disabled:opacity-40 disabled:cursor-not-allowed hover:scale-[1.01] active:scale-95 transition-transform">
|
||||||
|
✨ Hochladen!
|
||||||
|
</button>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Meine Fotos -->
|
||||||
|
<section class="bg-white rounded-3xl p-8 shadow-sm">
|
||||||
|
<div class="flex items-center justify-between mb-6">
|
||||||
|
<h2 class="text-xl font-black text-gray-800 flex items-center gap-2">
|
||||||
|
<span class="text-3xl">🗂️</span> Meine Fotos
|
||||||
|
<span id="photoCount" class="text-sm font-bold text-pink-500 bg-pink-50 px-3 py-1 rounded-full"></span>
|
||||||
|
</h2>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<button onclick="filterPhotos('Alle')" class="filter-btn active text-xs font-bold px-3 py-1.5 rounded-full bg-pink-600 text-white">Alle</button>
|
||||||
|
<button onclick="filterPhotos('Training')" class="filter-btn text-xs font-bold px-3 py-1.5 rounded-full bg-gray-100 text-gray-600 hover:bg-gray-200">Training</button>
|
||||||
|
<button onclick="filterPhotos('Wettkämpfe')" class="filter-btn text-xs font-bold px-3 py-1.5 rounded-full bg-gray-100 text-gray-600 hover:bg-gray-200">Wettkämpfe</button>
|
||||||
|
<button onclick="filterPhotos('Gürtelprüfungen')" class="filter-btn text-xs font-bold px-3 py-1.5 rounded-full bg-gray-100 text-gray-600 hover:bg-gray-200">Gürtel</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div id="photoGrid" class="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 gap-4">
|
||||||
|
<div class="col-span-full text-center py-16 text-gray-300">
|
||||||
|
<div class="text-5xl mb-3">📭</div>
|
||||||
|
<p class="font-bold">Noch keine Fotos</p>
|
||||||
|
<p class="text-sm">Lad dein erstes Foto hoch!</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- Edit Modal -->
|
||||||
|
<div id="editModal" class="hidden fixed inset-0 bg-black/50 backdrop-blur-sm z-50 flex items-center justify-center p-4">
|
||||||
|
<div class="bg-white rounded-3xl p-8 w-full max-w-md shadow-2xl">
|
||||||
|
<h3 class="text-xl font-black text-gray-800 mb-6">✏️ Foto bearbeiten</h3>
|
||||||
|
<input type="hidden" id="editId"/>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<div>
|
||||||
|
<label class="block text-sm font-bold text-gray-600 mb-2">Titel</label>
|
||||||
|
<input id="editTitle" type="text"
|
||||||
|
class="w-full border-2 border-pink-100 rounded-xl px-4 py-3 focus:outline-none focus:border-pink-400 text-gray-700"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label class="block text-sm font-bold text-gray-600 mb-2">Kategorie</label>
|
||||||
|
<select id="editKat"
|
||||||
|
class="w-full border-2 border-pink-100 rounded-xl px-4 py-3 focus:outline-none focus:border-pink-400 text-gray-700">
|
||||||
|
<option>Training</option>
|
||||||
|
<option>Wettkämpfe</option>
|
||||||
|
<option>Gürtelprüfungen</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-3 mt-8">
|
||||||
|
<button onclick="saveEdit()" class="flex-1 py-3 rounded-xl font-black bg-gradient-to-r from-pink-500 to-purple-600 text-white hover:scale-[1.02] active:scale-95 transition-transform">💾 Speichern</button>
|
||||||
|
<button onclick="closeModal()" class="px-6 py-3 rounded-xl font-bold bg-gray-100 text-gray-600 hover:bg-gray-200 transition-colors">Abbrechen</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Toast -->
|
||||||
|
<div id="toast" class="hidden fixed bottom-6 left-1/2 -translate-x-1/2 bg-gray-900 text-white px-6 py-3 rounded-full font-bold text-sm shadow-xl z-50"></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
let selectedKat = 'Training';
|
||||||
|
let allPhotos = [];
|
||||||
|
let currentFilter = 'Alle';
|
||||||
|
let pendingFiles = []; // speichert Dateien aus Drag&Drop UND file input
|
||||||
|
|
||||||
|
function setKat(btn, kat) {
|
||||||
|
selectedKat = kat;
|
||||||
|
document.querySelectorAll('.kategorie-btn').forEach(b => {
|
||||||
|
b.classList.remove('active');
|
||||||
|
b.classList.add('bg-white', 'text-gray-600');
|
||||||
|
});
|
||||||
|
btn.classList.add('active');
|
||||||
|
btn.classList.remove('bg-white', 'text-gray-600');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Drop-Zone
|
||||||
|
const dz = document.getElementById('dropZone');
|
||||||
|
const fi = document.getElementById('fileInput');
|
||||||
|
dz.addEventListener('dragover', e => { e.preventDefault(); dz.classList.add('drag-over'); });
|
||||||
|
dz.addEventListener('dragleave', () => dz.classList.remove('drag-over'));
|
||||||
|
dz.addEventListener('drop', e => {
|
||||||
|
e.preventDefault();
|
||||||
|
dz.classList.remove('drag-over');
|
||||||
|
pendingFiles = Array.from(e.dataTransfer.files).filter(f => f.type.startsWith('image/'));
|
||||||
|
showPreviews(pendingFiles);
|
||||||
|
});
|
||||||
|
fi.addEventListener('change', () => {
|
||||||
|
pendingFiles = Array.from(fi.files);
|
||||||
|
showPreviews(pendingFiles);
|
||||||
|
});
|
||||||
|
|
||||||
|
function showPreviews(files) {
|
||||||
|
const list = document.getElementById('previewList');
|
||||||
|
list.innerHTML = '';
|
||||||
|
if (!files.length) { list.classList.add('hidden'); document.getElementById('uploadBtn').disabled = true; return; }
|
||||||
|
list.classList.remove('hidden');
|
||||||
|
document.getElementById('uploadBtn').disabled = false;
|
||||||
|
files.forEach(f => {
|
||||||
|
const url = URL.createObjectURL(f);
|
||||||
|
const div = document.createElement('div');
|
||||||
|
div.className = 'aspect-square rounded-xl overflow-hidden bg-gray-100 relative';
|
||||||
|
div.innerHTML = \`<img src="\${url}" class="w-full h-full object-cover"/>\`;
|
||||||
|
list.appendChild(div);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function startUpload() {
|
||||||
|
const files = pendingFiles;
|
||||||
|
if (!files.length) return;
|
||||||
|
|
||||||
|
const btn = document.getElementById('uploadBtn');
|
||||||
|
const wrap = document.getElementById('progressWrap');
|
||||||
|
const bar = document.getElementById('progressBar');
|
||||||
|
const txt = document.getElementById('progressText');
|
||||||
|
|
||||||
|
btn.disabled = true;
|
||||||
|
wrap.classList.remove('hidden');
|
||||||
|
|
||||||
|
const fd = new FormData();
|
||||||
|
Array.from(files).forEach(f => fd.append('photos', f));
|
||||||
|
fd.append('title', document.getElementById('uploadTitle').value);
|
||||||
|
fd.append('kategorie', selectedKat);
|
||||||
|
|
||||||
|
// Simulate progress during upload
|
||||||
|
let prog = 0;
|
||||||
|
const ticker = setInterval(() => {
|
||||||
|
prog = Math.min(prog + 3, 90);
|
||||||
|
bar.style.width = prog + '%';
|
||||||
|
}, 100);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const r = await fetch('/api/upload', { method: 'POST', body: fd });
|
||||||
|
clearInterval(ticker);
|
||||||
|
bar.style.width = '100%';
|
||||||
|
txt.textContent = 'Fertig! ✨';
|
||||||
|
|
||||||
|
if (r.ok) {
|
||||||
|
showToast('✅ ' + files.length + ' Foto(s) hochgeladen!');
|
||||||
|
fi.value = '';
|
||||||
|
pendingFiles = [];
|
||||||
|
document.getElementById('previewList').innerHTML = '';
|
||||||
|
document.getElementById('previewList').classList.add('hidden');
|
||||||
|
document.getElementById('uploadTitle').value = '';
|
||||||
|
setTimeout(() => { wrap.classList.add('hidden'); bar.style.width = '0%'; btn.disabled = false; }, 1500);
|
||||||
|
loadPhotos();
|
||||||
|
} else {
|
||||||
|
showToast('❌ Fehler beim Hochladen');
|
||||||
|
btn.disabled = false;
|
||||||
|
}
|
||||||
|
} catch(e) {
|
||||||
|
clearInterval(ticker);
|
||||||
|
showToast('❌ Verbindungsfehler');
|
||||||
|
btn.disabled = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadPhotos() {
|
||||||
|
const r = await fetch('/api/photos');
|
||||||
|
const data = await r.json();
|
||||||
|
allPhotos = data.photos || [];
|
||||||
|
renderPhotos();
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderPhotos() {
|
||||||
|
const grid = document.getElementById('photoGrid');
|
||||||
|
const count = document.getElementById('photoCount');
|
||||||
|
const filtered = currentFilter === 'Alle' ? allPhotos : allPhotos.filter(p => p.kategorie === currentFilter);
|
||||||
|
|
||||||
|
count.textContent = allPhotos.length + ' Fotos';
|
||||||
|
|
||||||
|
if (!filtered.length) {
|
||||||
|
grid.innerHTML = \`<div class="col-span-full text-center py-16 text-gray-300">
|
||||||
|
<div class="text-5xl mb-3">\${currentFilter === 'Alle' ? '📭' : '🔍'}</div>
|
||||||
|
<p class="font-bold">\${currentFilter === 'Alle' ? 'Noch keine Fotos' : 'Keine Fotos in dieser Kategorie'}</p>
|
||||||
|
\${currentFilter === 'Alle' ? '<p class="text-sm">Lad dein erstes Foto hoch!</p>' : ''}
|
||||||
|
</div>\`;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const katColors = { 'Training': 'bg-pink-100 text-pink-700', 'Wettkämpfe': 'bg-purple-100 text-purple-700', 'Gürtelprüfungen': 'bg-amber-100 text-amber-700' };
|
||||||
|
|
||||||
|
grid.innerHTML = filtered.map(p => \`
|
||||||
|
<div class="photo-card group relative bg-white rounded-2xl overflow-hidden shadow-sm border border-gray-100" data-id="\${p.id}" data-kat="\${p.kategorie}">
|
||||||
|
<div class="aspect-square overflow-hidden">
|
||||||
|
<img src="/images/\${p.thumb}" class="w-full h-full object-cover group-hover:scale-105 transition-transform duration-300" alt="\${p.title}"/>
|
||||||
|
</div>
|
||||||
|
<div class="p-3">
|
||||||
|
<p class="font-bold text-gray-800 text-sm truncate mb-1">\${p.title}</p>
|
||||||
|
<span class="text-[10px] font-bold px-2 py-0.5 rounded-full \${katColors[p.kategorie] || 'bg-gray-100 text-gray-600'}">\${p.kategorie}</span>
|
||||||
|
</div>
|
||||||
|
<div class="absolute top-2 right-2 flex gap-1.5 opacity-0 group-hover:opacity-100 transition-opacity">
|
||||||
|
<button onclick="openEdit('\${p.id}',\`\${p.title}\`,'\${p.kategorie}')"
|
||||||
|
class="w-8 h-8 bg-white rounded-full shadow-md flex items-center justify-center text-base hover:scale-110 transition-transform">✏️</button>
|
||||||
|
<button onclick="deletePhoto('\${p.id}')"
|
||||||
|
class="w-8 h-8 bg-red-500 rounded-full shadow-md flex items-center justify-center text-base hover:scale-110 transition-transform">🗑️</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
\`).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterPhotos(kat) {
|
||||||
|
currentFilter = kat;
|
||||||
|
document.querySelectorAll('.filter-btn').forEach(b => {
|
||||||
|
b.classList.remove('bg-pink-600', 'text-white');
|
||||||
|
b.classList.add('bg-gray-100', 'text-gray-600');
|
||||||
|
});
|
||||||
|
event.target.classList.add('bg-pink-600', 'text-white');
|
||||||
|
event.target.classList.remove('bg-gray-100', 'text-gray-600');
|
||||||
|
renderPhotos();
|
||||||
|
}
|
||||||
|
|
||||||
|
function openEdit(id, title, kat) {
|
||||||
|
document.getElementById('editId').value = id;
|
||||||
|
document.getElementById('editTitle').value = title;
|
||||||
|
document.getElementById('editKat').value = kat;
|
||||||
|
document.getElementById('editModal').classList.remove('hidden');
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeModal() {
|
||||||
|
document.getElementById('editModal').classList.add('hidden');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveEdit() {
|
||||||
|
const id = document.getElementById('editId').value;
|
||||||
|
const r = await fetch('/api/photo/' + id, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
title: document.getElementById('editTitle').value,
|
||||||
|
kategorie: document.getElementById('editKat').value
|
||||||
|
})
|
||||||
|
});
|
||||||
|
if (r.ok) {
|
||||||
|
closeModal();
|
||||||
|
showToast('✅ Gespeichert!');
|
||||||
|
loadPhotos();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deletePhoto(id) {
|
||||||
|
if (!confirm('Foto wirklich löschen? Das kann nicht rückgängig gemacht werden.')) return;
|
||||||
|
const r = await fetch('/api/photo/' + id, { method: 'DELETE' });
|
||||||
|
if (r.ok) {
|
||||||
|
showToast('🗑️ Foto gelöscht');
|
||||||
|
loadPhotos();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function showToast(msg) {
|
||||||
|
const t = document.getElementById('toast');
|
||||||
|
t.textContent = msg;
|
||||||
|
t.classList.remove('hidden');
|
||||||
|
setTimeout(() => t.classList.add('hidden'), 3000);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enter in Edit-Modal
|
||||||
|
document.addEventListener('keydown', e => {
|
||||||
|
if (e.key === 'Escape') closeModal();
|
||||||
|
if (e.key === 'Enter' && !document.getElementById('editModal').classList.contains('hidden')) saveEdit();
|
||||||
|
});
|
||||||
|
|
||||||
|
loadPhotos();
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>`);
|
||||||
|
});
|
||||||
|
|
||||||
|
app.listen(PORT, () => {
|
||||||
|
console.log(`\n✨ MiyaKarate Admin läuft auf http://localhost:${PORT}\n`);
|
||||||
|
});
|
||||||
1054
admin.html
Normal file
1054
admin.html
Normal file
File diff suppressed because it is too large
Load diff
5
archetypes/default.md
Normal file
5
archetypes/default.md
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
+++
|
||||||
|
date = '{{ .Date }}'
|
||||||
|
draft = true
|
||||||
|
title = '{{ replace .File.ContentBaseName "-" " " | title }}'
|
||||||
|
+++
|
||||||
3
content/erfolge/_index.md
Normal file
3
content/erfolge/_index.md
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
title: "Erfolge"
|
||||||
|
---
|
||||||
8
content/erfolge/landesmeisterschaft-2024.md
Normal file
8
content/erfolge/landesmeisterschaft-2024.md
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
---
|
||||||
|
title: "Landesmeisterschaft Berlin 2024"
|
||||||
|
rang: "Gold"
|
||||||
|
date: 2024-11-15
|
||||||
|
summary: "Erster Platz in der Kategorie Kata U14 bei der Berliner Landesmeisterschaft. Ein unvergesslicher Moment!"
|
||||||
|
---
|
||||||
|
|
||||||
|
Es war ein langer Weg bis zur Landesmeisterschaft. Monatelang haben wir jeden Tag trainiert, die Kata immer wieder verfeinert. Und dann dieser Moment auf der Matte...
|
||||||
8
content/erfolge/norddeutsche-2024.md
Normal file
8
content/erfolge/norddeutsche-2024.md
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
---
|
||||||
|
title: "Norddeutsche Meisterschaft 2024"
|
||||||
|
rang: "Silber"
|
||||||
|
date: 2024-08-20
|
||||||
|
summary: "Silbermedaille beim Norddeutschen Turnier – das bisher größte Turnier, an dem ich teilgenommen habe."
|
||||||
|
---
|
||||||
|
|
||||||
|
Über 200 Teilnehmer, drei Kampftage, und am Ende eine silberne Medaille um den Hals. Ein Riesenschritt für mich!
|
||||||
3
content/gaestebuch/_index.md
Normal file
3
content/gaestebuch/_index.md
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
title: "Gästebuch"
|
||||||
|
---
|
||||||
3
content/galerie/_index.md
Normal file
3
content/galerie/_index.md
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
title: "Galerie"
|
||||||
|
---
|
||||||
3
content/uebermich/_index.md
Normal file
3
content/uebermich/_index.md
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
title: "Über mich"
|
||||||
|
---
|
||||||
1
data/categories.json
Normal file
1
data/categories.json
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
["Training", "Wettkämpfe", "Gürtelprüfungen"]
|
||||||
28
data/gallery.json
Normal file
28
data/gallery.json
Normal file
|
|
@ -0,0 +1,28 @@
|
||||||
|
{
|
||||||
|
"photos": [
|
||||||
|
{
|
||||||
|
"id": "1775774640975-43j95",
|
||||||
|
"filename": "1775774640975-43j95.webp",
|
||||||
|
"thumb": "1775774640975-43j95-thumb.webp",
|
||||||
|
"title": "IMG_1106",
|
||||||
|
"kategorie": "Training",
|
||||||
|
"datum": "2026-04-09"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "1775774312130-1yqrc",
|
||||||
|
"filename": "1775774312130-1yqrc.webp",
|
||||||
|
"thumb": "1775774312130-1yqrc-thumb.webp",
|
||||||
|
"title": "Test",
|
||||||
|
"kategorie": "Wettkämpfe",
|
||||||
|
"datum": "2026-04-09"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "1775774187000-q74m0",
|
||||||
|
"filename": "1775774187000-q74m0.webp",
|
||||||
|
"thumb": "1775774187000-q74m0-thumb.webp",
|
||||||
|
"title": "Test",
|
||||||
|
"kategorie": "Training",
|
||||||
|
"datum": "2026-04-09"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
21
data/homepage.json
Normal file
21
data/homepage.json
Normal file
|
|
@ -0,0 +1,21 @@
|
||||||
|
{
|
||||||
|
"hero": {
|
||||||
|
"badge": "MEINE REISE IN KATA hh",
|
||||||
|
"description": "Entdecke die Welt des Karate durch meine Augen. Von den ersten Schritten im Dojo bis hin zu nationalen Meisterschaften – hier teile ich meine Leidenschaft für Bewegung, Disziplin und Erfolg. hh",
|
||||||
|
"image": "hero.webp"
|
||||||
|
},
|
||||||
|
"stats": [
|
||||||
|
{
|
||||||
|
"value": "7+",
|
||||||
|
"label": "Jahre Training"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"value": "1",
|
||||||
|
"label": "Gold Medaillen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"value": "11",
|
||||||
|
"label": "Turniere"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
22
deploy.sh
Executable file
22
deploy.sh
Executable file
|
|
@ -0,0 +1,22 @@
|
||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
SITE_DIR="$(cd "$(dirname "$0")" && pwd)"
|
||||||
|
SERVER="root@217.160.212.198"
|
||||||
|
SSH_KEY="/Users/jessi/.ssh/vpsserver/vpsserver"
|
||||||
|
REMOTE_DIR="/var/www/emy.bonzeipunk.de"
|
||||||
|
|
||||||
|
echo "▶ Hugo bauen..."
|
||||||
|
cd "$SITE_DIR"
|
||||||
|
hugo --minify
|
||||||
|
|
||||||
|
echo "▶ SSH-Key laden..."
|
||||||
|
SSH_ASKPASS_REQUIRE=never ssh-add "$SSH_KEY" <<< "bonzeikiller" 2>/dev/null || true
|
||||||
|
|
||||||
|
echo "▶ Auf Server übertragen..."
|
||||||
|
rsync -az --delete \
|
||||||
|
-e "ssh -o StrictHostKeyChecking=no -i $SSH_KEY" \
|
||||||
|
"$SITE_DIR/public/" "$SERVER:$REMOTE_DIR/"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "✓ Fertig! https://emy.bonzeipunk.de"
|
||||||
3
hugo.toml
Normal file
3
hugo.toml
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
baseURL = "https://emy.bonzeipunk.de/"
|
||||||
|
languageCode = "de"
|
||||||
|
title = "MiyaKarate"
|
||||||
16
layouts/_default/single.html
Normal file
16
layouts/_default/single.html
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta content="width=device-width, initial-scale=1.0" name="viewport"/>
|
||||||
|
<title>{{ .Title }} | MiyaKarate</title>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Lexend:wght@700;900&family=Be+Vietnam+Pro:wght@400;500&display=swap" rel="stylesheet"/>
|
||||||
|
<script src="https://cdn.tailwindcss.com"></script>
|
||||||
|
</head>
|
||||||
|
<body class="bg-gray-50 text-gray-800 font-sans p-8 max-w-3xl mx-auto">
|
||||||
|
<a href="/" class="text-pink-600 font-bold uppercase tracking-widest text-sm hover:underline">← Zurück</a>
|
||||||
|
<h1 class="text-5xl font-black mt-8 mb-4">{{ .Title }}</h1>
|
||||||
|
{{ with .Params.rang }}<span class="bg-purple-700 text-white px-4 py-1 rounded-full text-sm font-bold uppercase">{{ . }}</span>{{ end }}
|
||||||
|
<div class="mt-8 prose prose-lg">{{ .Content }}</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
243
layouts/erfolge/list.html
Normal file
243
layouts/erfolge/list.html
Normal file
|
|
@ -0,0 +1,243 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html class="scroll-smooth" lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta content="width=device-width, initial-scale=1.0" name="viewport"/>
|
||||||
|
<title>Erfolge | MiyaKarate</title>
|
||||||
|
<link href="https://fonts.googleapis.com" rel="preconnect"/>
|
||||||
|
<link crossorigin="" href="https://fonts.gstatic.com" rel="preconnect"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Lexend:wght@300;400;500;600;700;800;900&family=Be+Vietnam+Pro:wght@300;400;500;600;700&display=swap" rel="stylesheet"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&display=swap" rel="stylesheet"/>
|
||||||
|
<script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
|
||||||
|
<script id="tailwind-config">
|
||||||
|
tailwind.config = {
|
||||||
|
darkMode: "class",
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
"colors": {
|
||||||
|
"tertiary-container": "#ffffff", "surface-bright": "#f5f6f7",
|
||||||
|
"on-tertiary-container": "#636262", "error-dim": "#a70138",
|
||||||
|
"on-surface-variant": "#595c5d", "surface-tint": "#b30065",
|
||||||
|
"surface-container-highest": "#dadddf", "on-secondary-fixed-variant": "#7d21a4",
|
||||||
|
"inverse-on-surface": "#9b9d9e", "inverse-surface": "#0c0f10",
|
||||||
|
"surface-container": "#e6e8ea", "surface": "#f5f6f7",
|
||||||
|
"primary-fixed": "#ff6ea9", "surface-container-low": "#eff1f2",
|
||||||
|
"secondary": "#8930b0", "tertiary": "#5c5b5b",
|
||||||
|
"on-secondary": "#ffedff", "secondary-dim": "#7c20a3",
|
||||||
|
"secondary-container": "#f0c1ff", "surface-dim": "#d1d5d7",
|
||||||
|
"primary": "#b30065", "on-primary-fixed": "#000000",
|
||||||
|
"surface-container-high": "#e0e3e4", "primary-fixed-dim": "#ff4e9e",
|
||||||
|
"primary-dim": "#9d0058", "outline": "#757778",
|
||||||
|
"error": "#b41340", "outline-variant": "#abadae",
|
||||||
|
"secondary-fixed": "#f0c1ff", "secondary-fixed-dim": "#eaaeff",
|
||||||
|
"tertiary-fixed-dim": "#f3f0ef", "on-surface": "#2c2f30",
|
||||||
|
"on-tertiary-fixed-variant": "#6e6d6d", "on-tertiary": "#f5f2f1",
|
||||||
|
"tertiary-fixed": "#ffffff", "background": "#f5f6f7",
|
||||||
|
"on-primary": "#ffeff2", "surface-container-lowest": "#ffffff",
|
||||||
|
"inverse-primary": "#ff479c", "primary-container": "#ff6ea9",
|
||||||
|
"on-primary-fixed-variant": "#5c0031", "on-secondary-container": "#72129a",
|
||||||
|
"error-container": "#f74b6d", "surface-variant": "#dadddf",
|
||||||
|
"on-background": "#2c2f30", "on-error": "#ffefef",
|
||||||
|
"on-tertiary-fixed": "#515050", "tertiary-dim": "#504f4f",
|
||||||
|
"on-primary-container": "#4b0027", "on-secondary-fixed": "#580079",
|
||||||
|
"on-error-container": "#510017"
|
||||||
|
},
|
||||||
|
"borderRadius": {
|
||||||
|
"DEFAULT": "1rem", "lg": "2rem", "xl": "3rem", "full": "9999px"
|
||||||
|
},
|
||||||
|
"fontFamily": {
|
||||||
|
"headline": ["Lexend"], "body": ["Be Vietnam Pro"], "label": ["Be Vietnam Pro"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.material-symbols-outlined { font-variation-settings: 'FILL' 0, 'wght' 400, 'GRAD' 0, 'opsz' 24; }
|
||||||
|
.text-glass-gradient {
|
||||||
|
background: linear-gradient(135deg, #b30065, #ff6ea9);
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
-webkit-text-fill-color: transparent;
|
||||||
|
}
|
||||||
|
.editorial-shadow { box-shadow: 0 24px 40px -10px rgba(44, 47, 48, 0.06); }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="bg-surface font-body text-on-surface">
|
||||||
|
|
||||||
|
<!-- TopAppBar -->
|
||||||
|
<header class="bg-white/70 dark:bg-zinc-900/70 backdrop-blur-lg fixed top-0 left-0 w-full z-50 shadow-xl shadow-pink-500/5">
|
||||||
|
<div class="flex justify-between items-center h-20 px-6 md:px-12 max-w-screen-2xl mx-auto">
|
||||||
|
<a href="/" class="text-2xl font-black text-pink-600 dark:text-pink-400 italic font-headline tracking-tight uppercase">MiyaKarate</a>
|
||||||
|
<nav class="hidden md:flex items-center space-x-8 font-headline tracking-tight uppercase font-bold">
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/">Home</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/galerie/">Galerie</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/uebermich/">Über mich</a>
|
||||||
|
<a class="text-pink-600 dark:text-pink-400 border-b-2 border-pink-500 pb-1" href="/erfolge/">Erfolge</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/gaestebuch/">Gästebuch</a>
|
||||||
|
</nav>
|
||||||
|
<button class="bg-gradient-to-br from-primary to-primary-container text-on-primary px-8 py-3 rounded-xl font-bold font-headline uppercase tracking-wider scale-95 active:scale-90 transition-transform">
|
||||||
|
Kontakt
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main class="pt-32 pb-20">
|
||||||
|
<!-- Hero -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 mb-24">
|
||||||
|
<div class="grid grid-cols-1 lg:grid-cols-12 gap-12 items-center">
|
||||||
|
<div class="lg:col-span-7">
|
||||||
|
<span class="text-primary font-bold tracking-[0.2em] uppercase text-sm mb-4 block">Meine Wettkampf-Geschichte</span>
|
||||||
|
<h1 class="text-6xl md:text-8xl font-headline font-extrabold text-on-surface leading-[0.9] tracking-tighter mb-8">
|
||||||
|
JEDER SIEG <span class="text-glass-gradient">ZÄHLT.</span>
|
||||||
|
</h1>
|
||||||
|
<p class="text-xl text-on-surface-variant leading-relaxed max-w-xl">
|
||||||
|
Von der ersten Gürtelprüfung bis zur Landesmeisterschaft – hier sammle ich alle Momente, auf die ich besonders stolz bin.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="lg:col-span-5 relative">
|
||||||
|
<div class="aspect-[4/5] rounded-xl overflow-hidden editorial-shadow bg-surface-container-highest">
|
||||||
|
<img class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuD_gAlNnXIZRs_NnR68igfJUfHX1ueNZlrjMt15L5xNz3bnL4235YADrLG-nT3wz2ZWYEZy6Dom8-fgsolN77eu_0JF52Xp-YjWEre5kwxN2D6V5LAoXI_T8I2YSU6LI5ZybF1Y1Ynqp8Y2IzCh0DYZOqY_tlPOuzsExGMn7nO0jWw58sR7Ny1674begJzhSNxELjEE7oQfgLjSZaO17dv1vGG5LykvtL9NEqM2JXdOVh0SlsnUN4416utgLZpV8km6wUza_TY7Fg"
|
||||||
|
alt="Miya beim Wettkampf"/>
|
||||||
|
</div>
|
||||||
|
<div class="absolute -bottom-10 -left-10 bg-white p-8 rounded-lg editorial-shadow hidden md:block">
|
||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<div class="w-12 h-12 rounded-full bg-secondary flex items-center justify-center text-on-secondary">
|
||||||
|
<span class="material-symbols-outlined" style="font-variation-settings: 'FILL' 1;">workspace_premium</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div class="text-sm font-bold uppercase tracking-widest text-secondary">Aktueller Rang</div>
|
||||||
|
<div class="text-2xl font-headline font-black text-on-surface">Blaugurt</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Alle Erfolge aus Content-Dateien -->
|
||||||
|
<section class="bg-surface-container-low py-24 rounded-t-[5rem]">
|
||||||
|
<div class="max-w-7xl mx-auto px-6">
|
||||||
|
<h2 class="text-4xl font-headline font-bold text-on-surface mb-16">Alle Erfolge</h2>
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-8">
|
||||||
|
{{ range (where .Site.RegularPages "Section" "erfolge") }}
|
||||||
|
<div class="bg-surface-container-lowest rounded-xl p-8 editorial-shadow flex flex-col justify-between relative overflow-hidden group hover:shadow-xl transition-all duration-500">
|
||||||
|
<div class="relative z-10">
|
||||||
|
<div class="flex items-center gap-4 mb-4">
|
||||||
|
<span class="material-symbols-outlined text-3xl text-primary" style="font-variation-settings: 'FILL' 1;">military_tech</span>
|
||||||
|
<span class="bg-secondary text-white px-4 py-1 rounded-full text-xs font-bold uppercase tracking-widest">{{ .Params.rang | default "Highlight" }}</span>
|
||||||
|
</div>
|
||||||
|
<h3 class="text-2xl font-headline font-bold text-on-surface mb-3">{{ .Title }}</h3>
|
||||||
|
<p class="text-on-surface-variant mb-6">{{ .Summary }}</p>
|
||||||
|
<a class="flex items-center gap-2 text-primary font-bold hover:gap-4 transition-all" href="{{ .RelPermalink }}">
|
||||||
|
Bericht lesen <span class="material-symbols-outlined">arrow_forward</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<span class="material-symbols-outlined absolute -bottom-10 -right-10 text-[12rem] text-surface-container opacity-10 group-hover:opacity-20 transition-opacity">sports_martial_arts</span>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Meilensteine Bento Grid -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 mt-24">
|
||||||
|
<h2 class="text-5xl font-headline font-black mb-16 tracking-tight">MEILENSTEINE DES <span class="text-primary italic">ERFOLGS</span></h2>
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-3 gap-8">
|
||||||
|
<!-- Große Karte -->
|
||||||
|
<div class="md:col-span-2 bg-surface-container-lowest rounded-xl p-10 editorial-shadow flex flex-col justify-between relative overflow-hidden group">
|
||||||
|
<div class="relative z-10">
|
||||||
|
<div class="flex items-center gap-4 mb-6">
|
||||||
|
<span class="material-symbols-outlined text-4xl text-primary" style="font-variation-settings: 'FILL' 1;">military_tech</span>
|
||||||
|
<span class="font-headline font-extrabold text-2xl uppercase italic">Landesmeisterschaft Berlin</span>
|
||||||
|
</div>
|
||||||
|
<h3 class="text-4xl font-headline font-bold text-on-surface mb-4">Gold – Kata U14</h3>
|
||||||
|
<p class="text-on-surface-variant max-w-md text-lg">Mein bisher größter Erfolg. Monatelanges hartes Training gipfelte in einer Vorführung, die meinen Wettkampfgeist unter Beweis stellte.</p>
|
||||||
|
</div>
|
||||||
|
<div class="mt-12 flex items-center gap-6 relative z-10">
|
||||||
|
<div>
|
||||||
|
<div class="text-3xl font-black text-on-surface">2024</div>
|
||||||
|
<div class="text-xs uppercase tracking-widest font-bold text-primary">Berlin</div>
|
||||||
|
</div>
|
||||||
|
<div class="h-10 w-[1px] bg-outline-variant/30"></div>
|
||||||
|
<div>
|
||||||
|
<div class="text-3xl font-black text-on-surface">U14</div>
|
||||||
|
<div class="text-xs uppercase tracking-widest font-bold text-primary">Kategorie</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<span class="material-symbols-outlined absolute -bottom-10 -right-10 text-[15rem] text-surface-container opacity-20 group-hover:opacity-30 transition-opacity">sports_martial_arts</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Weitere Auszeichnungen -->
|
||||||
|
<div class="bg-primary rounded-xl p-8 text-on-primary flex flex-col gap-8 shadow-2xl shadow-primary/20">
|
||||||
|
<h4 class="font-headline font-bold text-xl border-b border-on-primary/20 pb-4">Weitere Auszeichnungen</h4>
|
||||||
|
<div class="flex gap-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-white/20 flex items-center justify-center shrink-0">
|
||||||
|
<span class="material-symbols-outlined text-xl">star</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Norddeutsche Meisterschaft</p>
|
||||||
|
<p class="text-sm opacity-80">Silber – Kata 2024</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-white/20 flex items-center justify-center shrink-0">
|
||||||
|
<span class="material-symbols-outlined text-xl">emoji_events</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Dojo Champion</p>
|
||||||
|
<p class="text-sm opacity-80">Kiai Berlin, intern</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-white/20 flex items-center justify-center shrink-0">
|
||||||
|
<span class="material-symbols-outlined text-xl">rewarded_ads</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Fairness-Preis</p>
|
||||||
|
<p class="text-sm opacity-80">Dojo-Auszeichnung 2023</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Stats -->
|
||||||
|
<div class="bg-secondary-container rounded-xl p-8 flex flex-col justify-center gap-2">
|
||||||
|
<span class="text-on-secondary-container font-headline font-black text-6xl italic">5+</span>
|
||||||
|
<span class="text-secondary font-bold uppercase tracking-[0.2em] text-sm">Jahre Training</span>
|
||||||
|
</div>
|
||||||
|
<div class="bg-surface-container-highest rounded-xl p-8 flex flex-col justify-center gap-2">
|
||||||
|
<span class="text-on-surface font-headline font-black text-6xl italic">32</span>
|
||||||
|
<span class="text-on-surface-variant font-bold uppercase tracking-[0.2em] text-sm">Turniere</span>
|
||||||
|
</div>
|
||||||
|
<div class="bg-white rounded-xl p-8 editorial-shadow flex flex-col justify-center gap-2 border-t-4 border-pink-500">
|
||||||
|
<span class="text-pink-600 font-headline font-black text-6xl italic">12</span>
|
||||||
|
<span class="text-zinc-500 font-bold uppercase tracking-[0.2em] text-sm">Goldmedaillen</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Zitat -->
|
||||||
|
<section class="max-w-4xl mx-auto px-6 py-32 text-center">
|
||||||
|
<span class="material-symbols-outlined text-primary text-6xl mb-8" style="font-variation-settings: 'FILL' 1;">format_quote</span>
|
||||||
|
<blockquote class="text-3xl md:text-5xl font-headline font-bold text-on-surface leading-tight tracking-tight italic">
|
||||||
|
"Karate ist nicht nur ein Sport. Es ist eine Linse, durch die ich die Welt sehe – mit Präzision, Respekt und unbeugsamer Konzentration."
|
||||||
|
</blockquote>
|
||||||
|
<p class="mt-8 font-bold uppercase tracking-widest text-on-surface-variant">— Miya</p>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- Footer -->
|
||||||
|
<footer class="bg-zinc-50 dark:bg-zinc-950 w-full rounded-t-[3rem] mt-20">
|
||||||
|
<div class="flex flex-col md:flex-row justify-between items-center py-12 px-8 max-w-7xl mx-auto gap-6 text-sm tracking-wide">
|
||||||
|
<div class="text-lg font-bold text-zinc-900 dark:text-zinc-100 font-headline uppercase italic">MiyaKarate</div>
|
||||||
|
<div class="flex gap-8 text-zinc-500">
|
||||||
|
<a class="hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity" href="#">Instagram</a>
|
||||||
|
<a class="hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity" href="#">YouTube</a>
|
||||||
|
<a class="hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity" href="#">Email</a>
|
||||||
|
</div>
|
||||||
|
<div class="text-zinc-500 opacity-80">© 2024 MiyaKarate. Alle Rechte vorbehalten.</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
299
layouts/gaestebuch/list.html
Normal file
299
layouts/gaestebuch/list.html
Normal file
|
|
@ -0,0 +1,299 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html class="scroll-smooth" lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta content="width=device-width, initial-scale=1.0" name="viewport"/>
|
||||||
|
<title>Gästebuch | MiyaKarate</title>
|
||||||
|
<script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Lexend:wght@400;700;800;900&family=Be+Vietnam+Pro:wght@400;500;600;700&display=swap" rel="stylesheet"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&display=swap" rel="stylesheet"/>
|
||||||
|
<script id="tailwind-config">
|
||||||
|
tailwind.config = {
|
||||||
|
darkMode: "class",
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
"colors": {
|
||||||
|
"tertiary-container": "#ffffff", "surface-bright": "#f5f6f7",
|
||||||
|
"on-tertiary-container": "#636262", "error-dim": "#a70138",
|
||||||
|
"on-surface-variant": "#595c5d", "surface-tint": "#b30065",
|
||||||
|
"surface-container-highest": "#dadddf", "on-secondary-fixed-variant": "#7d21a4",
|
||||||
|
"inverse-on-surface": "#9b9d9e", "inverse-surface": "#0c0f10",
|
||||||
|
"surface-container": "#e6e8ea", "surface": "#f5f6f7",
|
||||||
|
"primary-fixed": "#ff6ea9", "surface-container-low": "#eff1f2",
|
||||||
|
"secondary": "#8930b0", "tertiary": "#5c5b5b",
|
||||||
|
"on-secondary": "#ffedff", "secondary-dim": "#7c20a3",
|
||||||
|
"secondary-container": "#f0c1ff", "surface-dim": "#d1d5d7",
|
||||||
|
"primary": "#b30065", "on-primary-fixed": "#000000",
|
||||||
|
"surface-container-high": "#e0e3e4", "primary-fixed-dim": "#ff4e9e",
|
||||||
|
"primary-dim": "#9d0058", "outline": "#757778",
|
||||||
|
"error": "#b41340", "outline-variant": "#abadae",
|
||||||
|
"secondary-fixed": "#f0c1ff", "secondary-fixed-dim": "#eaaeff",
|
||||||
|
"tertiary-fixed-dim": "#f3f0ef", "on-surface": "#2c2f30",
|
||||||
|
"on-tertiary-fixed-variant": "#6e6d6d", "on-tertiary": "#f5f2f1",
|
||||||
|
"tertiary-fixed": "#ffffff", "background": "#f5f6f7",
|
||||||
|
"on-primary": "#ffeff2", "surface-container-lowest": "#ffffff",
|
||||||
|
"inverse-primary": "#ff479c", "primary-container": "#ff6ea9",
|
||||||
|
"on-primary-fixed-variant": "#5c0031", "on-secondary-container": "#72129a",
|
||||||
|
"error-container": "#f74b6d", "surface-variant": "#dadddf",
|
||||||
|
"on-background": "#2c2f30", "on-error": "#ffefef",
|
||||||
|
"on-tertiary-fixed": "#515050", "tertiary-dim": "#504f4f",
|
||||||
|
"on-primary-container": "#4b0027", "on-secondary-fixed": "#580079",
|
||||||
|
"on-error-container": "#510017"
|
||||||
|
},
|
||||||
|
"borderRadius": {
|
||||||
|
"DEFAULT": "1rem", "lg": "2rem", "xl": "3rem", "full": "9999px"
|
||||||
|
},
|
||||||
|
"fontFamily": {
|
||||||
|
"headline": ["Lexend"], "body": ["Be Vietnam Pro"], "label": ["Be Vietnam Pro"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.material-symbols-outlined { font-variation-settings: 'FILL' 0, 'wght' 400, 'GRAD' 0, 'opsz' 24; }
|
||||||
|
.editorial-text { text-wrap: balance; }
|
||||||
|
.glass-card { background: rgba(255, 255, 255, 0.7); backdrop-filter: blur(20px); }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="bg-surface text-on-surface font-body selection:bg-primary-container selection:text-on-primary-container">
|
||||||
|
|
||||||
|
<!-- TopAppBar -->
|
||||||
|
<header class="fixed top-0 left-0 w-full z-50 bg-white/70 dark:bg-zinc-900/70 backdrop-blur-lg shadow-xl shadow-pink-500/5">
|
||||||
|
<nav class="flex justify-between items-center h-20 px-6 md:px-12 max-w-screen-2xl mx-auto">
|
||||||
|
<a href="/" class="text-2xl font-black text-pink-600 dark:text-pink-400 italic font-headline tracking-tight uppercase">MiyaKarate</a>
|
||||||
|
<div class="hidden md:flex items-center gap-8">
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/">Home</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/galerie/">Galerie</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/uebermich/">Über mich</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="#">Erfolge</a>
|
||||||
|
<a class="text-pink-600 dark:text-pink-400 border-b-2 border-pink-500 pb-1 font-headline tracking-tight uppercase font-bold" href="/gaestebuch/">Gästebuch</a>
|
||||||
|
</div>
|
||||||
|
<button class="bg-gradient-to-br from-primary to-primary-container text-on-primary px-8 py-3 rounded-xl font-bold uppercase tracking-wider scale-95 active:scale-90 transition-transform shadow-lg shadow-primary/20">
|
||||||
|
Kontakt
|
||||||
|
</button>
|
||||||
|
</nav>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main class="pt-32 pb-20">
|
||||||
|
<!-- Hero -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 mb-20">
|
||||||
|
<div class="relative rounded-xl overflow-hidden min-h-[400px] flex items-center p-8 md:p-16">
|
||||||
|
<img alt="Karate Dojo" class="absolute inset-0 w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuDlVHSCQA9wTqwkVgz4Q76piwsWS33jFlIXlTw7TBSDubv3dQTnKkIUtU-rBAoPyO3eJ1BHdeKBNqS_CCcyhfI0XH1VHSDFSgxgH1oDxrICcX8b-GoehyY1x8udqwWfuKtSTI-nNW4i15l0HWDcVvSrtlM4u25KjM63nBbo_pfDq6f-larFS9uVzVlBn5KMcj8uioIHmQqPlVxWqlkETOL_6zr_Y-k_doAHs6brW-DdT4clT2MxJjtDFjxxr-cTzgKALEfMvTTJUg"/>
|
||||||
|
<div class="absolute inset-0 bg-gradient-to-r from-on-surface/80 via-on-surface/40 to-transparent"></div>
|
||||||
|
<div class="relative z-10 max-w-2xl">
|
||||||
|
<span class="inline-block px-4 py-1 rounded-full bg-secondary text-on-secondary font-label text-xs tracking-[0.1em] uppercase mb-4">Melde dich bei mir</span>
|
||||||
|
<h1 class="font-headline text-5xl md:text-7xl font-extrabold text-white leading-tight mb-6 tracking-tighter">
|
||||||
|
Schreib mir & <span class="text-primary-container">teile deine Geschichte</span>
|
||||||
|
</h1>
|
||||||
|
<p class="text-lg text-surface font-medium max-w-lg editorial-text">
|
||||||
|
Ob du eine Frage zu meinem Training hast oder einfach eine nette Nachricht hinterlassen möchtest – ich freue mich von dir zu hören!
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Bento: Formular + Sidebar -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 grid grid-cols-1 lg:grid-cols-12 gap-8">
|
||||||
|
<!-- Kontaktformular -->
|
||||||
|
<div class="lg:col-span-7 bg-surface-container-lowest rounded-xl p-8 md:p-12 shadow-sm relative overflow-hidden">
|
||||||
|
<div class="absolute top-0 right-0 w-64 h-64 bg-primary/5 rounded-full -mr-32 -mt-32"></div>
|
||||||
|
<div class="relative z-10">
|
||||||
|
<h2 class="font-headline text-3xl font-bold mb-8 flex items-center gap-3">
|
||||||
|
<span class="material-symbols-outlined text-primary text-4xl">send</span>
|
||||||
|
Nachricht schicken
|
||||||
|
</h2>
|
||||||
|
<form class="space-y-6">
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-6">
|
||||||
|
<div class="space-y-2">
|
||||||
|
<label class="text-sm font-bold font-label uppercase tracking-widest text-on-surface-variant px-1">Dein Name</label>
|
||||||
|
<input class="w-full bg-surface-container-low border-none rounded-sm focus:ring-2 focus:ring-primary/20 p-4 transition-all placeholder:text-outline-variant" placeholder="Mia Muster" type="text"/>
|
||||||
|
</div>
|
||||||
|
<div class="space-y-2">
|
||||||
|
<label class="text-sm font-bold font-label uppercase tracking-widest text-on-surface-variant px-1">E-Mail-Adresse</label>
|
||||||
|
<input class="w-full bg-surface-container-low border-none rounded-sm focus:ring-2 focus:ring-primary/20 p-4 transition-all placeholder:text-outline-variant" placeholder="hallo@beispiel.de" type="email"/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="space-y-2">
|
||||||
|
<label class="text-sm font-bold font-label uppercase tracking-widest text-on-surface-variant px-1">Betreff</label>
|
||||||
|
<div class="flex flex-wrap gap-3 py-2">
|
||||||
|
<label class="cursor-pointer">
|
||||||
|
<input checked class="hidden peer" name="subject" type="radio"/>
|
||||||
|
<span class="px-6 py-2 rounded-full border-2 border-surface-container-high peer-checked:bg-secondary peer-checked:text-on-secondary peer-checked:border-secondary transition-all text-sm font-bold">Allgemeine Frage</span>
|
||||||
|
</label>
|
||||||
|
<label class="cursor-pointer">
|
||||||
|
<input class="hidden peer" name="subject" type="radio"/>
|
||||||
|
<span class="px-6 py-2 rounded-full border-2 border-surface-container-high peer-checked:bg-secondary peer-checked:text-on-secondary peer-checked:border-secondary transition-all text-sm font-bold">Trainingstipps</span>
|
||||||
|
</label>
|
||||||
|
<label class="cursor-pointer">
|
||||||
|
<input class="hidden peer" name="subject" type="radio"/>
|
||||||
|
<span class="px-6 py-2 rounded-full border-2 border-surface-container-high peer-checked:bg-secondary peer-checked:text-on-secondary peer-checked:border-secondary transition-all text-sm font-bold">Sonstiges</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="space-y-2">
|
||||||
|
<label class="text-sm font-bold font-label uppercase tracking-widest text-on-surface-variant px-1">Nachricht</label>
|
||||||
|
<textarea class="w-full bg-surface-container-low border-none rounded-sm focus:ring-2 focus:ring-primary/20 p-4 transition-all placeholder:text-outline-variant resize-none" placeholder="Schreib deine Nachricht hier..." rows="5"></textarea>
|
||||||
|
</div>
|
||||||
|
<button class="w-full bg-gradient-to-r from-primary to-primary-container text-on-primary py-5 rounded-xl font-headline text-xl font-bold shadow-xl shadow-primary/30 transition-transform active:scale-95">
|
||||||
|
Nachricht absenden
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Sidebar -->
|
||||||
|
<div class="lg:col-span-5 flex flex-col gap-8">
|
||||||
|
<div class="bg-secondary-container text-on-secondary-container rounded-xl p-8 shadow-sm">
|
||||||
|
<h3 class="font-headline text-2xl font-bold mb-4">Direktkontakt</h3>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<div class="w-12 h-12 rounded-full bg-on-secondary-container/10 flex items-center justify-center">
|
||||||
|
<span class="material-symbols-outlined">alternate_email</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="text-xs font-bold uppercase opacity-60">E-Mail</p>
|
||||||
|
<p class="font-bold">hallo@miyakarate.de</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<div class="w-12 h-12 rounded-full bg-on-secondary-container/10 flex items-center justify-center">
|
||||||
|
<span class="material-symbols-outlined">location_on</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="text-xs font-bold uppercase opacity-60">Dojo</p>
|
||||||
|
<p class="font-bold">Kiai Dojo Berlin</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Neuester Eintrag -->
|
||||||
|
<div class="bg-surface-container-low rounded-xl p-8 flex-grow">
|
||||||
|
<div class="flex justify-between items-end mb-8">
|
||||||
|
<h3 class="font-headline text-2xl font-bold">Neuester Eintrag</h3>
|
||||||
|
<a class="text-primary font-bold text-sm hover:underline" href="#gaestebuch-wall">Alle ansehen</a>
|
||||||
|
</div>
|
||||||
|
<div class="glass-card p-6 rounded-lg shadow-sm border border-white/40">
|
||||||
|
<div class="flex items-center gap-4 mb-4">
|
||||||
|
<div class="w-12 h-12 rounded-full bg-surface-container-highest overflow-hidden">
|
||||||
|
<img alt="Gast Avatar" class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuCyr8FcSn01QbwNHFaE4oEJCeSKkIcv6Qjd7WO_jrfclu0LeV9KMmCJ6VqttBmEZM9Cz7ddfK6gmQ8qGQmJ9YWno683_KcaFT_BhaOws3_rpcvPU2j1xquZD0bzluyw84dqcvLLIXsyqWVYMGqlAwXoW78olS5BfpuvMiBkh3bIIgo8kIhWJkq7R1MZB4yCmnTN2UyzrVSy9aUM_vmaU1AZFhWUgk7UOsN1BST2xOACH2NhCB6HP_93OnTDjAAXBja3bufXiAjZkw"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h4 class="font-bold">Jonas L.</h4>
|
||||||
|
<p class="text-xs text-on-surface-variant font-medium">vor 2 Stunden</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p class="text-on-surface-variant leading-relaxed">"Mega Fortschritte bei deiner letzten Kata! Deine Hingabe ist wirklich inspirierend. Weiter so! Oss!"</p>
|
||||||
|
<div class="mt-4 flex gap-2">
|
||||||
|
<span class="px-3 py-1 rounded-full bg-primary/10 text-primary text-[10px] font-bold uppercase">Teamkamerad</span>
|
||||||
|
<span class="px-3 py-1 rounded-full bg-secondary/10 text-secondary text-[10px] font-bold uppercase">Inspiriert</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Gästebuch-Wand -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 mt-20" id="gaestebuch-wall">
|
||||||
|
<div class="text-center mb-12">
|
||||||
|
<h2 class="font-headline text-4xl md:text-5xl font-extrabold mb-4">Gästebuch-Wand</h2>
|
||||||
|
<p class="text-on-surface-variant max-w-2xl mx-auto">Nachrichten von Trainingspartnern, Familie und Freunden aus aller Welt.</p>
|
||||||
|
</div>
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
|
||||||
|
<div class="bg-white rounded-lg p-6 shadow-xl shadow-black/5 hover:-translate-y-2 transition-transform duration-300">
|
||||||
|
<div class="flex items-center gap-3 mb-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-surface-container overflow-hidden">
|
||||||
|
<img alt="Gast" class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuAxxSG4TUDr606KVR8_FBXcG1vZu-IryVXVsyk0ildPn-mTI7Xhf3_kclVMF91JkR_jx9eWesri4dDj4vf9U7gKuzWCMe6W52Bwgndg_mOwNf6WgregdNT9uH1bwxH2reR5LR_M8Stb6dLfl3O1u0qC3l8W_jcpH2FrTQY-onMUIiasTQNndMFCSqJy2Y8NBCjJkq7eSFELqR7CicXvpe3dSLeE_WyzxADxwlGyY3tPS7nYZuYCJvctYZl0hDvGF1HYDhKj3Xw5rg"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold text-sm">Trainerin Elena</p>
|
||||||
|
<p class="text-[10px] uppercase font-bold text-outline">Sensei</p>
|
||||||
|
</div>
|
||||||
|
<span class="ml-auto material-symbols-outlined text-primary text-xl" style="font-variation-settings: 'FILL' 1;">favorite</span>
|
||||||
|
</div>
|
||||||
|
<p class="text-sm text-on-surface-variant leading-relaxed">Deine Konzentration beim letzten Grading war außergewöhnlich. Die Körperkontrolle, die du entwickelst, ist selten für dein Alter. Ich bin stolz auf dich!</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="bg-white rounded-lg p-6 shadow-xl shadow-black/5 hover:-translate-y-2 transition-transform duration-300">
|
||||||
|
<div class="flex items-center gap-3 mb-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-surface-container overflow-hidden">
|
||||||
|
<img alt="Gast" class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuCcEhcufxMA4pz_Ukcsfi9JIzXT6dS8M_1HuhSOhs00ftkIx_Dm74awVwHZURNxLYx0JSSosudUn5s21X0gOha6KwIsVz-esI6HO9WqQcgRm3Y6rxuM0eL2QbJtx16wJdxq9y05JLIjBpLGEQB7iemBIX7lH6RF86cxV0Cq0I_KA_vTkcLwxPunt79vO6NUt-zb0v3PP59gVEHOJOfFnnOlRqTe42kS28CfXGeWY5920CQDOjHFpQ6SH45XSQPSKMICsP5ANa3Wdw"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold text-sm">Opa Karl</p>
|
||||||
|
<p class="text-[10px] uppercase font-bold text-outline">Familie</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p class="text-sm text-on-surface-variant leading-relaxed">Wir drücken dir von zuhause alle die Daumen! Kann es kaum erwarten, den nächsten Pokal auf dem Regal zu sehen!</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="bg-white rounded-lg p-6 shadow-xl shadow-black/5 hover:-translate-y-2 transition-transform duration-300">
|
||||||
|
<div class="flex items-center gap-3 mb-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-surface-container overflow-hidden">
|
||||||
|
<img alt="Gast" class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuCx78Ehwgx2xoiA9H_WHtwOteg8d8slllmcu2c66Ty-RnDNmX6sUEtNNsUXBXtINwZ4tg8WGxUjfbOSbtBWbcAUlaKqTTGgVOicODmvzt6PsT-hNu-JvwoOsmr-MSza1orgDZlUWKAcCY1XSByhyrJ6pFRT964VljwFvJd9KUKJFtd18GKw5qqh5IqcTt9_jrZFtOMFAGio1uK1GT3pyJA3WyGYMyUAbUWucPVTcnB9qo44WUkfg4k4LcmL7soUZK4bvTGxLg_m-g"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold text-sm">Lea M.</p>
|
||||||
|
<p class="text-[10px] uppercase font-bold text-outline">Freundin</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p class="text-sm text-on-surface-variant leading-relaxed">Die Website sieht so professionell aus! Mega cool, alle deine Erfolge an einem Ort zu sehen. Wir müssen bald mal wieder eis essen gehen!</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="bg-gradient-to-br from-secondary/5 to-primary/5 border border-white rounded-lg p-6 shadow-xl shadow-black/5 md:col-span-2">
|
||||||
|
<div class="flex items-center gap-4 mb-4">
|
||||||
|
<div class="w-12 h-12 rounded-full bg-secondary text-on-secondary flex items-center justify-center font-bold text-xl">T</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Trainer Thomas</p>
|
||||||
|
<p class="text-xs font-medium text-secondary">Dojo Kiai Berlin</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<blockquote class="text-on-surface-variant italic text-lg leading-relaxed mb-4">
|
||||||
|
"Ich trainiere viele Kinder, aber die kinetische Flüssigkeit in deinen Bewegungen ist etwas Besonderes. Du machst jede Kata zu einem Tanz. Weiter so – der nächste schwarze Gürtel wartet schon!"
|
||||||
|
</blockquote>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="bg-white rounded-lg p-6 shadow-xl shadow-black/5 hover:-translate-y-2 transition-transform duration-300">
|
||||||
|
<div class="flex items-center gap-3 mb-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-surface-container overflow-hidden">
|
||||||
|
<img alt="Gast" class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuA09PdHK61xpLhrsPtonFk8dj5CKIW26fygdA6IlzV7oS96jGwscnrGsFu947MCZ2oPdbWP8STBciRmN8FDLKPOdHJKd2dBi3-pwlmDsyDAU8Myi4GtGEQLay6v7P_DOoRN64fPfZcEaXRNrje0eYe-DyAiDAXCRfqYVwK_ducZzu4e-ZOQIyXaJRA1LtJwkiF1EHDj2Kcp4z9TmQGWpc9HtLA7D7uQ7XcncDH-HSYLS8ZR3rC4h90gf9_CJfzUfHC8MSeSdQKABg"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold text-sm">Herr Braun</p>
|
||||||
|
<p class="text-[10px] uppercase font-bold text-outline">Lehrer</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p class="text-sm text-on-surface-variant leading-relaxed">Balance im Sport führt zu Balance im Leben. Halt die Disziplin, die du im Dojo zeigst, auch in der Schule!</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mt-12 text-center">
|
||||||
|
<button class="bg-surface-container-highest hover:bg-surface-container-high px-10 py-4 rounded-xl font-bold transition-colors">
|
||||||
|
Weitere Einträge laden
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- Footer -->
|
||||||
|
<footer class="w-full rounded-t-[3rem] mt-20 bg-zinc-50 dark:bg-zinc-950">
|
||||||
|
<div class="flex flex-col md:flex-row justify-between items-center py-12 px-8 max-w-7xl mx-auto gap-6 text-sm tracking-wide">
|
||||||
|
<div class="text-lg font-bold text-zinc-900 dark:text-zinc-100 font-headline uppercase italic">MiyaKarate</div>
|
||||||
|
<div class="flex gap-8 text-zinc-500">
|
||||||
|
<a class="hover:text-pink-500 transition-opacity hover:underline decoration-pink-500 decoration-2 underline-offset-4" href="#">Instagram</a>
|
||||||
|
<a class="hover:text-pink-500 transition-opacity hover:underline decoration-pink-500 decoration-2 underline-offset-4" href="#">YouTube</a>
|
||||||
|
<a class="hover:text-pink-500 transition-opacity hover:underline decoration-pink-500 decoration-2 underline-offset-4" href="#">Email</a>
|
||||||
|
</div>
|
||||||
|
<div class="text-zinc-500">© 2024 MiyaKarate. Alle Rechte vorbehalten.</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
247
layouts/galerie/list.html
Normal file
247
layouts/galerie/list.html
Normal file
|
|
@ -0,0 +1,247 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html class="scroll-smooth" lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta content="width=device-width, initial-scale=1.0" name="viewport"/>
|
||||||
|
<title>Galerie | MiyaKarate</title>
|
||||||
|
<script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Lexend:wght@300;400;600;700;800;900&family=Be+Vietnam+Pro:wght@300;400;500;600;700&display=swap" rel="stylesheet"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&display=swap" rel="stylesheet"/>
|
||||||
|
<script id="tailwind-config">
|
||||||
|
tailwind.config = {
|
||||||
|
darkMode: "class",
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
"colors": {
|
||||||
|
"tertiary-container": "#ffffff", "surface-bright": "#f5f6f7",
|
||||||
|
"on-tertiary-container": "#636262", "error-dim": "#a70138",
|
||||||
|
"on-surface-variant": "#595c5d", "surface-tint": "#b30065",
|
||||||
|
"surface-container-highest": "#dadddf", "on-secondary-fixed-variant": "#7d21a4",
|
||||||
|
"inverse-on-surface": "#9b9d9e", "inverse-surface": "#0c0f10",
|
||||||
|
"surface-container": "#e6e8ea", "surface": "#f5f6f7",
|
||||||
|
"primary-fixed": "#ff6ea9", "surface-container-low": "#eff1f2",
|
||||||
|
"secondary": "#8930b0", "tertiary": "#5c5b5b",
|
||||||
|
"on-secondary": "#ffedff", "secondary-dim": "#7c20a3",
|
||||||
|
"secondary-container": "#f0c1ff", "surface-dim": "#d1d5d7",
|
||||||
|
"primary": "#b30065", "on-primary-fixed": "#000000",
|
||||||
|
"surface-container-high": "#e0e3e4", "primary-fixed-dim": "#ff4e9e",
|
||||||
|
"primary-dim": "#9d0058", "outline": "#757778",
|
||||||
|
"error": "#b41340", "outline-variant": "#abadae",
|
||||||
|
"secondary-fixed": "#f0c1ff", "secondary-fixed-dim": "#eaaeff",
|
||||||
|
"tertiary-fixed-dim": "#f3f0ef", "on-surface": "#2c2f30",
|
||||||
|
"on-tertiary-fixed-variant": "#6e6d6d", "on-tertiary": "#f5f2f1",
|
||||||
|
"tertiary-fixed": "#ffffff", "background": "#f5f6f7",
|
||||||
|
"on-primary": "#ffeff2", "surface-container-lowest": "#ffffff",
|
||||||
|
"inverse-primary": "#ff479c", "primary-container": "#ff6ea9",
|
||||||
|
"on-primary-fixed-variant": "#5c0031", "on-secondary-container": "#72129a",
|
||||||
|
"error-container": "#f74b6d", "surface-variant": "#dadddf",
|
||||||
|
"on-background": "#2c2f30", "on-error": "#ffefef",
|
||||||
|
"on-tertiary-fixed": "#515050", "tertiary-dim": "#504f4f",
|
||||||
|
"on-primary-container": "#4b0027", "on-secondary-fixed": "#580079",
|
||||||
|
"on-error-container": "#510017"
|
||||||
|
},
|
||||||
|
"borderRadius": {
|
||||||
|
"DEFAULT": "1rem", "lg": "2rem", "xl": "3rem", "full": "9999px"
|
||||||
|
},
|
||||||
|
"fontFamily": {
|
||||||
|
"headline": ["Lexend"], "body": ["Be Vietnam Pro"], "label": ["Be Vietnam Pro"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.material-symbols-outlined { font-variation-settings: 'FILL' 0, 'wght' 400, 'GRAD' 0, 'opsz' 24; }
|
||||||
|
.editorial-shadow { box-shadow: 0 24px 40px -12px rgba(44, 47, 48, 0.06); }
|
||||||
|
.no-scrollbar::-webkit-scrollbar { display: none; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="bg-surface font-body text-on-surface selection:bg-primary-container selection:text-on-primary-container">
|
||||||
|
|
||||||
|
<!-- TopAppBar -->
|
||||||
|
<header class="fixed top-0 left-0 w-full z-50 bg-white/70 dark:bg-zinc-900/70 backdrop-blur-lg shadow-xl shadow-pink-500/5">
|
||||||
|
<div class="flex justify-between items-center h-20 px-6 md:px-12 max-w-screen-2xl mx-auto">
|
||||||
|
<a href="/" class="text-2xl font-black text-pink-600 dark:text-pink-400 italic font-headline tracking-tight uppercase">MiyaKarate</a>
|
||||||
|
<nav class="hidden md:flex items-center gap-8">
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/">Home</a>
|
||||||
|
<a class="text-pink-600 dark:text-pink-400 border-b-2 border-pink-500 pb-1 font-headline tracking-tight uppercase font-bold" href="/galerie/">Galerie</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/uebermich/">Über mich</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/erfolge/">Erfolge</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 transition-colors duration-300" href="/gaestebuch/">Gästebuch</a>
|
||||||
|
</nav>
|
||||||
|
<button class="bg-gradient-to-br from-primary to-primary-container text-on-primary px-8 py-3 rounded-xl font-bold font-headline tracking-tight uppercase active:scale-90 transition-transform">
|
||||||
|
Kontakt
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main class="pt-32 pb-20 px-6 max-w-7xl mx-auto">
|
||||||
|
<!-- Header -->
|
||||||
|
<header class="mb-16 relative">
|
||||||
|
<span class="font-label text-sm uppercase tracking-[0.2em] font-bold text-primary mb-4 block">Visuelle Reise</span>
|
||||||
|
<h1 class="font-headline text-5xl md:text-7xl font-black text-on-surface leading-[1.1] tracking-tight mb-6">
|
||||||
|
Eingefangene <br/><span class="text-transparent bg-clip-text bg-gradient-to-r from-primary to-secondary">Momente.</span>
|
||||||
|
</h1>
|
||||||
|
<p class="max-w-xl text-on-surface-variant text-lg leading-relaxed">
|
||||||
|
Die Kunst der Disziplin durch die Linse. Von intensiven Trainingseinheiten bis zum Triumph bei Gürtelprüfungen.
|
||||||
|
</p>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<!-- Filter -->
|
||||||
|
<div class="flex flex-wrap items-center gap-3 mb-12" id="filterBar">
|
||||||
|
<button data-filter="Alle" class="filter-btn px-6 py-3 rounded-full bg-secondary text-on-secondary font-bold text-sm tracking-wide shadow-lg shadow-secondary/20 active:scale-95 transition-transform">Alle</button>
|
||||||
|
<button data-filter="Training" class="filter-btn px-6 py-3 rounded-full bg-surface-container-low text-on-surface-variant font-bold text-sm tracking-wide hover:bg-surface-container-high transition-colors">Training</button>
|
||||||
|
<button data-filter="Wettkämpfe" class="filter-btn px-6 py-3 rounded-full bg-surface-container-low text-on-surface-variant font-bold text-sm tracking-wide hover:bg-surface-container-high transition-colors">Wettkämpfe</button>
|
||||||
|
<button data-filter="Gürtelprüfungen" class="filter-btn px-6 py-3 rounded-full bg-surface-container-low text-on-surface-variant font-bold text-sm tracking-wide hover:bg-surface-container-high transition-colors">Gürtelprüfungen</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Eigene Fotos (dynamisch vom Admin) -->
|
||||||
|
{{ if .Site.Data.gallery.photos }}
|
||||||
|
<div id="photoGrid" class="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4 mb-12">
|
||||||
|
{{ range .Site.Data.gallery.photos }}
|
||||||
|
{{ $katColor := "text-primary-fixed" }}
|
||||||
|
{{ if eq .kategorie "Wettkämpfe" }}{{ $katColor = "text-secondary-fixed" }}{{ end }}
|
||||||
|
{{ if eq .kategorie "Gürtelprüfungen" }}{{ $katColor = "text-primary-container" }}{{ end }}
|
||||||
|
<div class="group relative overflow-hidden rounded-xl cursor-pointer aspect-square gallery-item" data-kat="{{ .kategorie }}" onclick="openLightbox(this)">
|
||||||
|
<img class="w-full h-full object-cover transition-transform duration-700 group-hover:scale-105"
|
||||||
|
src="/gallery/images/{{ .thumb }}" data-full="/gallery/images/{{ .filename }}"
|
||||||
|
alt="{{ .title }}"/>
|
||||||
|
<div class="absolute inset-0 bg-gradient-to-t from-black/60 via-transparent to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300 flex flex-col justify-end p-4">
|
||||||
|
<span class="{{ $katColor }} font-bold text-xs uppercase tracking-widest mb-1">{{ .kategorie }}</span>
|
||||||
|
<h3 class="text-white text-sm font-bold font-headline">{{ .title }}</h3>
|
||||||
|
</div>
|
||||||
|
<div class="absolute top-3 right-3 p-2 bg-white/20 backdrop-blur-md rounded-full text-white opacity-0 group-hover:opacity-100 transition-opacity">
|
||||||
|
<span class="material-symbols-outlined text-lg">zoom_in</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
{{ else }}
|
||||||
|
<!-- Platzhalter-Bento (solange noch keine eigenen Fotos) -->
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-12 gap-6 auto-rows-[250px] mb-12">
|
||||||
|
<div class="md:col-span-8 md:row-span-2 group relative overflow-hidden rounded-xl cursor-pointer" onclick="openLightbox(this)">
|
||||||
|
<img class="w-full h-full object-cover transition-transform duration-700 group-hover:scale-105"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuCfodpf04EOEGQf-nK3kHQTYwr9XD2AcsKt5R14MJZ3OLhsXJZGmkfG2l1YMjyPyGMl9B42kR8cSqjNPA-DNmN7C66PwajUc8IXha9UeTubW45B_wN2Gr0XWnTOqG6wzFXA8PqesKDWkAqoH1enTRjgtQOQ3X9AuZnQqkKlWr_SppBXpNody64LpAbkAYJsjqfQIwwmotKEIbP6edxW80fuf46Fc4c31e1M27SVtDPR6Rp9yvfcyCM4b_IegYAEc3NTq295pAX9Pw"
|
||||||
|
alt="High Kick Training"/>
|
||||||
|
<div class="absolute inset-0 bg-gradient-to-t from-black/60 via-transparent to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300 flex flex-col justify-end p-8">
|
||||||
|
<span class="text-primary-fixed font-bold text-sm uppercase tracking-widest mb-2">Training</span>
|
||||||
|
<h3 class="text-white text-2xl font-bold font-headline">The Art of the High Kick</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="md:col-span-4 md:row-span-3 group relative overflow-hidden rounded-xl cursor-pointer" onclick="openLightbox(this)">
|
||||||
|
<img class="w-full h-full object-cover transition-transform duration-700 group-hover:scale-105"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuBkzRybmeOLCatrp_PsX09PIQuRlaH9iwtDWOoTDtVZ7VeTgz76tk3_NmyDT6L30vQaaEPa7Ir2YqTNdbb5wsXLBbAL3X-fvk9gNVgHRAoV5Qm6QPRY5S0p89zRkoAy51TZgIJEKGTv4YnoEWbI6QN0Zsvgp6xU95Ubn7x2cvN4dGbrIK2gXSralq7NVC56vUm9kA7L_L6ebOtzKGpI_6FZa_jHUr86ASTT2JMDKpQ0EmJJcUD-J4oDnsADRU7B91qTPaMmAMNvcQ"
|
||||||
|
alt="Portrait"/>
|
||||||
|
<div class="absolute inset-0 bg-gradient-to-t from-black/60 via-transparent to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300 flex flex-col justify-end p-8">
|
||||||
|
<span class="text-secondary-fixed font-bold text-sm uppercase tracking-widest mb-2">Fokus</span>
|
||||||
|
<h3 class="text-white text-2xl font-bold font-headline">Vorbereitung</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="md:col-span-4 md:row-span-1 group relative overflow-hidden rounded-xl cursor-pointer" onclick="openLightbox(this)">
|
||||||
|
<img class="w-full h-full object-cover transition-transform duration-700 group-hover:scale-105"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuAvps_MgZKjN93kl4O8188gvviUsvhBuGlXVGvHU40hNS_Rxo1mou3qsQqlC9yDmpY0g1vv9yFf2uL3FHLZcczmZDrLdqVX3n6Cw6xjuiBmHnWEfdqEz7uBNL5tTYAhL5OvjwjzwfGLlKNJw5sL46GzGLuyEl8UKyFb8Z6VOFMO9XSH1Jm70hpY9kiGV38wv1JzNk9d0cySLSlJ1Z6EE9p5JLDILHVBREHg3VH_23pdRO5gRVCorfvfP2HgXrp8bl8o04WJD4CGXQ"
|
||||||
|
alt="Meisterschaft"/>
|
||||||
|
<div class="absolute inset-0 bg-gradient-to-t from-black/60 via-transparent to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300 flex flex-col justify-end p-6">
|
||||||
|
<span class="text-primary-fixed font-bold text-xs uppercase tracking-widest mb-1">Wettkämpfe</span>
|
||||||
|
<h3 class="text-white text-lg font-bold font-headline">Nationale Meisterschaften</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="md:col-span-4 md:row-span-1 bg-gradient-to-br from-secondary to-secondary-dim rounded-xl p-8 flex flex-col justify-between text-on-secondary">
|
||||||
|
<span class="material-symbols-outlined text-4xl" style="font-variation-settings: 'FILL' 1;">rewarded_ads</span>
|
||||||
|
<div>
|
||||||
|
<p class="text-xl font-bold font-headline mb-2">Deine Fotos kommen hier!</p>
|
||||||
|
<p class="text-sm opacity-80 leading-relaxed">Lad Fotos im Admin-Bereich hoch.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- Stats Ribbon -->
|
||||||
|
<section class="bg-surface-container-low py-20 px-6">
|
||||||
|
<div class="max-w-7xl mx-auto flex flex-col md:flex-row justify-between items-center gap-12">
|
||||||
|
<div class="flex-1">
|
||||||
|
<h2 class="font-headline text-3xl font-bold mb-4">Hinter der Linse</h2>
|
||||||
|
<p class="text-on-surface-variant max-w-md">Unsere Galerie ist nicht nur Fotos – sie ist ein Zeugnis der Disziplin, die wir jeden Tag im MiyaKarate Dojo leben.</p>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-8 overflow-x-auto pb-4 no-scrollbar w-full md:w-auto">
|
||||||
|
<div class="flex-shrink-0 text-center">
|
||||||
|
<p class="text-4xl font-black text-primary font-headline">24</p>
|
||||||
|
<p class="text-xs uppercase tracking-widest font-bold opacity-60">Turniere</p>
|
||||||
|
</div>
|
||||||
|
<div class="h-12 w-px bg-outline-variant/30 hidden md:block"></div>
|
||||||
|
<div class="flex-shrink-0 text-center">
|
||||||
|
<p class="text-4xl font-black text-secondary font-headline">150+</p>
|
||||||
|
<p class="text-xs uppercase tracking-widest font-bold opacity-60">Schüler</p>
|
||||||
|
</div>
|
||||||
|
<div class="h-12 w-px bg-outline-variant/30 hidden md:block"></div>
|
||||||
|
<div class="flex-shrink-0 text-center">
|
||||||
|
<p class="text-4xl font-black text-primary font-headline">85</p>
|
||||||
|
<p class="text-xs uppercase tracking-widest font-bold opacity-60">Gürtelprüfungen</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Footer -->
|
||||||
|
<footer class="bg-zinc-50 dark:bg-zinc-950 w-full rounded-t-[3rem] mt-20">
|
||||||
|
<div class="flex flex-col md:flex-row justify-between items-center py-12 px-8 max-w-7xl mx-auto gap-6 text-sm tracking-wide">
|
||||||
|
<div class="text-lg font-bold text-zinc-900 dark:text-zinc-100 font-headline uppercase italic">MiyaKarate</div>
|
||||||
|
<div class="flex gap-8">
|
||||||
|
<a class="text-zinc-500 hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity opacity-80 hover:opacity-100" href="#">Instagram</a>
|
||||||
|
<a class="text-zinc-500 hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity opacity-80 hover:opacity-100" href="#">YouTube</a>
|
||||||
|
<a class="text-zinc-500 hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity opacity-80 hover:opacity-100" href="#">Email</a>
|
||||||
|
</div>
|
||||||
|
<div class="text-zinc-500">© 2024 MiyaKarate. Alle Rechte vorbehalten.</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
<!-- Lightbox -->
|
||||||
|
<div class="fixed inset-0 z-[100] bg-black/90 backdrop-blur-xl hidden items-center justify-center p-6 md:p-20" id="lightbox">
|
||||||
|
<button class="absolute top-10 right-10 text-white hover:text-primary transition-colors" onclick="closeLightbox()">
|
||||||
|
<span class="material-symbols-outlined text-4xl">close</span>
|
||||||
|
</button>
|
||||||
|
<img class="max-w-full max-h-full rounded-lg shadow-2xl" id="lightbox-img" src="" alt=""/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
function openLightbox(el) {
|
||||||
|
const img = el.querySelector('img');
|
||||||
|
document.getElementById('lightbox-img').src = img.src;
|
||||||
|
document.getElementById('lightbox-img').alt = img.alt;
|
||||||
|
const lb = document.getElementById('lightbox');
|
||||||
|
lb.classList.remove('hidden');
|
||||||
|
lb.classList.add('flex');
|
||||||
|
}
|
||||||
|
function closeLightbox() {
|
||||||
|
const lb = document.getElementById('lightbox');
|
||||||
|
lb.classList.add('hidden');
|
||||||
|
lb.classList.remove('flex');
|
||||||
|
}
|
||||||
|
document.getElementById('lightbox').addEventListener('click', function(e) {
|
||||||
|
if (e.target === this) closeLightbox();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Filter-Logik
|
||||||
|
document.getElementById('filterBar').addEventListener('click', function(e) {
|
||||||
|
const btn = e.target.closest('.filter-btn');
|
||||||
|
if (!btn) return;
|
||||||
|
const filter = btn.dataset.filter;
|
||||||
|
document.querySelectorAll('.filter-btn').forEach(b => {
|
||||||
|
const active = b === btn;
|
||||||
|
b.classList.toggle('bg-secondary', active);
|
||||||
|
b.classList.toggle('text-on-secondary', active);
|
||||||
|
b.classList.toggle('shadow-lg', active);
|
||||||
|
b.classList.toggle('shadow-secondary/20', active);
|
||||||
|
b.classList.toggle('bg-surface-container-low', !active);
|
||||||
|
b.classList.toggle('text-on-surface-variant', !active);
|
||||||
|
});
|
||||||
|
document.querySelectorAll('.gallery-item').forEach(item => {
|
||||||
|
const show = filter === 'Alle' || item.dataset.kat === filter;
|
||||||
|
item.style.display = show ? '' : 'none';
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
269
layouts/index.html
Normal file
269
layouts/index.html
Normal file
|
|
@ -0,0 +1,269 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html class="scroll-smooth" lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta content="width=device-width, initial-scale=1.0" name="viewport"/>
|
||||||
|
<title>{{ .Title }} | MiyaKarate</title>
|
||||||
|
<link href="https://fonts.googleapis.com" rel="preconnect"/>
|
||||||
|
<link crossorigin="" href="https://fonts.gstatic.com" rel="preconnect"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Lexend:wght@300;400;500;600;700;800;900&family=Be+Vietnam+Pro:wght@300;400;500;600;700&display=swap" rel="stylesheet"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&display=swap" rel="stylesheet"/>
|
||||||
|
<script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
|
||||||
|
<script id="tailwind-config">
|
||||||
|
tailwind.config = {
|
||||||
|
darkMode: "class",
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
"colors": {
|
||||||
|
"tertiary-container": "#ffffff",
|
||||||
|
"surface-bright": "#f5f6f7",
|
||||||
|
"on-tertiary-container": "#636262",
|
||||||
|
"error-dim": "#a70138",
|
||||||
|
"on-surface-variant": "#595c5d",
|
||||||
|
"surface-tint": "#b30065",
|
||||||
|
"surface-container-highest": "#dadddf",
|
||||||
|
"on-secondary-fixed-variant": "#7d21a4",
|
||||||
|
"inverse-on-surface": "#9b9d9e",
|
||||||
|
"inverse-surface": "#0c0f10",
|
||||||
|
"surface-container": "#e6e8ea",
|
||||||
|
"surface": "#f5f6f7",
|
||||||
|
"primary-fixed": "#ff6ea9",
|
||||||
|
"surface-container-low": "#eff1f2",
|
||||||
|
"secondary": "#8930b0",
|
||||||
|
"tertiary": "#5c5b5b",
|
||||||
|
"on-secondary": "#ffedff",
|
||||||
|
"secondary-dim": "#7c20a3",
|
||||||
|
"secondary-container": "#f0c1ff",
|
||||||
|
"surface-dim": "#d1d5d7",
|
||||||
|
"primary": "#b30065",
|
||||||
|
"on-primary-fixed": "#000000",
|
||||||
|
"surface-container-high": "#e0e3e4",
|
||||||
|
"primary-fixed-dim": "#ff4e9e",
|
||||||
|
"primary-dim": "#9d0058",
|
||||||
|
"outline": "#757778",
|
||||||
|
"error": "#b41340",
|
||||||
|
"outline-variant": "#abadae",
|
||||||
|
"secondary-fixed": "#f0c1ff",
|
||||||
|
"secondary-fixed-dim": "#eaaeff",
|
||||||
|
"tertiary-fixed-dim": "#f3f0ef",
|
||||||
|
"on-surface": "#2c2f30",
|
||||||
|
"on-tertiary-fixed-variant": "#6e6d6d",
|
||||||
|
"on-tertiary": "#f5f2f1",
|
||||||
|
"tertiary-fixed": "#ffffff",
|
||||||
|
"background": "#f5f6f7",
|
||||||
|
"on-primary": "#ffeff2",
|
||||||
|
"surface-container-lowest": "#ffffff",
|
||||||
|
"inverse-primary": "#ff479c",
|
||||||
|
"primary-container": "#ff6ea9",
|
||||||
|
"on-primary-fixed-variant": "#5c0031",
|
||||||
|
"on-secondary-container": "#72129a",
|
||||||
|
"error-container": "#f74b6d",
|
||||||
|
"surface-variant": "#dadddf",
|
||||||
|
"on-background": "#2c2f30",
|
||||||
|
"on-error": "#ffefef",
|
||||||
|
"on-tertiary-fixed": "#515050",
|
||||||
|
"tertiary-dim": "#504f4f",
|
||||||
|
"on-primary-container": "#4b0027",
|
||||||
|
"on-secondary-fixed": "#580079",
|
||||||
|
"on-error-container": "#510017"
|
||||||
|
},
|
||||||
|
"borderRadius": {
|
||||||
|
"DEFAULT": "1rem",
|
||||||
|
"lg": "2rem",
|
||||||
|
"xl": "3rem",
|
||||||
|
"full": "9999px"
|
||||||
|
},
|
||||||
|
"fontFamily": {
|
||||||
|
"headline": ["Lexend"],
|
||||||
|
"body": ["Be Vietnam Pro"],
|
||||||
|
"label": ["Be Vietnam Pro"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.material-symbols-outlined {
|
||||||
|
font-variation-settings: 'FILL' 0, 'wght' 400, 'GRAD' 0, 'opsz' 24;
|
||||||
|
}
|
||||||
|
.editorial-text-shadow {
|
||||||
|
text-shadow: 0 10px 30px rgba(179, 0, 101, 0.2);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="bg-surface font-body text-on-surface overflow-x-hidden">
|
||||||
|
|
||||||
|
<!-- TopAppBar -->
|
||||||
|
<nav class="fixed top-0 left-0 w-full z-50 bg-white/70 dark:bg-zinc-900/70 backdrop-blur-lg shadow-xl shadow-pink-500/5">
|
||||||
|
<div class="flex justify-between items-center h-20 px-6 md:px-12 max-w-screen-2xl mx-auto">
|
||||||
|
<div class="text-2xl font-black text-pink-600 dark:text-pink-400 italic font-headline tracking-tight uppercase">
|
||||||
|
MiyaKarate
|
||||||
|
</div>
|
||||||
|
<div class="hidden md:flex items-center gap-8">
|
||||||
|
<a class="text-pink-600 dark:text-pink-400 border-b-2 border-pink-500 pb-1 font-headline tracking-tight uppercase font-bold transition-colors duration-300" href="/">Home</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/galerie/">Galerie</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/uebermich/">Über mich</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/erfolge/">Erfolge</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium font-headline tracking-tight uppercase hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/gaestebuch/">Gästebuch</a>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<button class="bg-gradient-to-br from-primary to-primary-container text-on-primary px-8 py-3 rounded-xl font-bold font-headline tracking-tight uppercase scale-95 active:scale-90 transition-transform">
|
||||||
|
Kontakt
|
||||||
|
</button>
|
||||||
|
<button class="md:hidden text-primary">
|
||||||
|
<span class="material-symbols-outlined">menu</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<main class="pt-20">
|
||||||
|
<!-- Hero Section -->
|
||||||
|
<section class="relative min-h-[921px] flex items-center overflow-hidden bg-surface">
|
||||||
|
<div class="absolute -right-20 top-20 w-[600px] h-[600px] bg-secondary-container/20 rounded-full blur-[120px]"></div>
|
||||||
|
<div class="max-w-screen-2xl mx-auto px-6 md:px-12 grid grid-cols-1 lg:grid-cols-12 gap-12 items-center relative z-10">
|
||||||
|
<div class="lg:col-span-6 order-2 lg:order-1">
|
||||||
|
<span class="inline-block px-4 py-1 rounded-full bg-secondary-container text-on-secondary-container text-sm font-bold tracking-widest uppercase mb-6 font-label">{{ .Site.Data.homepage.hero.badge }}</span>
|
||||||
|
<h1 class="text-6xl md:text-8xl font-headline font-black text-on-surface leading-[0.9] editorial-text-shadow mb-8 italic">
|
||||||
|
{{ .Site.Title }}
|
||||||
|
</h1>
|
||||||
|
<p class="text-xl text-on-surface-variant max-w-lg mb-10 leading-relaxed font-body">
|
||||||
|
{{ .Site.Data.homepage.hero.description }}
|
||||||
|
</p>
|
||||||
|
<div class="flex flex-wrap gap-4">
|
||||||
|
<button class="bg-gradient-to-br from-primary to-primary-container text-on-primary px-10 py-5 rounded-xl font-bold font-headline text-lg shadow-xl shadow-primary/20 hover:scale-105 active:scale-95 transition-all">
|
||||||
|
Erfahre mehr
|
||||||
|
</button>
|
||||||
|
<button class="px-10 py-5 rounded-xl font-bold font-headline text-lg text-primary border-2 border-primary/20 hover:bg-primary/5 transition-all">
|
||||||
|
Galerie ansehen
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="lg:col-span-6 order-1 lg:order-2 relative">
|
||||||
|
<div class="relative w-full aspect-square rounded-xl overflow-hidden shadow-2xl shadow-primary/10 rotate-3 hover:rotate-0 transition-transform duration-700">
|
||||||
|
{{ if .Site.Data.homepage.hero.image }}
|
||||||
|
<img alt="Miya beim Karate-Tritt" class="w-full h-full object-cover" src="/hero/{{ .Site.Data.homepage.hero.image }}"/>
|
||||||
|
{{ else }}
|
||||||
|
<img alt="Miya beim Karate-Tritt" class="w-full h-full object-cover" src="https://lh3.googleusercontent.com/aida-public/AB6AXuAHCy-F1uHI1Lq8D0LuZVWXLZA4URfCtFVtXNvimgi_5HnX4ovI2C3Cl_nLp5awhMAL-cwWnK-fL-yQHgwYZ2SWg5JbnH7RkQtVdUqPCb9PcOAFIzdX9haBXQGoCYywwzqNXK4QqQoJ5XxnamSZghNPUK0pOLszlu2jowGPO8VWtQmD7PcJTOGfpOCUxw8tNzeeNTQsCDmPoGD3N8ZjTyTGD6Sk48MrYJrRUgiRvBi9tznnMXIgqMsN8G0v8JA3aQeF5jQlZmJ3kw"/>
|
||||||
|
{{ end }}
|
||||||
|
<div class="absolute bottom-8 -left-8 bg-white/40 backdrop-blur-xl p-6 rounded-lg shadow-xl border border-white/20 -rotate-6">
|
||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<div class="w-12 h-12 bg-secondary rounded-full flex items-center justify-center text-white">
|
||||||
|
<span class="material-symbols-outlined" style="font-variation-settings: 'FILL' 1;">stars</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-headline font-bold text-on-surface uppercase leading-none">Blaugurt</p>
|
||||||
|
<p class="text-xs text-on-surface-variant uppercase tracking-widest mt-1">Status 2024</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Stats Section -->
|
||||||
|
{{ $colors := slice "text-primary" "text-secondary" "text-primary-dim" }}
|
||||||
|
<section class="bg-surface-container-low py-16">
|
||||||
|
<div class="max-w-7xl mx-auto px-6 flex flex-wrap justify-around gap-12 text-center">
|
||||||
|
{{ range $i, $s := .Site.Data.homepage.stats }}
|
||||||
|
<div>
|
||||||
|
<h3 class="text-5xl font-black font-headline {{ index $colors $i }}">{{ $s.value }}</h3>
|
||||||
|
<p class="text-sm font-label uppercase tracking-widest text-on-surface-variant mt-2">{{ $s.label }}</p>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Erfolge Section -->
|
||||||
|
<section class="py-24 max-w-7xl mx-auto px-6">
|
||||||
|
<div class="mb-16">
|
||||||
|
<span class="text-primary font-bold tracking-widest uppercase text-sm font-label">Die neuesten Erfolge</span>
|
||||||
|
<h2 class="text-5xl font-black font-headline mt-4">Meine Highlights</h2>
|
||||||
|
</div>
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-12 gap-8">
|
||||||
|
{{ range (where .Site.RegularPages "Section" "erfolge") }}
|
||||||
|
<div class="md:col-span-8 bg-surface-container-lowest rounded-xl p-8 shadow-sm group hover:shadow-xl transition-all duration-500 overflow-hidden relative">
|
||||||
|
<div class="relative z-10">
|
||||||
|
<span class="bg-secondary text-white px-4 py-1 rounded-full text-xs font-bold uppercase tracking-widest">{{ .Params.rang | default "Highlight" }}</span>
|
||||||
|
<h3 class="text-3xl font-black font-headline mt-6 mb-4">{{ .Title }}</h3>
|
||||||
|
<p class="text-on-surface-variant mb-8 max-w-md">{{ .Summary }}</p>
|
||||||
|
<a class="flex items-center gap-2 text-primary font-bold hover:gap-4 transition-all" href="{{ .RelPermalink }}">
|
||||||
|
Bericht lesen <span class="material-symbols-outlined">arrow_forward</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="absolute right-0 bottom-0 w-1/2 h-full opacity-10 group-hover:opacity-20 group-hover:scale-110 transition-all duration-700">
|
||||||
|
<img alt="Highlight Thumbnail" class="w-full h-full object-cover grayscale brightness-50"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuCt0zGFYDYcvBDPfNXlVvPqNdkvn4AvVSTlFysp0raGeWEmbAnpQkad18FIakDIrPbq4d93sRhkJnquI7QoXZrLf22SBA8nG_IjRg0JkMadeTHr_KOs0vgEVpV48jXsqKBSI2Rx4J02al6QxsLdWCA6XRBKslA9R0v-u_SrGGB7oNpfxjRV-6L6REjgsuzlGHPwdyY7bLrk_MBOvHUG9hfQ5bJiaiTiKfchBEBdNdtk3MPJow72blhWqMMflZL_bdxzAfX8TCyWKQ"/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
<!-- Karte: Prüfung bestanden -->
|
||||||
|
<div class="md:col-span-4 bg-primary text-on-primary rounded-xl p-8 flex flex-col justify-between hover:scale-[1.02] transition-transform shadow-xl shadow-primary/20">
|
||||||
|
<span class="material-symbols-outlined text-4xl" style="font-variation-settings: 'FILL' 1;">military_tech</span>
|
||||||
|
<div>
|
||||||
|
<h3 class="text-2xl font-black font-headline mb-2">Prüfung bestanden</h3>
|
||||||
|
<p class="text-on-primary/80">Neuer Gürtelgrad erreicht! Jetzt stolze Trägerin des Blaugurts.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Karte: Dojo Champion -->
|
||||||
|
<div class="md:col-span-4 bg-surface-container-low rounded-xl p-8 hover:bg-surface-container-high transition-colors">
|
||||||
|
<div class="w-12 h-12 bg-white rounded-lg flex items-center justify-center mb-6 shadow-sm">
|
||||||
|
<span class="material-symbols-outlined text-secondary">fitness_center</span>
|
||||||
|
</div>
|
||||||
|
<h3 class="text-xl font-black font-headline mb-2">Dojo Champion</h3>
|
||||||
|
<p class="text-sm text-on-surface-variant italic">Hausinterner Wettbewerb im Dojo "Kiai Berlin".</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Bild: Team Spirit -->
|
||||||
|
<div class="md:col-span-8 h-64 rounded-xl overflow-hidden relative group">
|
||||||
|
<img alt="Teamtraining" class="w-full h-full object-cover group-hover:scale-110 transition-transform duration-1000"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuAxl_8pFN3p4UiOW5vHu3eUKrCj1VrD6wYTDuajtl3iHvjkNJEOLbqbWt2QL5xcC4jC8H4unZhLKoapi4aH8paj81HeIEcOZwkfBBDZyLz6vQpXOODIg4KyEBDvfGzCyJWGf_qTxRmpzgjNfLAOeKA1dCFuRq5MJGSi3gsuobQBv7cR1kGAkgMKXEMTjsKrR_ZvwVQYyb7tWPzAWfTqEt97ViRO1y05wIOMbBiqTx2qXcbzVfsdVvtyAjaMh9Wh7npl8UTJIlvMXA"/>
|
||||||
|
<div class="absolute inset-0 bg-gradient-to-t from-black/80 to-transparent flex items-end p-8">
|
||||||
|
<div>
|
||||||
|
<h3 class="text-2xl font-bold font-headline text-white">Team Spirit</h3>
|
||||||
|
<p class="text-white/70">Gemeinsam trainieren macht am meisten Spaß.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- CTA Galerie Section -->
|
||||||
|
<section class="py-24 relative overflow-hidden">
|
||||||
|
<div class="absolute inset-0 bg-zinc-900 -z-10"></div>
|
||||||
|
<div class="max-w-7xl mx-auto px-6 text-center text-white">
|
||||||
|
<h2 class="text-5xl md:text-7xl font-black font-headline italic mb-8">Bereit für mehr <span class="text-primary-fixed">Action?</span></h2>
|
||||||
|
<p class="text-xl text-zinc-400 max-w-2xl mx-auto mb-12">Schau dir hunderte Fotos von Trainingseinheiten, Turnieren und Reisen in meiner großen Galerie an.</p>
|
||||||
|
<a class="inline-flex items-center gap-4 bg-primary text-white px-12 py-6 rounded-full font-black font-headline text-xl hover:bg-primary-dim transition-colors group" href="#">
|
||||||
|
ZUR GALERIE GEHEN
|
||||||
|
<span class="material-symbols-outlined group-hover:translate-x-2 transition-transform">auto_awesome_motion</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="absolute top-1/2 left-0 -translate-y-1/2 opacity-5 pointer-events-none whitespace-nowrap">
|
||||||
|
<span class="text-[200px] font-black font-headline italic text-white uppercase tracking-tighter">GALLERY GALLERY GALLERY</span>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- Footer -->
|
||||||
|
<footer class="w-full rounded-t-[3rem] mt-20 bg-zinc-50 dark:bg-zinc-950">
|
||||||
|
<div class="flex flex-col md:flex-row justify-between items-center py-12 px-8 max-w-7xl mx-auto gap-6">
|
||||||
|
<div class="text-lg font-bold text-zinc-900 dark:text-zinc-100 font-headline uppercase italic">
|
||||||
|
MiyaKarate
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-8 text-sm tracking-wide">
|
||||||
|
<a class="text-zinc-500 hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity opacity-80 hover:opacity-100" href="#">Instagram</a>
|
||||||
|
<a class="text-zinc-500 hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity opacity-80 hover:opacity-100" href="#">YouTube</a>
|
||||||
|
<a class="text-zinc-500 hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity opacity-80 hover:opacity-100" href="#">Email</a>
|
||||||
|
</div>
|
||||||
|
<div class="text-sm tracking-wide text-zinc-500">
|
||||||
|
© 2024 MiyaKarate. Alle Rechte vorbehalten.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
268
layouts/uebermich/list.html
Normal file
268
layouts/uebermich/list.html
Normal file
|
|
@ -0,0 +1,268 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html class="scroll-smooth" lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta content="width=device-width, initial-scale=1.0" name="viewport"/>
|
||||||
|
<title>Über mich | MiyaKarate</title>
|
||||||
|
<link href="https://fonts.googleapis.com" rel="preconnect"/>
|
||||||
|
<link crossorigin="" href="https://fonts.gstatic.com" rel="preconnect"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Lexend:wght@300;400;500;600;700;800;900&family=Be+Vietnam+Pro:wght@300;400;500;600;700&display=swap" rel="stylesheet"/>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&display=swap" rel="stylesheet"/>
|
||||||
|
<script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
|
||||||
|
<script id="tailwind-config">
|
||||||
|
tailwind.config = {
|
||||||
|
darkMode: "class",
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
"colors": {
|
||||||
|
"tertiary-container": "#ffffff", "surface-bright": "#f5f6f7",
|
||||||
|
"on-tertiary-container": "#636262", "error-dim": "#a70138",
|
||||||
|
"on-surface-variant": "#595c5d", "surface-tint": "#b30065",
|
||||||
|
"surface-container-highest": "#dadddf", "on-secondary-fixed-variant": "#7d21a4",
|
||||||
|
"inverse-on-surface": "#9b9d9e", "inverse-surface": "#0c0f10",
|
||||||
|
"surface-container": "#e6e8ea", "surface": "#f5f6f7",
|
||||||
|
"primary-fixed": "#ff6ea9", "surface-container-low": "#eff1f2",
|
||||||
|
"secondary": "#8930b0", "tertiary": "#5c5b5b",
|
||||||
|
"on-secondary": "#ffedff", "secondary-dim": "#7c20a3",
|
||||||
|
"secondary-container": "#f0c1ff", "surface-dim": "#d1d5d7",
|
||||||
|
"primary": "#b30065", "on-primary-fixed": "#000000",
|
||||||
|
"surface-container-high": "#e0e3e4", "primary-fixed-dim": "#ff4e9e",
|
||||||
|
"primary-dim": "#9d0058", "outline": "#757778",
|
||||||
|
"error": "#b41340", "outline-variant": "#abadae",
|
||||||
|
"secondary-fixed": "#f0c1ff", "secondary-fixed-dim": "#eaaeff",
|
||||||
|
"tertiary-fixed-dim": "#f3f0ef", "on-surface": "#2c2f30",
|
||||||
|
"on-tertiary-fixed-variant": "#6e6d6d", "on-tertiary": "#f5f2f1",
|
||||||
|
"tertiary-fixed": "#ffffff", "background": "#f5f6f7",
|
||||||
|
"on-primary": "#ffeff2", "surface-container-lowest": "#ffffff",
|
||||||
|
"inverse-primary": "#ff479c", "primary-container": "#ff6ea9",
|
||||||
|
"on-primary-fixed-variant": "#5c0031", "on-secondary-container": "#72129a",
|
||||||
|
"error-container": "#f74b6d", "surface-variant": "#dadddf",
|
||||||
|
"on-background": "#2c2f30", "on-error": "#ffefef",
|
||||||
|
"on-tertiary-fixed": "#515050", "tertiary-dim": "#504f4f",
|
||||||
|
"on-primary-container": "#4b0027", "on-secondary-fixed": "#580079",
|
||||||
|
"on-error-container": "#510017"
|
||||||
|
},
|
||||||
|
"borderRadius": {
|
||||||
|
"DEFAULT": "1rem", "lg": "2rem", "xl": "3rem", "full": "9999px"
|
||||||
|
},
|
||||||
|
"fontFamily": {
|
||||||
|
"headline": ["Lexend"], "body": ["Be Vietnam Pro"], "label": ["Be Vietnam Pro"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.material-symbols-outlined { font-variation-settings: 'FILL' 0, 'wght' 400, 'GRAD' 0, 'opsz' 24; }
|
||||||
|
.text-glass-gradient {
|
||||||
|
background: linear-gradient(135deg, #b30065, #ff6ea9);
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
-webkit-text-fill-color: transparent;
|
||||||
|
}
|
||||||
|
.editorial-shadow { box-shadow: 0 24px 40px -10px rgba(44, 47, 48, 0.06); }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="bg-surface font-body text-on-surface">
|
||||||
|
|
||||||
|
<!-- TopAppBar -->
|
||||||
|
<header class="bg-white/70 dark:bg-zinc-900/70 backdrop-blur-lg fixed top-0 left-0 w-full z-50 shadow-xl shadow-pink-500/5">
|
||||||
|
<div class="flex justify-between items-center h-20 px-6 md:px-12 max-w-screen-2xl mx-auto">
|
||||||
|
<a href="/" class="text-2xl font-black text-pink-600 dark:text-pink-400 italic font-headline tracking-tight uppercase">MiyaKarate</a>
|
||||||
|
<nav class="hidden md:flex items-center space-x-8 font-headline tracking-tight uppercase font-bold">
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/">Home</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/galerie/">Galerie</a>
|
||||||
|
<a class="text-pink-600 dark:text-pink-400 border-b-2 border-pink-500 pb-1" href="/uebermich/">Über mich</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/erfolge/">Erfolge</a>
|
||||||
|
<a class="text-zinc-600 dark:text-zinc-400 font-medium hover:text-pink-500 dark:hover:text-pink-300 transition-colors duration-300" href="/gaestebuch/">Gästebuch</a>
|
||||||
|
</nav>
|
||||||
|
<button class="bg-gradient-to-br from-primary to-primary-container text-on-primary px-8 py-3 rounded-xl font-bold font-headline uppercase tracking-wider scale-95 active:scale-90 transition-transform">
|
||||||
|
Kontakt
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main class="pt-32 pb-20">
|
||||||
|
<!-- Hero -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 mb-24">
|
||||||
|
<div class="grid grid-cols-1 lg:grid-cols-12 gap-12 items-center">
|
||||||
|
<div class="lg:col-span-7">
|
||||||
|
<span class="text-primary font-bold tracking-[0.2em] uppercase text-sm mb-4 block">Meine Reise</span>
|
||||||
|
<h1 class="text-6xl md:text-8xl font-headline font-extrabold text-on-surface leading-[0.9] tracking-tighter mb-8">
|
||||||
|
KINETISCHE <span class="text-glass-gradient">ELEGANZ</span> IN JEDEM SCHLAG.
|
||||||
|
</h1>
|
||||||
|
<p class="text-xl text-on-surface-variant leading-relaxed max-w-xl">
|
||||||
|
Seit ich sechs Jahre alt bin, ist das Dojo mein zweites Zuhause. Für mich ist Karate mehr als Medaillen – es ist eine Symphonie aus Fokus, Disziplin und explosiver Energie in einer einzigen Bewegung.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="lg:col-span-5 relative">
|
||||||
|
<div class="aspect-[4/5] rounded-xl overflow-hidden editorial-shadow bg-surface-container-highest">
|
||||||
|
<img class="w-full h-full object-cover"
|
||||||
|
src="https://lh3.googleusercontent.com/aida-public/AB6AXuD_gAlNnXIZRs_NnR68igfJUfHX1ueNZlrjMt15L5xNz3bnL4235YADrLG-nT3wz2ZWYEZy6Dom8-fgsolN77eu_0JF52Xp-YjWEre5kwxN2D6V5LAoXI_T8I2YSU6LI5ZybF1Y1Ynqp8Y2IzCh0DYZOqY_tlPOuzsExGMn7nO0jWw58sR7Ny1674begJzhSNxELjEE7oQfgLjSZaO17dv1vGG5LykvtL9NEqM2JXdOVh0SlsnUN4416utgLZpV8km6wUza_TY7Fg"
|
||||||
|
alt="Miya beim Karate-Kick"/>
|
||||||
|
</div>
|
||||||
|
<div class="absolute -bottom-10 -left-10 bg-white p-8 rounded-lg editorial-shadow hidden md:block">
|
||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<div class="w-12 h-12 rounded-full bg-secondary flex items-center justify-center text-on-secondary">
|
||||||
|
<span class="material-symbols-outlined" style="font-variation-settings: 'FILL' 1;">workspace_premium</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div class="text-sm font-bold uppercase tracking-widest text-secondary">Aktueller Rang</div>
|
||||||
|
<div class="text-2xl font-headline font-black text-on-surface">Blaugurt</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Gürtel-Fortschritt -->
|
||||||
|
<section class="bg-surface-container-low py-24 rounded-t-[5rem]">
|
||||||
|
<div class="max-w-7xl mx-auto px-6">
|
||||||
|
<div class="flex flex-col md:flex-row justify-between items-end mb-16 gap-6">
|
||||||
|
<div>
|
||||||
|
<h2 class="text-4xl font-headline font-bold text-on-surface mb-2">Der Weg zur Meisterschaft</h2>
|
||||||
|
<p class="text-on-surface-variant">Jeder Gürtel erzählt eine Geschichte aus Hingabe und Wachstum.</p>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<span class="px-4 py-2 bg-secondary text-on-secondary rounded-full text-xs font-bold uppercase tracking-widest">Aktiver Status</span>
|
||||||
|
<span class="px-4 py-2 bg-surface-container-highest text-on-surface-variant rounded-full text-xs font-bold uppercase tracking-widest">Wettkampf-Athletin</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-wrap justify-between gap-4">
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-40">
|
||||||
|
<div class="w-24 h-4 bg-white rounded-full border-2 border-zinc-300"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Weiß</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-40">
|
||||||
|
<div class="w-24 h-4 bg-yellow-400 rounded-full"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Gelb</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-40">
|
||||||
|
<div class="w-24 h-4 bg-orange-500 rounded-full"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Orange</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-50">
|
||||||
|
<div class="w-24 h-4 bg-green-600 rounded-full"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Grün</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4">
|
||||||
|
<div class="w-24 h-6 bg-blue-600 rounded-full shadow-lg ring-4 ring-primary/20"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter text-primary">Blau ✓</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-30">
|
||||||
|
<div class="w-24 h-4 bg-purple-700 rounded-full"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Lila</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-20">
|
||||||
|
<div class="w-24 h-4 bg-red-700 rounded-full"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Braun</span>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col items-center gap-4 opacity-10">
|
||||||
|
<div class="w-24 h-4 bg-zinc-900 rounded-full"></div>
|
||||||
|
<span class="font-bold text-xs uppercase tracking-tighter">Schwarz</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Meilensteine Bento Grid -->
|
||||||
|
<section class="max-w-7xl mx-auto px-6 mt-24">
|
||||||
|
<h2 class="text-5xl font-headline font-black mb-16 tracking-tight">MEILENSTEINE DES <span class="text-primary italic">ERFOLGS</span></h2>
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-3 gap-8">
|
||||||
|
<!-- Große Achievement-Karte -->
|
||||||
|
<div class="md:col-span-2 bg-surface-container-lowest rounded-xl p-10 editorial-shadow flex flex-col justify-between relative overflow-hidden group">
|
||||||
|
<div class="relative z-10">
|
||||||
|
<div class="flex items-center gap-4 mb-6">
|
||||||
|
<span class="material-symbols-outlined text-4xl text-primary" style="font-variation-settings: 'FILL' 1;">military_tech</span>
|
||||||
|
<span class="font-headline font-extrabold text-2xl uppercase italic">Landesmeisterschaft Berlin</span>
|
||||||
|
</div>
|
||||||
|
<h3 class="text-4xl font-headline font-bold text-on-surface mb-4">Gold – Kata U14</h3>
|
||||||
|
<p class="text-on-surface-variant max-w-md text-lg">Mein bisher größter Erfolg. Monatelanges hartes Training gipfelte in einer Vorführung, die meinen Wettkampfgeist unter Beweis stellte.</p>
|
||||||
|
</div>
|
||||||
|
<div class="mt-12 flex items-center gap-6 relative z-10">
|
||||||
|
<div class="text-left">
|
||||||
|
<div class="text-3xl font-black text-on-surface">2024</div>
|
||||||
|
<div class="text-xs uppercase tracking-widest font-bold text-primary">Berlin</div>
|
||||||
|
</div>
|
||||||
|
<div class="h-10 w-[1px] bg-outline-variant/30"></div>
|
||||||
|
<div class="text-left">
|
||||||
|
<div class="text-3xl font-black text-on-surface">U14</div>
|
||||||
|
<div class="text-xs uppercase tracking-widest font-bold text-primary">Kategorie</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<span class="material-symbols-outlined absolute -bottom-10 -right-10 text-[15rem] text-surface-container opacity-20 group-hover:opacity-30 transition-opacity">sports_martial_arts</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Weitere Auszeichnungen -->
|
||||||
|
<div class="bg-primary rounded-xl p-8 text-on-primary flex flex-col gap-8 shadow-2xl shadow-primary/20">
|
||||||
|
<h4 class="font-headline font-bold text-xl border-b border-on-primary/20 pb-4">Weitere Erfolge</h4>
|
||||||
|
<div class="flex gap-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-white/20 flex items-center justify-center shrink-0">
|
||||||
|
<span class="material-symbols-outlined text-xl">star</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Norddeutsche Meisterschaft</p>
|
||||||
|
<p class="text-sm opacity-80">Silber – Kata 2024</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-white/20 flex items-center justify-center shrink-0">
|
||||||
|
<span class="material-symbols-outlined text-xl">emoji_events</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Dojo Champion</p>
|
||||||
|
<p class="text-sm opacity-80">Kiai Berlin, intern</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-4">
|
||||||
|
<div class="w-10 h-10 rounded-full bg-white/20 flex items-center justify-center shrink-0">
|
||||||
|
<span class="material-symbols-outlined text-xl">rewarded_ads</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p class="font-bold">Fairness-Preis</p>
|
||||||
|
<p class="text-sm opacity-80">Dojo-Auszeichnung 2023</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Stats -->
|
||||||
|
<div class="bg-secondary-container rounded-xl p-8 flex flex-col justify-center gap-2">
|
||||||
|
<span class="text-on-secondary-container font-headline font-black text-6xl italic">5+</span>
|
||||||
|
<span class="text-secondary font-bold uppercase tracking-[0.2em] text-sm">Jahre Training</span>
|
||||||
|
</div>
|
||||||
|
<div class="bg-surface-container-highest rounded-xl p-8 flex flex-col justify-center gap-2">
|
||||||
|
<span class="text-on-surface font-headline font-black text-6xl italic">32</span>
|
||||||
|
<span class="text-on-surface-variant font-bold uppercase tracking-[0.2em] text-sm">Turniere</span>
|
||||||
|
</div>
|
||||||
|
<div class="bg-white rounded-xl p-8 editorial-shadow flex flex-col justify-center gap-2 border-t-4 border-pink-500">
|
||||||
|
<span class="text-pink-600 font-headline font-black text-6xl italic">12</span>
|
||||||
|
<span class="text-zinc-500 font-bold uppercase tracking-[0.2em] text-sm">Goldmedaillen</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Zitat -->
|
||||||
|
<section class="max-w-4xl mx-auto px-6 py-32 text-center">
|
||||||
|
<span class="material-symbols-outlined text-primary text-6xl mb-8" style="font-variation-settings: 'FILL' 1;">format_quote</span>
|
||||||
|
<blockquote class="text-3xl md:text-5xl font-headline font-bold text-on-surface leading-tight tracking-tight italic">
|
||||||
|
"Karate ist nicht nur ein Sport. Es ist eine Linse, durch die ich die Welt sehe – mit Präzision, Respekt und unbeugsamer Konzentration."
|
||||||
|
</blockquote>
|
||||||
|
<p class="mt-8 font-bold uppercase tracking-widest text-on-surface-variant">— Miya</p>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- Footer -->
|
||||||
|
<footer class="bg-zinc-50 dark:bg-zinc-950 w-full rounded-t-[3rem] mt-20">
|
||||||
|
<div class="flex flex-col md:flex-row justify-between items-center py-12 px-8 max-w-7xl mx-auto gap-6 text-sm tracking-wide">
|
||||||
|
<div class="text-lg font-bold text-zinc-900 dark:text-zinc-100 font-headline uppercase italic">MiyaKarate</div>
|
||||||
|
<div class="flex gap-8 text-zinc-500">
|
||||||
|
<a class="hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity" href="#">Instagram</a>
|
||||||
|
<a class="hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity" href="#">YouTube</a>
|
||||||
|
<a class="hover:text-pink-500 hover:underline decoration-pink-500 decoration-2 underline-offset-4 transition-opacity" href="#">Email</a>
|
||||||
|
</div>
|
||||||
|
<div class="text-zinc-500 opacity-80">© 2024 MiyaKarate. Alle Rechte vorbehalten.</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
191
node_modules/.bin/semver
generated
vendored
Executable file
191
node_modules/.bin/semver
generated
vendored
Executable file
|
|
@ -0,0 +1,191 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
// Standalone semver comparison program.
|
||||||
|
// Exits successfully and prints matching version(s) if
|
||||||
|
// any supplied version is valid and passes all tests.
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
const argv = process.argv.slice(2)
|
||||||
|
|
||||||
|
let versions = []
|
||||||
|
|
||||||
|
const range = []
|
||||||
|
|
||||||
|
let inc = null
|
||||||
|
|
||||||
|
const version = require('../package.json').version
|
||||||
|
|
||||||
|
let loose = false
|
||||||
|
|
||||||
|
let includePrerelease = false
|
||||||
|
|
||||||
|
let coerce = false
|
||||||
|
|
||||||
|
let rtl = false
|
||||||
|
|
||||||
|
let identifier
|
||||||
|
|
||||||
|
let identifierBase
|
||||||
|
|
||||||
|
const semver = require('../')
|
||||||
|
const parseOptions = require('../internal/parse-options')
|
||||||
|
|
||||||
|
let reverse = false
|
||||||
|
|
||||||
|
let options = {}
|
||||||
|
|
||||||
|
const main = () => {
|
||||||
|
if (!argv.length) {
|
||||||
|
return help()
|
||||||
|
}
|
||||||
|
while (argv.length) {
|
||||||
|
let a = argv.shift()
|
||||||
|
const indexOfEqualSign = a.indexOf('=')
|
||||||
|
if (indexOfEqualSign !== -1) {
|
||||||
|
const value = a.slice(indexOfEqualSign + 1)
|
||||||
|
a = a.slice(0, indexOfEqualSign)
|
||||||
|
argv.unshift(value)
|
||||||
|
}
|
||||||
|
switch (a) {
|
||||||
|
case '-rv': case '-rev': case '--rev': case '--reverse':
|
||||||
|
reverse = true
|
||||||
|
break
|
||||||
|
case '-l': case '--loose':
|
||||||
|
loose = true
|
||||||
|
break
|
||||||
|
case '-p': case '--include-prerelease':
|
||||||
|
includePrerelease = true
|
||||||
|
break
|
||||||
|
case '-v': case '--version':
|
||||||
|
versions.push(argv.shift())
|
||||||
|
break
|
||||||
|
case '-i': case '--inc': case '--increment':
|
||||||
|
switch (argv[0]) {
|
||||||
|
case 'major': case 'minor': case 'patch': case 'prerelease':
|
||||||
|
case 'premajor': case 'preminor': case 'prepatch':
|
||||||
|
case 'release':
|
||||||
|
inc = argv.shift()
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
inc = 'patch'
|
||||||
|
break
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case '--preid':
|
||||||
|
identifier = argv.shift()
|
||||||
|
break
|
||||||
|
case '-r': case '--range':
|
||||||
|
range.push(argv.shift())
|
||||||
|
break
|
||||||
|
case '-n':
|
||||||
|
identifierBase = argv.shift()
|
||||||
|
if (identifierBase === 'false') {
|
||||||
|
identifierBase = false
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case '-c': case '--coerce':
|
||||||
|
coerce = true
|
||||||
|
break
|
||||||
|
case '--rtl':
|
||||||
|
rtl = true
|
||||||
|
break
|
||||||
|
case '--ltr':
|
||||||
|
rtl = false
|
||||||
|
break
|
||||||
|
case '-h': case '--help': case '-?':
|
||||||
|
return help()
|
||||||
|
default:
|
||||||
|
versions.push(a)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
options = parseOptions({ loose, includePrerelease, rtl })
|
||||||
|
|
||||||
|
versions = versions.map((v) => {
|
||||||
|
return coerce ? (semver.coerce(v, options) || { version: v }).version : v
|
||||||
|
}).filter((v) => {
|
||||||
|
return semver.valid(v, options)
|
||||||
|
})
|
||||||
|
if (!versions.length) {
|
||||||
|
return fail()
|
||||||
|
}
|
||||||
|
if (inc && (versions.length !== 1 || range.length)) {
|
||||||
|
return failInc()
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0, l = range.length; i < l; i++) {
|
||||||
|
versions = versions.filter((v) => {
|
||||||
|
return semver.satisfies(v, range[i], options)
|
||||||
|
})
|
||||||
|
if (!versions.length) {
|
||||||
|
return fail()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
versions
|
||||||
|
.sort((a, b) => semver[reverse ? 'rcompare' : 'compare'](a, b, options))
|
||||||
|
.map(v => semver.clean(v, options))
|
||||||
|
.map(v => inc ? semver.inc(v, inc, options, identifier, identifierBase) : v)
|
||||||
|
.forEach(v => console.log(v))
|
||||||
|
}
|
||||||
|
|
||||||
|
const failInc = () => {
|
||||||
|
console.error('--inc can only be used on a single version with no range')
|
||||||
|
fail()
|
||||||
|
}
|
||||||
|
|
||||||
|
const fail = () => process.exit(1)
|
||||||
|
|
||||||
|
const help = () => console.log(
|
||||||
|
`SemVer ${version}
|
||||||
|
|
||||||
|
A JavaScript implementation of the https://semver.org/ specification
|
||||||
|
Copyright Isaac Z. Schlueter
|
||||||
|
|
||||||
|
Usage: semver [options] <version> [<version> [...]]
|
||||||
|
Prints valid versions sorted by SemVer precedence
|
||||||
|
|
||||||
|
Options:
|
||||||
|
-r --range <range>
|
||||||
|
Print versions that match the specified range.
|
||||||
|
|
||||||
|
-i --increment [<level>]
|
||||||
|
Increment a version by the specified level. Level can
|
||||||
|
be one of: major, minor, patch, premajor, preminor,
|
||||||
|
prepatch, prerelease, or release. Default level is 'patch'.
|
||||||
|
Only one version may be specified.
|
||||||
|
|
||||||
|
--preid <identifier>
|
||||||
|
Identifier to be used to prefix premajor, preminor,
|
||||||
|
prepatch or prerelease version increments.
|
||||||
|
|
||||||
|
-l --loose
|
||||||
|
Interpret versions and ranges loosely
|
||||||
|
|
||||||
|
-p --include-prerelease
|
||||||
|
Always include prerelease versions in range matching
|
||||||
|
|
||||||
|
-c --coerce
|
||||||
|
Coerce a string into SemVer if possible
|
||||||
|
(does not imply --loose)
|
||||||
|
|
||||||
|
--rtl
|
||||||
|
Coerce version strings right to left
|
||||||
|
|
||||||
|
--ltr
|
||||||
|
Coerce version strings left to right (default)
|
||||||
|
|
||||||
|
-n <base>
|
||||||
|
Base number to be used for the prerelease identifier.
|
||||||
|
Can be either 0 or 1, or false to omit the number altogether.
|
||||||
|
Defaults to 0.
|
||||||
|
|
||||||
|
Program exits successfully if any valid version satisfies
|
||||||
|
all supplied ranges, and prints all satisfying versions.
|
||||||
|
|
||||||
|
If no satisfying versions are found, then exits failure.
|
||||||
|
|
||||||
|
Versions are printed in ascending order, so supplying
|
||||||
|
multiple versions to the utility will just sort them.`)
|
||||||
|
|
||||||
|
main()
|
||||||
1094
node_modules/.package-lock.json
generated
vendored
Normal file
1094
node_modules/.package-lock.json
generated
vendored
Normal file
File diff suppressed because it is too large
Load diff
82
node_modules/@img/colour/LICENSE.md
generated
vendored
Normal file
82
node_modules/@img/colour/LICENSE.md
generated
vendored
Normal file
|
|
@ -0,0 +1,82 @@
|
||||||
|
# Licensing
|
||||||
|
|
||||||
|
## color
|
||||||
|
|
||||||
|
Copyright (c) 2012 Heather Arthur
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
## color-convert
|
||||||
|
|
||||||
|
Copyright (c) 2011-2016 Heather Arthur <fayearthur@gmail.com>.
|
||||||
|
Copyright (c) 2016-2021 Josh Junon <josh@junon.me>.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
## color-string
|
||||||
|
|
||||||
|
Copyright (c) 2011 Heather Arthur <fayearthur@gmail.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
## color-name
|
||||||
|
|
||||||
|
The MIT License (MIT)
|
||||||
|
Copyright (c) 2015 Dmitry Ivanov
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
15
node_modules/@img/colour/README.md
generated
vendored
Normal file
15
node_modules/@img/colour/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
# `@img/colour`
|
||||||
|
|
||||||
|
The latest version of the
|
||||||
|
[color](https://www.npmjs.com/package/color)
|
||||||
|
package is now ESM-only,
|
||||||
|
however some JavaScript runtimes do not yet support this,
|
||||||
|
which includes versions of Node.js prior to 20.19.0.
|
||||||
|
|
||||||
|
This package converts the `color` package and its dependencies,
|
||||||
|
all of which are MIT-licensed, to CommonJS.
|
||||||
|
|
||||||
|
- [color](https://www.npmjs.com/package/color)
|
||||||
|
- [color-convert](https://www.npmjs.com/package/color-convert)
|
||||||
|
- [color-string](https://www.npmjs.com/package/color-string)
|
||||||
|
- [color-name](https://www.npmjs.com/package/color-name)
|
||||||
1596
node_modules/@img/colour/color.cjs
generated
vendored
Normal file
1596
node_modules/@img/colour/color.cjs
generated
vendored
Normal file
File diff suppressed because it is too large
Load diff
1
node_modules/@img/colour/index.cjs
generated
vendored
Normal file
1
node_modules/@img/colour/index.cjs
generated
vendored
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
module.exports = require("./color.cjs").default;
|
||||||
929
node_modules/@img/colour/index.d.ts
generated
vendored
Normal file
929
node_modules/@img/colour/index.d.ts
generated
vendored
Normal file
|
|
@ -0,0 +1,929 @@
|
||||||
|
// Generated by dts-bundle-generator v9.5.1
|
||||||
|
|
||||||
|
type Channels = number;
|
||||||
|
type RGB = [
|
||||||
|
r: number,
|
||||||
|
g: number,
|
||||||
|
b: number
|
||||||
|
];
|
||||||
|
type HSL = [
|
||||||
|
h: number,
|
||||||
|
s: number,
|
||||||
|
l: number
|
||||||
|
];
|
||||||
|
type HSV = [
|
||||||
|
h: number,
|
||||||
|
s: number,
|
||||||
|
v: number
|
||||||
|
];
|
||||||
|
type CMYK = [
|
||||||
|
c: number,
|
||||||
|
m: number,
|
||||||
|
y: number,
|
||||||
|
k: number
|
||||||
|
];
|
||||||
|
type LAB = [
|
||||||
|
l: number,
|
||||||
|
a: number,
|
||||||
|
b: number
|
||||||
|
];
|
||||||
|
type LCH = [
|
||||||
|
l: number,
|
||||||
|
c: number,
|
||||||
|
h: number
|
||||||
|
];
|
||||||
|
type HCG = [
|
||||||
|
h: number,
|
||||||
|
c: number,
|
||||||
|
g: number
|
||||||
|
];
|
||||||
|
type HWB = [
|
||||||
|
h: number,
|
||||||
|
w: number,
|
||||||
|
b: number
|
||||||
|
];
|
||||||
|
type XYZ = [
|
||||||
|
x: number,
|
||||||
|
y: number,
|
||||||
|
z: number
|
||||||
|
];
|
||||||
|
type Apple = [
|
||||||
|
r16: number,
|
||||||
|
g16: number,
|
||||||
|
b16: number
|
||||||
|
];
|
||||||
|
type Gray = [
|
||||||
|
gray: number
|
||||||
|
];
|
||||||
|
type ANSI16 = number;
|
||||||
|
type ANSI256 = number;
|
||||||
|
type Keyword = string;
|
||||||
|
type HEX = string;
|
||||||
|
declare namespace route {
|
||||||
|
type rgb = {
|
||||||
|
hsl(from: RGB): HSL;
|
||||||
|
hsl(...from: RGB): HSL;
|
||||||
|
hsl(from: RGB): HSL;
|
||||||
|
hsl(...from: RGB): HSL;
|
||||||
|
hsv(from: RGB): HSV;
|
||||||
|
hsv(...from: RGB): HSV;
|
||||||
|
hwb(from: RGB): HWB;
|
||||||
|
hwb(...from: RGB): HWB;
|
||||||
|
cmyk(from: RGB): CMYK;
|
||||||
|
cmyk(...from: RGB): CMYK;
|
||||||
|
xyz(from: RGB): XYZ;
|
||||||
|
xyz(...from: RGB): XYZ;
|
||||||
|
lab(from: RGB): LAB;
|
||||||
|
lab(...from: RGB): LAB;
|
||||||
|
lch(from: RGB): LCH;
|
||||||
|
lch(...from: RGB): LCH;
|
||||||
|
hex(from: RGB): HEX;
|
||||||
|
hex(...from: RGB): HEX;
|
||||||
|
keyword(from: RGB): Keyword;
|
||||||
|
keyword(...from: RGB): Keyword;
|
||||||
|
ansi16(from: RGB): ANSI16;
|
||||||
|
ansi16(...from: RGB): ANSI16;
|
||||||
|
ansi256(from: RGB): ANSI256;
|
||||||
|
ansi256(...from: RGB): ANSI256;
|
||||||
|
hcg(from: RGB): HCG;
|
||||||
|
hcg(...from: RGB): HCG;
|
||||||
|
apple(from: RGB): Apple;
|
||||||
|
apple(...from: RGB): Apple;
|
||||||
|
gray(from: RGB): Gray;
|
||||||
|
gray(...from: RGB): Gray;
|
||||||
|
};
|
||||||
|
type hsl = {
|
||||||
|
rgb(from: HSL): RGB;
|
||||||
|
rgb(...from: HSL): RGB;
|
||||||
|
hsv(from: HSL): HSV;
|
||||||
|
hsv(...from: HSL): HSV;
|
||||||
|
hwb(from: HSL): HWB;
|
||||||
|
hwb(...from: HSL): HWB;
|
||||||
|
cmyk(from: HSL): CMYK;
|
||||||
|
cmyk(...from: HSL): CMYK;
|
||||||
|
xyz(from: HSL): XYZ;
|
||||||
|
xyz(...from: HSL): XYZ;
|
||||||
|
lab(from: HSL): LAB;
|
||||||
|
lab(...from: HSL): LAB;
|
||||||
|
lch(from: HSL): LCH;
|
||||||
|
lch(...from: HSL): LCH;
|
||||||
|
hex(from: HSL): HEX;
|
||||||
|
hex(...from: HSL): HEX;
|
||||||
|
keyword(from: HSL): Keyword;
|
||||||
|
keyword(...from: HSL): Keyword;
|
||||||
|
ansi16(from: HSL): ANSI16;
|
||||||
|
ansi16(...from: HSL): ANSI16;
|
||||||
|
ansi256(from: HSL): ANSI256;
|
||||||
|
ansi256(...from: HSL): ANSI256;
|
||||||
|
hcg(from: HSL): HCG;
|
||||||
|
hcg(...from: HSL): HCG;
|
||||||
|
apple(from: HSL): Apple;
|
||||||
|
apple(...from: HSL): Apple;
|
||||||
|
gray(from: HSL): Gray;
|
||||||
|
gray(...from: HSL): Gray;
|
||||||
|
};
|
||||||
|
type hsv = {
|
||||||
|
rgb(from: HSV): RGB;
|
||||||
|
rgb(...from: HSV): RGB;
|
||||||
|
hsl(from: HSV): HSL;
|
||||||
|
hsl(...from: HSV): HSL;
|
||||||
|
hwb(from: HSV): HWB;
|
||||||
|
hwb(...from: HSV): HWB;
|
||||||
|
cmyk(from: HSV): CMYK;
|
||||||
|
cmyk(...from: HSV): CMYK;
|
||||||
|
xyz(from: HSV): XYZ;
|
||||||
|
xyz(...from: HSV): XYZ;
|
||||||
|
lab(from: HSV): LAB;
|
||||||
|
lab(...from: HSV): LAB;
|
||||||
|
lch(from: HSV): LCH;
|
||||||
|
lch(...from: HSV): LCH;
|
||||||
|
hex(from: HSV): HEX;
|
||||||
|
hex(...from: HSV): HEX;
|
||||||
|
keyword(from: HSV): Keyword;
|
||||||
|
keyword(...from: HSV): Keyword;
|
||||||
|
ansi16(from: HSV): ANSI16;
|
||||||
|
ansi16(...from: HSV): ANSI16;
|
||||||
|
ansi256(from: HSV): ANSI256;
|
||||||
|
ansi256(...from: HSV): ANSI256;
|
||||||
|
hcg(from: HSV): HCG;
|
||||||
|
hcg(...from: HSV): HCG;
|
||||||
|
apple(from: HSV): Apple;
|
||||||
|
apple(...from: HSV): Apple;
|
||||||
|
gray(from: HSV): Gray;
|
||||||
|
gray(...from: HSV): Gray;
|
||||||
|
};
|
||||||
|
type hwb = {
|
||||||
|
rgb(from: HWB): RGB;
|
||||||
|
rgb(...from: HWB): RGB;
|
||||||
|
hsl(from: HWB): HSL;
|
||||||
|
hsl(...from: HWB): HSL;
|
||||||
|
hsv(from: HWB): HSV;
|
||||||
|
hsv(...from: HWB): HSV;
|
||||||
|
cmyk(from: HWB): CMYK;
|
||||||
|
cmyk(...from: HWB): CMYK;
|
||||||
|
xyz(from: HWB): XYZ;
|
||||||
|
xyz(...from: HWB): XYZ;
|
||||||
|
lab(from: HWB): LAB;
|
||||||
|
lab(...from: HWB): LAB;
|
||||||
|
lch(from: HWB): LCH;
|
||||||
|
lch(...from: HWB): LCH;
|
||||||
|
hex(from: HWB): HEX;
|
||||||
|
hex(...from: HWB): HEX;
|
||||||
|
keyword(from: HWB): Keyword;
|
||||||
|
keyword(...from: HWB): Keyword;
|
||||||
|
ansi16(from: HWB): ANSI16;
|
||||||
|
ansi16(...from: HWB): ANSI16;
|
||||||
|
ansi256(from: HWB): ANSI256;
|
||||||
|
ansi256(...from: HWB): ANSI256;
|
||||||
|
hcg(from: HWB): HCG;
|
||||||
|
hcg(...from: HWB): HCG;
|
||||||
|
apple(from: HWB): Apple;
|
||||||
|
apple(...from: HWB): Apple;
|
||||||
|
gray(from: HWB): Gray;
|
||||||
|
gray(...from: HWB): Gray;
|
||||||
|
};
|
||||||
|
type cmyk = {
|
||||||
|
rgb(from: CMYK): RGB;
|
||||||
|
rgb(...from: CMYK): RGB;
|
||||||
|
hsl(from: CMYK): HSL;
|
||||||
|
hsl(...from: CMYK): HSL;
|
||||||
|
hsv(from: CMYK): HSV;
|
||||||
|
hsv(...from: CMYK): HSV;
|
||||||
|
hwb(from: CMYK): HWB;
|
||||||
|
hwb(...from: CMYK): HWB;
|
||||||
|
xyz(from: CMYK): XYZ;
|
||||||
|
xyz(...from: CMYK): XYZ;
|
||||||
|
lab(from: CMYK): LAB;
|
||||||
|
lab(...from: CMYK): LAB;
|
||||||
|
lch(from: CMYK): LCH;
|
||||||
|
lch(...from: CMYK): LCH;
|
||||||
|
hex(from: CMYK): HEX;
|
||||||
|
hex(...from: CMYK): HEX;
|
||||||
|
keyword(from: CMYK): Keyword;
|
||||||
|
keyword(...from: CMYK): Keyword;
|
||||||
|
ansi16(from: CMYK): ANSI16;
|
||||||
|
ansi16(...from: CMYK): ANSI16;
|
||||||
|
ansi256(from: CMYK): ANSI256;
|
||||||
|
ansi256(...from: CMYK): ANSI256;
|
||||||
|
hcg(from: CMYK): HCG;
|
||||||
|
hcg(...from: CMYK): HCG;
|
||||||
|
apple(from: CMYK): Apple;
|
||||||
|
apple(...from: CMYK): Apple;
|
||||||
|
gray(from: CMYK): Gray;
|
||||||
|
gray(...from: CMYK): Gray;
|
||||||
|
};
|
||||||
|
type xyz = {
|
||||||
|
rgb(from: XYZ): RGB;
|
||||||
|
rgb(...from: XYZ): RGB;
|
||||||
|
hsl(from: XYZ): HSL;
|
||||||
|
hsl(...from: XYZ): HSL;
|
||||||
|
hsv(from: XYZ): HSV;
|
||||||
|
hsv(...from: XYZ): HSV;
|
||||||
|
hwb(from: XYZ): HWB;
|
||||||
|
hwb(...from: XYZ): HWB;
|
||||||
|
cmyk(from: XYZ): CMYK;
|
||||||
|
cmyk(...from: XYZ): CMYK;
|
||||||
|
lab(from: XYZ): LAB;
|
||||||
|
lab(...from: XYZ): LAB;
|
||||||
|
lch(from: XYZ): LCH;
|
||||||
|
lch(...from: XYZ): LCH;
|
||||||
|
hex(from: XYZ): HEX;
|
||||||
|
hex(...from: XYZ): HEX;
|
||||||
|
keyword(from: XYZ): Keyword;
|
||||||
|
keyword(...from: XYZ): Keyword;
|
||||||
|
ansi16(from: XYZ): ANSI16;
|
||||||
|
ansi16(...from: XYZ): ANSI16;
|
||||||
|
ansi256(from: XYZ): ANSI256;
|
||||||
|
ansi256(...from: XYZ): ANSI256;
|
||||||
|
hcg(from: XYZ): HCG;
|
||||||
|
hcg(...from: XYZ): HCG;
|
||||||
|
apple(from: XYZ): Apple;
|
||||||
|
apple(...from: XYZ): Apple;
|
||||||
|
gray(from: XYZ): Gray;
|
||||||
|
gray(...from: XYZ): Gray;
|
||||||
|
};
|
||||||
|
type lab = {
|
||||||
|
rgb(from: LAB): RGB;
|
||||||
|
rgb(...from: LAB): RGB;
|
||||||
|
hsl(from: LAB): HSL;
|
||||||
|
hsl(...from: LAB): HSL;
|
||||||
|
hsv(from: LAB): HSV;
|
||||||
|
hsv(...from: LAB): HSV;
|
||||||
|
hwb(from: LAB): HWB;
|
||||||
|
hwb(...from: LAB): HWB;
|
||||||
|
cmyk(from: LAB): CMYK;
|
||||||
|
cmyk(...from: LAB): CMYK;
|
||||||
|
xyz(from: LAB): XYZ;
|
||||||
|
xyz(...from: LAB): XYZ;
|
||||||
|
lch(from: LAB): LCH;
|
||||||
|
lch(...from: LAB): LCH;
|
||||||
|
hex(from: LAB): HEX;
|
||||||
|
hex(...from: LAB): HEX;
|
||||||
|
keyword(from: LAB): Keyword;
|
||||||
|
keyword(...from: LAB): Keyword;
|
||||||
|
ansi16(from: LAB): ANSI16;
|
||||||
|
ansi16(...from: LAB): ANSI16;
|
||||||
|
ansi256(from: LAB): ANSI256;
|
||||||
|
ansi256(...from: LAB): ANSI256;
|
||||||
|
hcg(from: LAB): HCG;
|
||||||
|
hcg(...from: LAB): HCG;
|
||||||
|
apple(from: LAB): Apple;
|
||||||
|
apple(...from: LAB): Apple;
|
||||||
|
gray(from: LAB): Gray;
|
||||||
|
gray(...from: LAB): Gray;
|
||||||
|
};
|
||||||
|
type lch = {
|
||||||
|
rgb(from: LCH): RGB;
|
||||||
|
rgb(...from: LCH): RGB;
|
||||||
|
hsl(from: LCH): HSL;
|
||||||
|
hsl(...from: LCH): HSL;
|
||||||
|
hsv(from: LCH): HSV;
|
||||||
|
hsv(...from: LCH): HSV;
|
||||||
|
hwb(from: LCH): HWB;
|
||||||
|
hwb(...from: LCH): HWB;
|
||||||
|
cmyk(from: LCH): CMYK;
|
||||||
|
cmyk(...from: LCH): CMYK;
|
||||||
|
xyz(from: LCH): XYZ;
|
||||||
|
xyz(...from: LCH): XYZ;
|
||||||
|
lab(from: LCH): LAB;
|
||||||
|
lab(...from: LCH): LAB;
|
||||||
|
hex(from: LCH): HEX;
|
||||||
|
hex(...from: LCH): HEX;
|
||||||
|
keyword(from: LCH): Keyword;
|
||||||
|
keyword(...from: LCH): Keyword;
|
||||||
|
ansi16(from: LCH): ANSI16;
|
||||||
|
ansi16(...from: LCH): ANSI16;
|
||||||
|
ansi256(from: LCH): ANSI256;
|
||||||
|
ansi256(...from: LCH): ANSI256;
|
||||||
|
hcg(from: LCH): HCG;
|
||||||
|
hcg(...from: LCH): HCG;
|
||||||
|
apple(from: LCH): Apple;
|
||||||
|
apple(...from: LCH): Apple;
|
||||||
|
gray(from: LCH): Gray;
|
||||||
|
gray(...from: LCH): Gray;
|
||||||
|
};
|
||||||
|
type hex = {
|
||||||
|
rgb(from: HEX): RGB;
|
||||||
|
hsl(from: HEX): HSL;
|
||||||
|
hsv(from: HEX): HSV;
|
||||||
|
hwb(from: HEX): HWB;
|
||||||
|
cmyk(from: HEX): CMYK;
|
||||||
|
xyz(from: HEX): XYZ;
|
||||||
|
lab(from: HEX): LAB;
|
||||||
|
lch(from: HEX): LCH;
|
||||||
|
keyword(from: HEX): Keyword;
|
||||||
|
ansi16(from: HEX): ANSI16;
|
||||||
|
ansi256(from: HEX): ANSI256;
|
||||||
|
hcg(from: HEX): HCG;
|
||||||
|
apple(from: HEX): Apple;
|
||||||
|
gray(from: HEX): Gray;
|
||||||
|
};
|
||||||
|
type keyword = {
|
||||||
|
rgb(from: Keyword): RGB;
|
||||||
|
hsl(from: Keyword): HSL;
|
||||||
|
hsv(from: Keyword): HSV;
|
||||||
|
hwb(from: Keyword): HWB;
|
||||||
|
cmyk(from: Keyword): CMYK;
|
||||||
|
xyz(from: Keyword): XYZ;
|
||||||
|
lab(from: Keyword): LAB;
|
||||||
|
lch(from: Keyword): LCH;
|
||||||
|
hex(from: Keyword): HEX;
|
||||||
|
ansi16(from: Keyword): ANSI16;
|
||||||
|
ansi256(from: Keyword): ANSI256;
|
||||||
|
hcg(from: Keyword): HCG;
|
||||||
|
apple(from: Keyword): Apple;
|
||||||
|
gray(from: Keyword): Gray;
|
||||||
|
};
|
||||||
|
type ansi16 = {
|
||||||
|
rgb(from: ANSI16): RGB;
|
||||||
|
hsl(from: ANSI16): HSL;
|
||||||
|
hsv(from: ANSI16): HSV;
|
||||||
|
hwb(from: ANSI16): HWB;
|
||||||
|
cmyk(from: ANSI16): CMYK;
|
||||||
|
xyz(from: ANSI16): XYZ;
|
||||||
|
lab(from: ANSI16): LAB;
|
||||||
|
lch(from: ANSI16): LCH;
|
||||||
|
hex(from: ANSI16): HEX;
|
||||||
|
keyword(from: ANSI16): Keyword;
|
||||||
|
ansi256(from: ANSI16): ANSI256;
|
||||||
|
hcg(from: ANSI16): HCG;
|
||||||
|
apple(from: ANSI16): Apple;
|
||||||
|
gray(from: ANSI16): Gray;
|
||||||
|
};
|
||||||
|
type ansi256 = {
|
||||||
|
rgb(from: ANSI256): RGB;
|
||||||
|
hsl(from: ANSI256): HSL;
|
||||||
|
hsv(from: ANSI256): HSV;
|
||||||
|
hwb(from: ANSI256): HWB;
|
||||||
|
cmyk(from: ANSI256): CMYK;
|
||||||
|
xyz(from: ANSI256): XYZ;
|
||||||
|
lab(from: ANSI256): LAB;
|
||||||
|
lch(from: ANSI256): LCH;
|
||||||
|
hex(from: ANSI256): HEX;
|
||||||
|
keyword(from: ANSI256): Keyword;
|
||||||
|
ansi16(from: ANSI256): ANSI16;
|
||||||
|
hcg(from: ANSI256): HCG;
|
||||||
|
apple(from: ANSI256): Apple;
|
||||||
|
gray(from: ANSI256): Gray;
|
||||||
|
};
|
||||||
|
type hcg = {
|
||||||
|
rgb(from: HCG): RGB;
|
||||||
|
rgb(...from: HCG): RGB;
|
||||||
|
hsl(from: HCG): HSL;
|
||||||
|
hsl(...from: HCG): HSL;
|
||||||
|
hsv(from: HCG): HSV;
|
||||||
|
hsv(...from: HCG): HSV;
|
||||||
|
hwb(from: HCG): HWB;
|
||||||
|
hwb(...from: HCG): HWB;
|
||||||
|
cmyk(from: HCG): CMYK;
|
||||||
|
cmyk(...from: HCG): CMYK;
|
||||||
|
xyz(from: HCG): XYZ;
|
||||||
|
xyz(...from: HCG): XYZ;
|
||||||
|
lab(from: HCG): LAB;
|
||||||
|
lab(...from: HCG): LAB;
|
||||||
|
lch(from: HCG): LCH;
|
||||||
|
lch(...from: HCG): LCH;
|
||||||
|
hex(from: HCG): HEX;
|
||||||
|
hex(...from: HCG): HEX;
|
||||||
|
keyword(from: HCG): Keyword;
|
||||||
|
keyword(...from: HCG): Keyword;
|
||||||
|
ansi16(from: HCG): ANSI16;
|
||||||
|
ansi16(...from: HCG): ANSI16;
|
||||||
|
ansi256(from: HCG): ANSI256;
|
||||||
|
ansi256(...from: HCG): ANSI256;
|
||||||
|
apple(from: HCG): Apple;
|
||||||
|
apple(...from: HCG): Apple;
|
||||||
|
gray(from: HCG): Gray;
|
||||||
|
gray(...from: HCG): Gray;
|
||||||
|
};
|
||||||
|
type apple = {
|
||||||
|
rgb(from: Apple): RGB;
|
||||||
|
rgb(...from: Apple): RGB;
|
||||||
|
hsl(from: Apple): HSL;
|
||||||
|
hsl(...from: Apple): HSL;
|
||||||
|
hsv(from: Apple): HSV;
|
||||||
|
hsv(...from: Apple): HSV;
|
||||||
|
hwb(from: Apple): HWB;
|
||||||
|
hwb(...from: Apple): HWB;
|
||||||
|
cmyk(from: Apple): CMYK;
|
||||||
|
cmyk(...from: Apple): CMYK;
|
||||||
|
xyz(from: Apple): XYZ;
|
||||||
|
xyz(...from: Apple): XYZ;
|
||||||
|
lab(from: Apple): LAB;
|
||||||
|
lab(...from: Apple): LAB;
|
||||||
|
lch(from: Apple): LCH;
|
||||||
|
lch(...from: Apple): LCH;
|
||||||
|
hex(from: Apple): HEX;
|
||||||
|
hex(...from: Apple): HEX;
|
||||||
|
keyword(from: Apple): Keyword;
|
||||||
|
keyword(...from: Apple): Keyword;
|
||||||
|
ansi16(from: Apple): ANSI16;
|
||||||
|
ansi16(...from: Apple): ANSI16;
|
||||||
|
ansi256(from: Apple): ANSI256;
|
||||||
|
ansi256(...from: Apple): ANSI256;
|
||||||
|
hcg(from: Apple): HCG;
|
||||||
|
hcg(...from: Apple): HCG;
|
||||||
|
gray(from: Apple): Gray;
|
||||||
|
gray(...from: Apple): Gray;
|
||||||
|
};
|
||||||
|
type gray = {
|
||||||
|
rgb(from: Gray): RGB;
|
||||||
|
rgb(...from: Gray): RGB;
|
||||||
|
hsl(from: Gray): HSL;
|
||||||
|
hsl(...from: Gray): HSL;
|
||||||
|
hsv(from: Gray): HSV;
|
||||||
|
hsv(...from: Gray): HSV;
|
||||||
|
hwb(from: Gray): HWB;
|
||||||
|
hwb(...from: Gray): HWB;
|
||||||
|
cmyk(from: Gray): CMYK;
|
||||||
|
cmyk(...from: Gray): CMYK;
|
||||||
|
xyz(from: Gray): XYZ;
|
||||||
|
xyz(...from: Gray): XYZ;
|
||||||
|
lab(from: Gray): LAB;
|
||||||
|
lab(...from: Gray): LAB;
|
||||||
|
lch(from: Gray): LCH;
|
||||||
|
lch(...from: Gray): LCH;
|
||||||
|
hex(from: Gray): HEX;
|
||||||
|
hex(...from: Gray): HEX;
|
||||||
|
keyword(from: Gray): Keyword;
|
||||||
|
keyword(...from: Gray): Keyword;
|
||||||
|
ansi16(from: Gray): ANSI16;
|
||||||
|
ansi16(...from: Gray): ANSI16;
|
||||||
|
ansi256(from: Gray): ANSI256;
|
||||||
|
ansi256(...from: Gray): ANSI256;
|
||||||
|
hcg(from: Gray): HCG;
|
||||||
|
hcg(...from: Gray): HCG;
|
||||||
|
apple(from: Gray): Apple;
|
||||||
|
apple(...from: Gray): Apple;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
declare function route(fromModel: "rgb"): route.rgb;
|
||||||
|
declare function route(fromModel: "hsl"): route.hsl;
|
||||||
|
declare function route(fromModel: "hsv"): route.hsv;
|
||||||
|
declare function route(fromModel: "hwb"): route.hwb;
|
||||||
|
declare function route(fromModel: "cmyk"): route.cmyk;
|
||||||
|
declare function route(fromModel: "xyz"): route.xyz;
|
||||||
|
declare function route(fromModel: "lab"): route.lab;
|
||||||
|
declare function route(fromModel: "lch"): route.lch;
|
||||||
|
declare function route(fromModel: "hex"): route.hex;
|
||||||
|
declare function route(fromModel: "keyword"): route.keyword;
|
||||||
|
declare function route(fromModel: "ansi16"): route.ansi16;
|
||||||
|
declare function route(fromModel: "ansi256"): route.ansi256;
|
||||||
|
declare function route(fromModel: "hcg"): route.hcg;
|
||||||
|
declare function route(fromModel: "apple"): route.apple;
|
||||||
|
declare function route(fromModel: "gray"): route.gray;
|
||||||
|
type Convert = {
|
||||||
|
rgb: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "rgb";
|
||||||
|
hsl: {
|
||||||
|
(...rgb: RGB): HSL;
|
||||||
|
raw: (...rgb: RGB) => HSL;
|
||||||
|
};
|
||||||
|
hsv: {
|
||||||
|
(...rgb: RGB): HSV;
|
||||||
|
raw: (...rgb: RGB) => HSV;
|
||||||
|
};
|
||||||
|
hwb: {
|
||||||
|
(...rgb: RGB): HWB;
|
||||||
|
raw: (...rgb: RGB) => HWB;
|
||||||
|
};
|
||||||
|
hcg: {
|
||||||
|
(...rgb: RGB): HCG;
|
||||||
|
raw: (...rgb: RGB) => HCG;
|
||||||
|
};
|
||||||
|
cmyk: {
|
||||||
|
(...rgb: RGB): CMYK;
|
||||||
|
raw: (...rgb: RGB) => CMYK;
|
||||||
|
};
|
||||||
|
keyword: {
|
||||||
|
(...rgb: RGB): Keyword;
|
||||||
|
raw: (...rgb: RGB) => Keyword;
|
||||||
|
};
|
||||||
|
ansi16: {
|
||||||
|
(...rgb: RGB): ANSI16;
|
||||||
|
raw: (...rgb: RGB) => ANSI16;
|
||||||
|
};
|
||||||
|
ansi256: {
|
||||||
|
(...rgb: RGB): ANSI256;
|
||||||
|
raw: (...rgb: RGB) => ANSI256;
|
||||||
|
};
|
||||||
|
apple: {
|
||||||
|
(...rgb: RGB): Apple;
|
||||||
|
raw: (...rgb: RGB) => Apple;
|
||||||
|
};
|
||||||
|
hex: {
|
||||||
|
(...rgb: RGB): HEX;
|
||||||
|
raw: (...rgb: RGB) => HEX;
|
||||||
|
};
|
||||||
|
gray: {
|
||||||
|
(...rgb: RGB): Gray;
|
||||||
|
raw: (...rgb: RGB) => Gray;
|
||||||
|
};
|
||||||
|
} & route.rgb & {
|
||||||
|
[F in keyof route.rgb]: {
|
||||||
|
raw: route.rgb[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
keyword: {
|
||||||
|
channels: Channels;
|
||||||
|
rgb: {
|
||||||
|
(keyword: Keyword): RGB;
|
||||||
|
raw: (keyword: Keyword) => RGB;
|
||||||
|
};
|
||||||
|
} & route.keyword & {
|
||||||
|
[F in keyof route.keyword]: {
|
||||||
|
raw: route.keyword[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
hsl: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "hsl";
|
||||||
|
rgb: {
|
||||||
|
(...hsl: HSL): RGB;
|
||||||
|
raw: (...hsl: HSL) => RGB;
|
||||||
|
};
|
||||||
|
hsv: {
|
||||||
|
(...hsl: HSL): HSV;
|
||||||
|
raw: (...hsl: HSL) => HSV;
|
||||||
|
};
|
||||||
|
hcg: {
|
||||||
|
(...hsl: HSL): HCG;
|
||||||
|
raw: (...hsl: HSL) => HCG;
|
||||||
|
};
|
||||||
|
} & route.hsl & {
|
||||||
|
[F in keyof route.hsl]: {
|
||||||
|
raw: route.hsl[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
hsv: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "hsv";
|
||||||
|
hcg: {
|
||||||
|
(...hsv: HSV): HCG;
|
||||||
|
raw: (...hsv: HSV) => HCG;
|
||||||
|
};
|
||||||
|
rgb: {
|
||||||
|
(...hsv: HSV): RGB;
|
||||||
|
raw: (...hsv: HSV) => RGB;
|
||||||
|
};
|
||||||
|
hsv: {
|
||||||
|
(...hsv: HSV): HSV;
|
||||||
|
raw: (...hsv: HSV) => HSV;
|
||||||
|
};
|
||||||
|
hsl: {
|
||||||
|
(...hsv: HSV): HSL;
|
||||||
|
raw: (...hsv: HSV) => HSL;
|
||||||
|
};
|
||||||
|
hwb: {
|
||||||
|
(...hsv: HSV): HWB;
|
||||||
|
raw: (...hsv: HSV) => HWB;
|
||||||
|
};
|
||||||
|
ansi16: {
|
||||||
|
(...hsv: HSV): ANSI16;
|
||||||
|
raw: (...hsv: HSV) => ANSI16;
|
||||||
|
};
|
||||||
|
} & route.hsv & {
|
||||||
|
[F in keyof route.hsv]: {
|
||||||
|
raw: route.hsv[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
hwb: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "hwb";
|
||||||
|
hcg: {
|
||||||
|
(...hwb: HWB): HCG;
|
||||||
|
raw: (...hwb: HWB) => HCG;
|
||||||
|
};
|
||||||
|
rgb: {
|
||||||
|
(...hwb: HWB): RGB;
|
||||||
|
raw: (...hwb: HWB) => RGB;
|
||||||
|
};
|
||||||
|
} & route.hwb & {
|
||||||
|
[F in keyof route.hwb]: {
|
||||||
|
raw: route.hwb[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
cmyk: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "cmyk";
|
||||||
|
rgb: {
|
||||||
|
(...cmyk: CMYK): RGB;
|
||||||
|
raw: (...cmyk: CMYK) => RGB;
|
||||||
|
};
|
||||||
|
} & route.cmyk & {
|
||||||
|
[F in keyof route.cmyk]: {
|
||||||
|
raw: route.cmyk[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
xyz: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "xyz";
|
||||||
|
rgb: {
|
||||||
|
(...xyz: XYZ): RGB;
|
||||||
|
raw: (...xyz: XYZ) => RGB;
|
||||||
|
};
|
||||||
|
lab: {
|
||||||
|
(...xyz: XYZ): LAB;
|
||||||
|
raw: (...xyz: XYZ) => LAB;
|
||||||
|
};
|
||||||
|
} & route.xyz & {
|
||||||
|
[F in keyof route.xyz]: {
|
||||||
|
raw: route.xyz[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
lab: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "lab";
|
||||||
|
xyz: {
|
||||||
|
(...lab: LAB): XYZ;
|
||||||
|
raw: (...lab: LAB) => XYZ;
|
||||||
|
};
|
||||||
|
lch: {
|
||||||
|
(...lab: LAB): LCH;
|
||||||
|
raw: (...lab: LAB) => LCH;
|
||||||
|
};
|
||||||
|
} & route.lab & {
|
||||||
|
[F in keyof route.lab]: {
|
||||||
|
raw: route.lab[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
lch: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: "lch";
|
||||||
|
lab: {
|
||||||
|
(...lch: LCH): LAB;
|
||||||
|
raw: (...lch: LCH) => LAB;
|
||||||
|
};
|
||||||
|
} & route.lch & {
|
||||||
|
[F in keyof route.lch]: {
|
||||||
|
raw: route.lch[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
hex: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: [
|
||||||
|
"hex"
|
||||||
|
];
|
||||||
|
rgb: {
|
||||||
|
(hex: HEX): RGB;
|
||||||
|
raw: (hex: HEX) => RGB;
|
||||||
|
};
|
||||||
|
} & route.hex & {
|
||||||
|
[F in keyof route.hex]: {
|
||||||
|
raw: route.hex[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
ansi16: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: [
|
||||||
|
"ansi16"
|
||||||
|
];
|
||||||
|
rgb: {
|
||||||
|
(ansi16: ANSI16): RGB;
|
||||||
|
raw: (ansi16: ANSI16) => RGB;
|
||||||
|
};
|
||||||
|
} & route.ansi16 & {
|
||||||
|
[F in keyof route.ansi16]: {
|
||||||
|
raw: route.ansi16[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
ansi256: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: [
|
||||||
|
"ansi256"
|
||||||
|
];
|
||||||
|
rgb: {
|
||||||
|
(ansi256: ANSI256): RGB;
|
||||||
|
raw: (ansi256: ANSI256) => RGB;
|
||||||
|
};
|
||||||
|
} & route.ansi256 & {
|
||||||
|
[F in keyof route.ansi256]: {
|
||||||
|
raw: route.ansi256[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
hcg: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: [
|
||||||
|
"h",
|
||||||
|
"c",
|
||||||
|
"g"
|
||||||
|
];
|
||||||
|
rgb: {
|
||||||
|
(...hcg: HCG): RGB;
|
||||||
|
raw: (...hcg: HCG) => RGB;
|
||||||
|
};
|
||||||
|
hsv: {
|
||||||
|
(...hcg: HCG): HSV;
|
||||||
|
raw: (...hcg: HCG) => HSV;
|
||||||
|
};
|
||||||
|
hwb: {
|
||||||
|
(...hcg: HCG): HWB;
|
||||||
|
raw: (...hcg: HCG) => HWB;
|
||||||
|
};
|
||||||
|
} & route.hcg & {
|
||||||
|
[F in keyof route.hcg]: {
|
||||||
|
raw: route.hcg[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
apple: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: [
|
||||||
|
"r16",
|
||||||
|
"g16",
|
||||||
|
"b16"
|
||||||
|
];
|
||||||
|
rgb: {
|
||||||
|
(...apple: Apple): RGB;
|
||||||
|
raw: (...apple: Apple) => RGB;
|
||||||
|
};
|
||||||
|
} & route.apple & {
|
||||||
|
[F in keyof route.apple]: {
|
||||||
|
raw: route.apple[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
gray: {
|
||||||
|
channels: Channels;
|
||||||
|
labels: [
|
||||||
|
"gray"
|
||||||
|
];
|
||||||
|
rgb: {
|
||||||
|
(...gray: Gray): RGB;
|
||||||
|
raw: (...gray: Gray) => RGB;
|
||||||
|
};
|
||||||
|
hsl: {
|
||||||
|
(...gray: Gray): HSL;
|
||||||
|
raw: (...gray: Gray) => HSL;
|
||||||
|
};
|
||||||
|
hsv: {
|
||||||
|
(...gray: Gray): HSV;
|
||||||
|
raw: (...gray: Gray) => HSV;
|
||||||
|
};
|
||||||
|
hwb: {
|
||||||
|
(...gray: Gray): HWB;
|
||||||
|
raw: (...gray: Gray) => HWB;
|
||||||
|
};
|
||||||
|
cmyk: {
|
||||||
|
(...gray: Gray): CMYK;
|
||||||
|
raw: (...gray: Gray) => CMYK;
|
||||||
|
};
|
||||||
|
lab: {
|
||||||
|
(...gray: Gray): LAB;
|
||||||
|
raw: (...gray: Gray) => LAB;
|
||||||
|
};
|
||||||
|
hex: {
|
||||||
|
(...gray: Gray): HEX;
|
||||||
|
raw: (...gray: Gray) => HEX;
|
||||||
|
};
|
||||||
|
} & route.gray & {
|
||||||
|
[F in keyof route.gray]: {
|
||||||
|
raw: route.gray[F];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
declare const convert: Convert;
|
||||||
|
export type ColorLike = ColorInstance | string | ArrayLike<number> | number | Record<string, any>;
|
||||||
|
export type ColorJson = {
|
||||||
|
model: string;
|
||||||
|
color: number[];
|
||||||
|
valpha: number;
|
||||||
|
};
|
||||||
|
export type ColorObject = {
|
||||||
|
alpha?: number | undefined;
|
||||||
|
} & Record<string, number>;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/consistent-type-definitions
|
||||||
|
export interface ColorInstance {
|
||||||
|
toString(): string;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
||||||
|
toJSON(): ColorJson;
|
||||||
|
string(places?: number): string;
|
||||||
|
percentString(places?: number): string;
|
||||||
|
array(): number[];
|
||||||
|
object(): ColorObject;
|
||||||
|
unitArray(): number[];
|
||||||
|
unitObject(): {
|
||||||
|
r: number;
|
||||||
|
g: number;
|
||||||
|
b: number;
|
||||||
|
alpha?: number | undefined;
|
||||||
|
};
|
||||||
|
round(places?: number): ColorInstance;
|
||||||
|
alpha(): number;
|
||||||
|
alpha(value: number): ColorInstance;
|
||||||
|
red(): number;
|
||||||
|
red(value: number): ColorInstance;
|
||||||
|
green(): number;
|
||||||
|
green(value: number): ColorInstance;
|
||||||
|
blue(): number;
|
||||||
|
blue(value: number): ColorInstance;
|
||||||
|
hue(): number;
|
||||||
|
hue(value: number): ColorInstance;
|
||||||
|
saturationl(): number;
|
||||||
|
saturationl(value: number): ColorInstance;
|
||||||
|
lightness(): number;
|
||||||
|
lightness(value: number): ColorInstance;
|
||||||
|
saturationv(): number;
|
||||||
|
saturationv(value: number): ColorInstance;
|
||||||
|
value(): number;
|
||||||
|
value(value: number): ColorInstance;
|
||||||
|
chroma(): number;
|
||||||
|
chroma(value: number): ColorInstance;
|
||||||
|
gray(): number;
|
||||||
|
gray(value: number): ColorInstance;
|
||||||
|
white(): number;
|
||||||
|
white(value: number): ColorInstance;
|
||||||
|
wblack(): number;
|
||||||
|
wblack(value: number): ColorInstance;
|
||||||
|
cyan(): number;
|
||||||
|
cyan(value: number): ColorInstance;
|
||||||
|
magenta(): number;
|
||||||
|
magenta(value: number): ColorInstance;
|
||||||
|
yellow(): number;
|
||||||
|
yellow(value: number): ColorInstance;
|
||||||
|
black(): number;
|
||||||
|
black(value: number): ColorInstance;
|
||||||
|
x(): number;
|
||||||
|
x(value: number): ColorInstance;
|
||||||
|
y(): number;
|
||||||
|
y(value: number): ColorInstance;
|
||||||
|
z(): number;
|
||||||
|
z(value: number): ColorInstance;
|
||||||
|
l(): number;
|
||||||
|
l(value: number): ColorInstance;
|
||||||
|
a(): number;
|
||||||
|
a(value: number): ColorInstance;
|
||||||
|
b(): number;
|
||||||
|
b(value: number): ColorInstance;
|
||||||
|
keyword(): string;
|
||||||
|
keyword<V extends string>(value: V): ColorInstance;
|
||||||
|
hex(): string;
|
||||||
|
hex<V extends string>(value: V): ColorInstance;
|
||||||
|
hexa(): string;
|
||||||
|
hexa<V extends string>(value: V): ColorInstance;
|
||||||
|
rgbNumber(): number;
|
||||||
|
luminosity(): number;
|
||||||
|
contrast(color2: ColorInstance): number;
|
||||||
|
level(color2: ColorInstance): "AAA" | "AA" | "";
|
||||||
|
isDark(): boolean;
|
||||||
|
isLight(): boolean;
|
||||||
|
negate(): ColorInstance;
|
||||||
|
lighten(ratio: number): ColorInstance;
|
||||||
|
darken(ratio: number): ColorInstance;
|
||||||
|
saturate(ratio: number): ColorInstance;
|
||||||
|
desaturate(ratio: number): ColorInstance;
|
||||||
|
whiten(ratio: number): ColorInstance;
|
||||||
|
blacken(ratio: number): ColorInstance;
|
||||||
|
grayscale(): ColorInstance;
|
||||||
|
fade(ratio: number): ColorInstance;
|
||||||
|
opaquer(ratio: number): ColorInstance;
|
||||||
|
rotate(degrees: number): ColorInstance;
|
||||||
|
mix(mixinColor: ColorInstance, weight?: number): ColorInstance;
|
||||||
|
rgb(...arguments_: number[]): ColorInstance;
|
||||||
|
hsl(...arguments_: number[]): ColorInstance;
|
||||||
|
hsv(...arguments_: number[]): ColorInstance;
|
||||||
|
hwb(...arguments_: number[]): ColorInstance;
|
||||||
|
cmyk(...arguments_: number[]): ColorInstance;
|
||||||
|
xyz(...arguments_: number[]): ColorInstance;
|
||||||
|
lab(...arguments_: number[]): ColorInstance;
|
||||||
|
lch(...arguments_: number[]): ColorInstance;
|
||||||
|
ansi16(...arguments_: number[]): ColorInstance;
|
||||||
|
ansi256(...arguments_: number[]): ColorInstance;
|
||||||
|
hcg(...arguments_: number[]): ColorInstance;
|
||||||
|
apple(...arguments_: number[]): ColorInstance;
|
||||||
|
}
|
||||||
|
export type ColorConstructor = {
|
||||||
|
(object?: ColorLike, model?: keyof (typeof convert)): ColorInstance;
|
||||||
|
new (object?: ColorLike, model?: keyof (typeof convert)): ColorInstance;
|
||||||
|
rgb(...value: number[]): ColorInstance;
|
||||||
|
rgb(color: ColorLike): ColorInstance;
|
||||||
|
hsl(...value: number[]): ColorInstance;
|
||||||
|
hsl(color: ColorLike): ColorInstance;
|
||||||
|
hsv(...value: number[]): ColorInstance;
|
||||||
|
hsv(color: ColorLike): ColorInstance;
|
||||||
|
hwb(...value: number[]): ColorInstance;
|
||||||
|
hwb(color: ColorLike): ColorInstance;
|
||||||
|
cmyk(...value: number[]): ColorInstance;
|
||||||
|
cmyk(color: ColorLike): ColorInstance;
|
||||||
|
xyz(...value: number[]): ColorInstance;
|
||||||
|
xyz(color: ColorLike): ColorInstance;
|
||||||
|
lab(...value: number[]): ColorInstance;
|
||||||
|
lab(color: ColorLike): ColorInstance;
|
||||||
|
lch(...value: number[]): ColorInstance;
|
||||||
|
lch(color: ColorLike): ColorInstance;
|
||||||
|
ansi16(...value: number[]): ColorInstance;
|
||||||
|
ansi16(color: ColorLike): ColorInstance;
|
||||||
|
ansi256(...value: number[]): ColorInstance;
|
||||||
|
ansi256(color: ColorLike): ColorInstance;
|
||||||
|
hcg(...value: number[]): ColorInstance;
|
||||||
|
hcg(color: ColorLike): ColorInstance;
|
||||||
|
apple(...value: number[]): ColorInstance;
|
||||||
|
apple(color: ColorLike): ColorInstance;
|
||||||
|
};
|
||||||
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
||||||
|
declare const Color: ColorConstructor;
|
||||||
|
|
||||||
|
export {
|
||||||
|
Color as default,
|
||||||
|
};
|
||||||
|
|
||||||
|
export {};
|
||||||
58
node_modules/@img/colour/package.json
generated
vendored
Normal file
58
node_modules/@img/colour/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,58 @@
|
||||||
|
{
|
||||||
|
"name": "@img/colour",
|
||||||
|
"version": "1.1.0",
|
||||||
|
"description": "The ESM-only 'color' package made compatible for use with CommonJS runtimes",
|
||||||
|
"license": "MIT",
|
||||||
|
"main": "index.cjs",
|
||||||
|
"types": "index.d.ts",
|
||||||
|
"exports": {
|
||||||
|
".": {
|
||||||
|
"types": "./index.d.ts",
|
||||||
|
"require": "./index.cjs",
|
||||||
|
"default": "./index.cjs"
|
||||||
|
},
|
||||||
|
"./package.json": "./package.json"
|
||||||
|
},
|
||||||
|
"authors": [
|
||||||
|
"Heather Arthur <fayearthur@gmail.com>",
|
||||||
|
"Josh Junon <josh@junon.me>",
|
||||||
|
"Maxime Thirouin",
|
||||||
|
"Dyma Ywanov <dfcreative@gmail.com>",
|
||||||
|
"LitoMore (https://github.com/LitoMore)"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"color.cjs",
|
||||||
|
"index.d.ts"
|
||||||
|
],
|
||||||
|
"publishConfig": {
|
||||||
|
"access": "public"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "git+https://github.com/lovell/colour.git"
|
||||||
|
},
|
||||||
|
"type": "commonjs",
|
||||||
|
"keywords": [
|
||||||
|
"color",
|
||||||
|
"colour",
|
||||||
|
"cjs",
|
||||||
|
"commonjs"
|
||||||
|
],
|
||||||
|
"scripts": {
|
||||||
|
"build:cjs": "esbuild node_modules/color/index.js --bundle --platform=node --outfile=color.cjs",
|
||||||
|
"build:dts": "dts-bundle-generator ./dts-src.ts -o index.d.ts --project tsconfig.build.json --external-inlines color --external-inlines color-convert --export-referenced-types=false",
|
||||||
|
"build": "npm run build:cjs && npm run build:dts",
|
||||||
|
"test": "node --test"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"color": "5.0.3",
|
||||||
|
"color-convert": "3.1.3",
|
||||||
|
"color-name": "2.1.0",
|
||||||
|
"color-string": "2.1.4",
|
||||||
|
"dts-bundle-generator": "^9.5.1",
|
||||||
|
"esbuild": "^0.27.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
191
node_modules/@img/sharp-darwin-arm64/LICENSE
generated
vendored
Normal file
191
node_modules/@img/sharp-darwin-arm64/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,191 @@
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction, and
|
||||||
|
distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by the copyright
|
||||||
|
owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all other entities
|
||||||
|
that control, are controlled by, or are under common control with that entity.
|
||||||
|
For the purposes of this definition, "control" means (i) the power, direct or
|
||||||
|
indirect, to cause the direction or management of such entity, whether by
|
||||||
|
contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity exercising
|
||||||
|
permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications, including
|
||||||
|
but not limited to software source code, documentation source, and configuration
|
||||||
|
files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical transformation or
|
||||||
|
translation of a Source form, including but not limited to compiled object code,
|
||||||
|
generated documentation, and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or Object form, made
|
||||||
|
available under the License, as indicated by a copyright notice that is included
|
||||||
|
in or attached to the work (an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object form, that
|
||||||
|
is based on (or derived from) the Work and for which the editorial revisions,
|
||||||
|
annotations, elaborations, or other modifications represent, as a whole, an
|
||||||
|
original work of authorship. For the purposes of this License, Derivative Works
|
||||||
|
shall not include works that remain separable from, or merely link (or bind by
|
||||||
|
name) to the interfaces of, the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including the original version
|
||||||
|
of the Work and any modifications or additions to that Work or Derivative Works
|
||||||
|
thereof, that is intentionally submitted to Licensor for inclusion in the Work
|
||||||
|
by the copyright owner or by an individual or Legal Entity authorized to submit
|
||||||
|
on behalf of the copyright owner. For the purposes of this definition,
|
||||||
|
"submitted" means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems, and
|
||||||
|
issue tracking systems that are managed by, or on behalf of, the Licensor for
|
||||||
|
the purpose of discussing and improving the Work, but excluding communication
|
||||||
|
that is conspicuously marked or otherwise designated in writing by the copyright
|
||||||
|
owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity on behalf
|
||||||
|
of whom a Contribution has been received by Licensor and subsequently
|
||||||
|
incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License.
|
||||||
|
|
||||||
|
Subject to the terms and conditions of this License, each Contributor hereby
|
||||||
|
grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
|
||||||
|
irrevocable copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the Work and such
|
||||||
|
Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License.
|
||||||
|
|
||||||
|
Subject to the terms and conditions of this License, each Contributor hereby
|
||||||
|
grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
|
||||||
|
irrevocable (except as stated in this section) patent license to make, have
|
||||||
|
made, use, offer to sell, sell, import, and otherwise transfer the Work, where
|
||||||
|
such license applies only to those patent claims licensable by such Contributor
|
||||||
|
that are necessarily infringed by their Contribution(s) alone or by combination
|
||||||
|
of their Contribution(s) with the Work to which such Contribution(s) was
|
||||||
|
submitted. If You institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work or a
|
||||||
|
Contribution incorporated within the Work constitutes direct or contributory
|
||||||
|
patent infringement, then any patent licenses granted to You under this License
|
||||||
|
for that Work shall terminate as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution.
|
||||||
|
|
||||||
|
You may reproduce and distribute copies of the Work or Derivative Works thereof
|
||||||
|
in any medium, with or without modifications, and in Source or Object form,
|
||||||
|
provided that You meet the following conditions:
|
||||||
|
|
||||||
|
You must give any other recipients of the Work or Derivative Works a copy of
|
||||||
|
this License; and
|
||||||
|
You must cause any modified files to carry prominent notices stating that You
|
||||||
|
changed the files; and
|
||||||
|
You must retain, in the Source form of any Derivative Works that You distribute,
|
||||||
|
all copyright, patent, trademark, and attribution notices from the Source form
|
||||||
|
of the Work, excluding those notices that do not pertain to any part of the
|
||||||
|
Derivative Works; and
|
||||||
|
If the Work includes a "NOTICE" text file as part of its distribution, then any
|
||||||
|
Derivative Works that You distribute must include a readable copy of the
|
||||||
|
attribution notices contained within such NOTICE file, excluding those notices
|
||||||
|
that do not pertain to any part of the Derivative Works, in at least one of the
|
||||||
|
following places: within a NOTICE text file distributed as part of the
|
||||||
|
Derivative Works; within the Source form or documentation, if provided along
|
||||||
|
with the Derivative Works; or, within a display generated by the Derivative
|
||||||
|
Works, if and wherever such third-party notices normally appear. The contents of
|
||||||
|
the NOTICE file are for informational purposes only and do not modify the
|
||||||
|
License. You may add Your own attribution notices within Derivative Works that
|
||||||
|
You distribute, alongside or as an addendum to the NOTICE text from the Work,
|
||||||
|
provided that such additional attribution notices cannot be construed as
|
||||||
|
modifying the License.
|
||||||
|
You may add Your own copyright statement to Your modifications and may provide
|
||||||
|
additional or different license terms and conditions for use, reproduction, or
|
||||||
|
distribution of Your modifications, or for any such Derivative Works as a whole,
|
||||||
|
provided Your use, reproduction, and distribution of the Work otherwise complies
|
||||||
|
with the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions.
|
||||||
|
|
||||||
|
Unless You explicitly state otherwise, any Contribution intentionally submitted
|
||||||
|
for inclusion in the Work by You to the Licensor shall be under the terms and
|
||||||
|
conditions of this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify the terms of
|
||||||
|
any separate license agreement you may have executed with Licensor regarding
|
||||||
|
such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks.
|
||||||
|
|
||||||
|
This License does not grant permission to use the trade names, trademarks,
|
||||||
|
service marks, or product names of the Licensor, except as required for
|
||||||
|
reasonable and customary use in describing the origin of the Work and
|
||||||
|
reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty.
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, Licensor provides the
|
||||||
|
Work (and each Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
|
||||||
|
including, without limitation, any warranties or conditions of TITLE,
|
||||||
|
NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are
|
||||||
|
solely responsible for determining the appropriateness of using or
|
||||||
|
redistributing the Work and assume any risks associated with Your exercise of
|
||||||
|
permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability.
|
||||||
|
|
||||||
|
In no event and under no legal theory, whether in tort (including negligence),
|
||||||
|
contract, or otherwise, unless required by applicable law (such as deliberate
|
||||||
|
and grossly negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special, incidental,
|
||||||
|
or consequential damages of any character arising as a result of this License or
|
||||||
|
out of the use or inability to use the Work (including but not limited to
|
||||||
|
damages for loss of goodwill, work stoppage, computer failure or malfunction, or
|
||||||
|
any and all other commercial damages or losses), even if such Contributor has
|
||||||
|
been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability.
|
||||||
|
|
||||||
|
While redistributing the Work or Derivative Works thereof, You may choose to
|
||||||
|
offer, and charge a fee for, acceptance of support, warranty, indemnity, or
|
||||||
|
other liability obligations and/or rights consistent with this License. However,
|
||||||
|
in accepting such obligations, You may act only on Your own behalf and on Your
|
||||||
|
sole responsibility, not on behalf of any other Contributor, and only if You
|
||||||
|
agree to indemnify, defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason of your
|
||||||
|
accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following boilerplate
|
||||||
|
notice, with the fields enclosed by brackets "[]" replaced with your own
|
||||||
|
identifying information. (Don't include the brackets!) The text should be
|
||||||
|
enclosed in the appropriate comment syntax for the file format. We also
|
||||||
|
recommend that a file or class name and description of purpose be included on
|
||||||
|
the same "printed page" as the copyright notice for easier identification within
|
||||||
|
third-party archives.
|
||||||
|
|
||||||
|
Copyright [yyyy] [name of copyright owner]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
18
node_modules/@img/sharp-darwin-arm64/README.md
generated
vendored
Normal file
18
node_modules/@img/sharp-darwin-arm64/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,18 @@
|
||||||
|
# `@img/sharp-darwin-arm64`
|
||||||
|
|
||||||
|
Prebuilt sharp for use with macOS 64-bit ARM.
|
||||||
|
|
||||||
|
## Licensing
|
||||||
|
|
||||||
|
Copyright 2013 Lovell Fuller and others.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
[https://www.apache.org/licenses/LICENSE-2.0](https://www.apache.org/licenses/LICENSE-2.0)
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
BIN
node_modules/@img/sharp-darwin-arm64/lib/sharp-darwin-arm64.node
generated
vendored
Normal file
BIN
node_modules/@img/sharp-darwin-arm64/lib/sharp-darwin-arm64.node
generated
vendored
Normal file
Binary file not shown.
40
node_modules/@img/sharp-darwin-arm64/package.json
generated
vendored
Normal file
40
node_modules/@img/sharp-darwin-arm64/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,40 @@
|
||||||
|
{
|
||||||
|
"name": "@img/sharp-darwin-arm64",
|
||||||
|
"version": "0.34.5",
|
||||||
|
"description": "Prebuilt sharp for use with macOS 64-bit ARM",
|
||||||
|
"author": "Lovell Fuller <npm@lovell.info>",
|
||||||
|
"homepage": "https://sharp.pixelplumbing.com",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "git+https://github.com/lovell/sharp.git",
|
||||||
|
"directory": "npm/darwin-arm64"
|
||||||
|
},
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://opencollective.com/libvips"
|
||||||
|
},
|
||||||
|
"preferUnplugged": true,
|
||||||
|
"optionalDependencies": {
|
||||||
|
"@img/sharp-libvips-darwin-arm64": "1.2.4"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"publishConfig": {
|
||||||
|
"access": "public"
|
||||||
|
},
|
||||||
|
"type": "commonjs",
|
||||||
|
"exports": {
|
||||||
|
"./sharp.node": "./lib/sharp-darwin-arm64.node",
|
||||||
|
"./package": "./package.json"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": "^18.17.0 || ^20.3.0 || >=21.0.0"
|
||||||
|
},
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
]
|
||||||
|
}
|
||||||
46
node_modules/@img/sharp-libvips-darwin-arm64/README.md
generated
vendored
Normal file
46
node_modules/@img/sharp-libvips-darwin-arm64/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,46 @@
|
||||||
|
# `@img/sharp-libvips-darwin-arm64`
|
||||||
|
|
||||||
|
Prebuilt libvips and dependencies for use with sharp on macOS 64-bit ARM.
|
||||||
|
|
||||||
|
## Licensing
|
||||||
|
|
||||||
|
This software contains third-party libraries
|
||||||
|
used under the terms of the following licences:
|
||||||
|
|
||||||
|
| Library | Used under the terms of |
|
||||||
|
|---------------|-----------------------------------------------------------------------------------------------------------|
|
||||||
|
| aom | BSD 2-Clause + [Alliance for Open Media Patent License 1.0](https://aomedia.org/license/patent-license/) |
|
||||||
|
| cairo | Mozilla Public License 2.0 |
|
||||||
|
| cgif | MIT Licence |
|
||||||
|
| expat | MIT Licence |
|
||||||
|
| fontconfig | [fontconfig Licence](https://gitlab.freedesktop.org/fontconfig/fontconfig/blob/main/COPYING) (BSD-like) |
|
||||||
|
| freetype | [freetype Licence](https://git.savannah.gnu.org/cgit/freetype/freetype2.git/tree/docs/FTL.TXT) (BSD-like) |
|
||||||
|
| fribidi | LGPLv3 |
|
||||||
|
| glib | LGPLv3 |
|
||||||
|
| harfbuzz | MIT Licence |
|
||||||
|
| highway | Apache-2.0 License, BSD 3-Clause |
|
||||||
|
| lcms | MIT Licence |
|
||||||
|
| libarchive | BSD 2-Clause |
|
||||||
|
| libexif | LGPLv3 |
|
||||||
|
| libffi | MIT Licence |
|
||||||
|
| libheif | LGPLv3 |
|
||||||
|
| libimagequant | [BSD 2-Clause](https://github.com/lovell/libimagequant/blob/main/COPYRIGHT) |
|
||||||
|
| libnsgif | MIT Licence |
|
||||||
|
| libpng | [libpng License](https://github.com/pnggroup/libpng/blob/master/LICENSE) |
|
||||||
|
| librsvg | LGPLv3 |
|
||||||
|
| libspng | [BSD 2-Clause, libpng License](https://github.com/randy408/libspng/blob/master/LICENSE) |
|
||||||
|
| libtiff | [libtiff License](https://gitlab.com/libtiff/libtiff/blob/master/LICENSE.md) (BSD-like) |
|
||||||
|
| libvips | LGPLv3 |
|
||||||
|
| libwebp | New BSD License |
|
||||||
|
| libxml2 | MIT Licence |
|
||||||
|
| mozjpeg | [zlib License, IJG License, BSD-3-Clause](https://github.com/mozilla/mozjpeg/blob/master/LICENSE.md) |
|
||||||
|
| pango | LGPLv3 |
|
||||||
|
| pixman | MIT Licence |
|
||||||
|
| proxy-libintl | LGPLv3 |
|
||||||
|
| zlib-ng | [zlib Licence](https://github.com/zlib-ng/zlib-ng/blob/develop/LICENSE.md) |
|
||||||
|
|
||||||
|
Use of libraries under the terms of the LGPLv3 is via the
|
||||||
|
"any later version" clause of the LGPLv2 or LGPLv2.1.
|
||||||
|
|
||||||
|
Please report any errors or omissions via
|
||||||
|
https://github.com/lovell/sharp-libvips/issues/new
|
||||||
220
node_modules/@img/sharp-libvips-darwin-arm64/lib/glib-2.0/include/glibconfig.h
generated
vendored
Normal file
220
node_modules/@img/sharp-libvips-darwin-arm64/lib/glib-2.0/include/glibconfig.h
generated
vendored
Normal file
|
|
@ -0,0 +1,220 @@
|
||||||
|
/* glibconfig.h
|
||||||
|
*
|
||||||
|
* This is a generated file. Please modify 'glibconfig.h.in'
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __GLIBCONFIG_H__
|
||||||
|
#define __GLIBCONFIG_H__
|
||||||
|
|
||||||
|
#include <glib/gmacros.h>
|
||||||
|
|
||||||
|
#include <limits.h>
|
||||||
|
#include <float.h>
|
||||||
|
#define GLIB_HAVE_ALLOCA_H
|
||||||
|
|
||||||
|
#define GLIB_STATIC_COMPILATION 1
|
||||||
|
#define GOBJECT_STATIC_COMPILATION 1
|
||||||
|
#define GIO_STATIC_COMPILATION 1
|
||||||
|
#define GMODULE_STATIC_COMPILATION 1
|
||||||
|
#define GI_STATIC_COMPILATION 1
|
||||||
|
#define G_INTL_STATIC_COMPILATION 1
|
||||||
|
#define FFI_STATIC_BUILD 1
|
||||||
|
|
||||||
|
/* Specifies that GLib's g_print*() functions wrap the
|
||||||
|
* system printf functions. This is useful to know, for example,
|
||||||
|
* when using glibc's register_printf_function().
|
||||||
|
*/
|
||||||
|
#define GLIB_USING_SYSTEM_PRINTF
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#define G_MINFLOAT FLT_MIN
|
||||||
|
#define G_MAXFLOAT FLT_MAX
|
||||||
|
#define G_MINDOUBLE DBL_MIN
|
||||||
|
#define G_MAXDOUBLE DBL_MAX
|
||||||
|
#define G_MINSHORT SHRT_MIN
|
||||||
|
#define G_MAXSHORT SHRT_MAX
|
||||||
|
#define G_MAXUSHORT USHRT_MAX
|
||||||
|
#define G_MININT INT_MIN
|
||||||
|
#define G_MAXINT INT_MAX
|
||||||
|
#define G_MAXUINT UINT_MAX
|
||||||
|
#define G_MINLONG LONG_MIN
|
||||||
|
#define G_MAXLONG LONG_MAX
|
||||||
|
#define G_MAXULONG ULONG_MAX
|
||||||
|
|
||||||
|
typedef signed char gint8;
|
||||||
|
typedef unsigned char guint8;
|
||||||
|
|
||||||
|
typedef signed short gint16;
|
||||||
|
typedef unsigned short guint16;
|
||||||
|
|
||||||
|
#define G_GINT16_MODIFIER "h"
|
||||||
|
#define G_GINT16_FORMAT "hi"
|
||||||
|
#define G_GUINT16_FORMAT "hu"
|
||||||
|
|
||||||
|
|
||||||
|
typedef signed int gint32;
|
||||||
|
typedef unsigned int guint32;
|
||||||
|
|
||||||
|
#define G_GINT32_MODIFIER ""
|
||||||
|
#define G_GINT32_FORMAT "i"
|
||||||
|
#define G_GUINT32_FORMAT "u"
|
||||||
|
|
||||||
|
|
||||||
|
#define G_HAVE_GINT64 1 /* deprecated, always true */
|
||||||
|
|
||||||
|
G_GNUC_EXTENSION typedef signed long long gint64;
|
||||||
|
G_GNUC_EXTENSION typedef unsigned long long guint64;
|
||||||
|
|
||||||
|
#define G_GINT64_CONSTANT(val) (G_GNUC_EXTENSION (val##LL))
|
||||||
|
#define G_GUINT64_CONSTANT(val) (G_GNUC_EXTENSION (val##ULL))
|
||||||
|
|
||||||
|
#define G_GINT64_MODIFIER "ll"
|
||||||
|
#define G_GINT64_FORMAT "lli"
|
||||||
|
#define G_GUINT64_FORMAT "llu"
|
||||||
|
|
||||||
|
|
||||||
|
#define GLIB_SIZEOF_VOID_P 8
|
||||||
|
#define GLIB_SIZEOF_LONG 8
|
||||||
|
#define GLIB_SIZEOF_SIZE_T 8
|
||||||
|
#define GLIB_SIZEOF_SSIZE_T 8
|
||||||
|
|
||||||
|
typedef signed long gssize;
|
||||||
|
typedef unsigned long gsize;
|
||||||
|
#define G_GSIZE_MODIFIER "l"
|
||||||
|
#define G_GSSIZE_MODIFIER "l"
|
||||||
|
#define G_GSIZE_FORMAT "lu"
|
||||||
|
#define G_GSSIZE_FORMAT "li"
|
||||||
|
|
||||||
|
#define G_MAXSIZE G_MAXULONG
|
||||||
|
#define G_MINSSIZE G_MINLONG
|
||||||
|
#define G_MAXSSIZE G_MAXLONG
|
||||||
|
|
||||||
|
typedef gint64 goffset;
|
||||||
|
#define G_MINOFFSET G_MININT64
|
||||||
|
#define G_MAXOFFSET G_MAXINT64
|
||||||
|
|
||||||
|
#define G_GOFFSET_MODIFIER G_GINT64_MODIFIER
|
||||||
|
#define G_GOFFSET_FORMAT G_GINT64_FORMAT
|
||||||
|
#define G_GOFFSET_CONSTANT(val) G_GINT64_CONSTANT(val)
|
||||||
|
|
||||||
|
#define G_POLLFD_FORMAT "%d"
|
||||||
|
|
||||||
|
#define GPOINTER_TO_INT(p) ((gint) (glong) (p))
|
||||||
|
#define GPOINTER_TO_UINT(p) ((guint) (gulong) (p))
|
||||||
|
|
||||||
|
#define GINT_TO_POINTER(i) ((gpointer) (glong) (i))
|
||||||
|
#define GUINT_TO_POINTER(u) ((gpointer) (gulong) (u))
|
||||||
|
|
||||||
|
typedef signed long gintptr;
|
||||||
|
typedef unsigned long guintptr;
|
||||||
|
|
||||||
|
#define G_GINTPTR_MODIFIER "l"
|
||||||
|
#define G_GINTPTR_FORMAT "li"
|
||||||
|
#define G_GUINTPTR_FORMAT "lu"
|
||||||
|
|
||||||
|
#define GLIB_MAJOR_VERSION 2
|
||||||
|
#define GLIB_MINOR_VERSION 86
|
||||||
|
#define GLIB_MICRO_VERSION 1
|
||||||
|
|
||||||
|
#define G_OS_UNIX
|
||||||
|
|
||||||
|
#define G_VA_COPY va_copy
|
||||||
|
|
||||||
|
|
||||||
|
#define G_HAVE_ISO_VARARGS 1
|
||||||
|
|
||||||
|
/* gcc-2.95.x supports both gnu style and ISO varargs, but if -ansi
|
||||||
|
* is passed ISO vararg support is turned off, and there is no work
|
||||||
|
* around to turn it on, so we unconditionally turn it off.
|
||||||
|
*/
|
||||||
|
#if __GNUC__ == 2 && __GNUC_MINOR__ == 95
|
||||||
|
# undef G_HAVE_ISO_VARARGS
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define G_HAVE_GROWING_STACK 0
|
||||||
|
|
||||||
|
#ifndef _MSC_VER
|
||||||
|
# define G_HAVE_GNUC_VARARGS 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__SUNPRO_C) && (__SUNPRO_C >= 0x590)
|
||||||
|
#define G_GNUC_INTERNAL __attribute__((visibility("hidden")))
|
||||||
|
#elif defined(__SUNPRO_C) && (__SUNPRO_C >= 0x550)
|
||||||
|
#define G_GNUC_INTERNAL __hidden
|
||||||
|
#elif defined (__GNUC__) && defined (G_HAVE_GNUC_VISIBILITY)
|
||||||
|
#define G_GNUC_INTERNAL __attribute__((visibility("hidden")))
|
||||||
|
#else
|
||||||
|
#define G_GNUC_INTERNAL
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define G_THREADS_ENABLED
|
||||||
|
#define G_THREADS_IMPL_POSIX
|
||||||
|
|
||||||
|
#define G_ATOMIC_LOCK_FREE
|
||||||
|
|
||||||
|
#define GINT16_TO_LE(val) ((gint16) (val))
|
||||||
|
#define GUINT16_TO_LE(val) ((guint16) (val))
|
||||||
|
#define GINT16_TO_BE(val) ((gint16) GUINT16_SWAP_LE_BE (val))
|
||||||
|
#define GUINT16_TO_BE(val) (GUINT16_SWAP_LE_BE (val))
|
||||||
|
|
||||||
|
#define GINT32_TO_LE(val) ((gint32) (val))
|
||||||
|
#define GUINT32_TO_LE(val) ((guint32) (val))
|
||||||
|
#define GINT32_TO_BE(val) ((gint32) GUINT32_SWAP_LE_BE (val))
|
||||||
|
#define GUINT32_TO_BE(val) (GUINT32_SWAP_LE_BE (val))
|
||||||
|
|
||||||
|
#define GINT64_TO_LE(val) ((gint64) (val))
|
||||||
|
#define GUINT64_TO_LE(val) ((guint64) (val))
|
||||||
|
#define GINT64_TO_BE(val) ((gint64) GUINT64_SWAP_LE_BE (val))
|
||||||
|
#define GUINT64_TO_BE(val) (GUINT64_SWAP_LE_BE (val))
|
||||||
|
|
||||||
|
#define GLONG_TO_LE(val) ((glong) GINT64_TO_LE (val))
|
||||||
|
#define GULONG_TO_LE(val) ((gulong) GUINT64_TO_LE (val))
|
||||||
|
#define GLONG_TO_BE(val) ((glong) GINT64_TO_BE (val))
|
||||||
|
#define GULONG_TO_BE(val) ((gulong) GUINT64_TO_BE (val))
|
||||||
|
#define GINT_TO_LE(val) ((gint) GINT32_TO_LE (val))
|
||||||
|
#define GUINT_TO_LE(val) ((guint) GUINT32_TO_LE (val))
|
||||||
|
#define GINT_TO_BE(val) ((gint) GINT32_TO_BE (val))
|
||||||
|
#define GUINT_TO_BE(val) ((guint) GUINT32_TO_BE (val))
|
||||||
|
#define GSIZE_TO_LE(val) ((gsize) GUINT64_TO_LE (val))
|
||||||
|
#define GSSIZE_TO_LE(val) ((gssize) GINT64_TO_LE (val))
|
||||||
|
#define GSIZE_TO_BE(val) ((gsize) GUINT64_TO_BE (val))
|
||||||
|
#define GSSIZE_TO_BE(val) ((gssize) GINT64_TO_BE (val))
|
||||||
|
#define G_BYTE_ORDER G_LITTLE_ENDIAN
|
||||||
|
|
||||||
|
#define GLIB_SYSDEF_POLLIN =1
|
||||||
|
#define GLIB_SYSDEF_POLLOUT =4
|
||||||
|
#define GLIB_SYSDEF_POLLPRI =2
|
||||||
|
#define GLIB_SYSDEF_POLLHUP =16
|
||||||
|
#define GLIB_SYSDEF_POLLERR =8
|
||||||
|
#define GLIB_SYSDEF_POLLNVAL =32
|
||||||
|
|
||||||
|
/* No way to disable deprecation warnings for macros, so only emit deprecation
|
||||||
|
* warnings on platforms where usage of this macro is broken */
|
||||||
|
#if defined(__APPLE__) || defined(_MSC_VER) || defined(__CYGWIN__)
|
||||||
|
#define G_MODULE_SUFFIX "so" GLIB_DEPRECATED_MACRO_IN_2_76
|
||||||
|
#else
|
||||||
|
#define G_MODULE_SUFFIX "so"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef int GPid;
|
||||||
|
#define G_PID_FORMAT "i"
|
||||||
|
|
||||||
|
#define GLIB_SYSDEF_AF_UNIX 1
|
||||||
|
#define GLIB_SYSDEF_AF_INET 2
|
||||||
|
#define GLIB_SYSDEF_AF_INET6 30
|
||||||
|
|
||||||
|
#define GLIB_SYSDEF_MSG_OOB 1
|
||||||
|
#define GLIB_SYSDEF_MSG_PEEK 2
|
||||||
|
#define GLIB_SYSDEF_MSG_DONTROUTE 4
|
||||||
|
|
||||||
|
#define G_DIR_SEPARATOR '/'
|
||||||
|
#define G_DIR_SEPARATOR_S "/"
|
||||||
|
#define G_SEARCHPATH_SEPARATOR ':'
|
||||||
|
#define G_SEARCHPATH_SEPARATOR_S ":"
|
||||||
|
|
||||||
|
#undef G_HAVE_FREE_SIZED
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* __GLIBCONFIG_H__ */
|
||||||
1
node_modules/@img/sharp-libvips-darwin-arm64/lib/index.js
generated
vendored
Normal file
1
node_modules/@img/sharp-libvips-darwin-arm64/lib/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
module.exports = __dirname;
|
||||||
BIN
node_modules/@img/sharp-libvips-darwin-arm64/lib/libvips-cpp.8.17.3.dylib
generated
vendored
Normal file
BIN
node_modules/@img/sharp-libvips-darwin-arm64/lib/libvips-cpp.8.17.3.dylib
generated
vendored
Normal file
Binary file not shown.
36
node_modules/@img/sharp-libvips-darwin-arm64/package.json
generated
vendored
Normal file
36
node_modules/@img/sharp-libvips-darwin-arm64/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,36 @@
|
||||||
|
{
|
||||||
|
"name": "@img/sharp-libvips-darwin-arm64",
|
||||||
|
"version": "1.2.4",
|
||||||
|
"description": "Prebuilt libvips and dependencies for use with sharp on macOS 64-bit ARM",
|
||||||
|
"author": "Lovell Fuller <npm@lovell.info>",
|
||||||
|
"homepage": "https://sharp.pixelplumbing.com",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "git+https://github.com/lovell/sharp-libvips.git",
|
||||||
|
"directory": "npm/darwin-arm64"
|
||||||
|
},
|
||||||
|
"license": "LGPL-3.0-or-later",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://opencollective.com/libvips"
|
||||||
|
},
|
||||||
|
"preferUnplugged": true,
|
||||||
|
"publishConfig": {
|
||||||
|
"access": "public"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"lib",
|
||||||
|
"versions.json"
|
||||||
|
],
|
||||||
|
"type": "commonjs",
|
||||||
|
"exports": {
|
||||||
|
"./lib": "./lib/index.js",
|
||||||
|
"./package": "./package.json",
|
||||||
|
"./versions": "./versions.json"
|
||||||
|
},
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
]
|
||||||
|
}
|
||||||
30
node_modules/@img/sharp-libvips-darwin-arm64/versions.json
generated
vendored
Normal file
30
node_modules/@img/sharp-libvips-darwin-arm64/versions.json
generated
vendored
Normal file
|
|
@ -0,0 +1,30 @@
|
||||||
|
{
|
||||||
|
"aom": "3.13.1",
|
||||||
|
"archive": "3.8.2",
|
||||||
|
"cairo": "1.18.4",
|
||||||
|
"cgif": "0.5.0",
|
||||||
|
"exif": "0.6.25",
|
||||||
|
"expat": "2.7.3",
|
||||||
|
"ffi": "3.5.2",
|
||||||
|
"fontconfig": "2.17.1",
|
||||||
|
"freetype": "2.14.1",
|
||||||
|
"fribidi": "1.0.16",
|
||||||
|
"glib": "2.86.1",
|
||||||
|
"harfbuzz": "12.1.0",
|
||||||
|
"heif": "1.20.2",
|
||||||
|
"highway": "1.3.0",
|
||||||
|
"imagequant": "2.4.1",
|
||||||
|
"lcms": "2.17",
|
||||||
|
"mozjpeg": "0826579",
|
||||||
|
"pango": "1.57.0",
|
||||||
|
"pixman": "0.46.4",
|
||||||
|
"png": "1.6.50",
|
||||||
|
"proxy-libintl": "0.5",
|
||||||
|
"rsvg": "2.61.2",
|
||||||
|
"spng": "0.7.4",
|
||||||
|
"tiff": "4.7.1",
|
||||||
|
"vips": "8.17.3",
|
||||||
|
"webp": "1.6.0",
|
||||||
|
"xml2": "2.15.1",
|
||||||
|
"zlib-ng": "2.2.5"
|
||||||
|
}
|
||||||
250
node_modules/accepts/HISTORY.md
generated
vendored
Normal file
250
node_modules/accepts/HISTORY.md
generated
vendored
Normal file
|
|
@ -0,0 +1,250 @@
|
||||||
|
2.0.0 / 2024-08-31
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Drop node <18 support
|
||||||
|
* deps: mime-types@^3.0.0
|
||||||
|
* deps: negotiator@^1.0.0
|
||||||
|
|
||||||
|
1.3.8 / 2022-02-02
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.34
|
||||||
|
- deps: mime-db@~1.51.0
|
||||||
|
* deps: negotiator@0.6.3
|
||||||
|
|
||||||
|
1.3.7 / 2019-04-29
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: negotiator@0.6.2
|
||||||
|
- Fix sorting charset, encoding, and language with extra parameters
|
||||||
|
|
||||||
|
1.3.6 / 2019-04-28
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.24
|
||||||
|
- deps: mime-db@~1.40.0
|
||||||
|
|
||||||
|
1.3.5 / 2018-02-28
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.18
|
||||||
|
- deps: mime-db@~1.33.0
|
||||||
|
|
||||||
|
1.3.4 / 2017-08-22
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.16
|
||||||
|
- deps: mime-db@~1.29.0
|
||||||
|
|
||||||
|
1.3.3 / 2016-05-02
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.11
|
||||||
|
- deps: mime-db@~1.23.0
|
||||||
|
* deps: negotiator@0.6.1
|
||||||
|
- perf: improve `Accept` parsing speed
|
||||||
|
- perf: improve `Accept-Charset` parsing speed
|
||||||
|
- perf: improve `Accept-Encoding` parsing speed
|
||||||
|
- perf: improve `Accept-Language` parsing speed
|
||||||
|
|
||||||
|
1.3.2 / 2016-03-08
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.10
|
||||||
|
- Fix extension of `application/dash+xml`
|
||||||
|
- Update primary extension for `audio/mp4`
|
||||||
|
- deps: mime-db@~1.22.0
|
||||||
|
|
||||||
|
1.3.1 / 2016-01-19
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.9
|
||||||
|
- deps: mime-db@~1.21.0
|
||||||
|
|
||||||
|
1.3.0 / 2015-09-29
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.7
|
||||||
|
- deps: mime-db@~1.19.0
|
||||||
|
* deps: negotiator@0.6.0
|
||||||
|
- Fix including type extensions in parameters in `Accept` parsing
|
||||||
|
- Fix parsing `Accept` parameters with quoted equals
|
||||||
|
- Fix parsing `Accept` parameters with quoted semicolons
|
||||||
|
- Lazy-load modules from main entry point
|
||||||
|
- perf: delay type concatenation until needed
|
||||||
|
- perf: enable strict mode
|
||||||
|
- perf: hoist regular expressions
|
||||||
|
- perf: remove closures getting spec properties
|
||||||
|
- perf: remove a closure from media type parsing
|
||||||
|
- perf: remove property delete from media type parsing
|
||||||
|
|
||||||
|
1.2.13 / 2015-09-06
|
||||||
|
===================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.6
|
||||||
|
- deps: mime-db@~1.18.0
|
||||||
|
|
||||||
|
1.2.12 / 2015-07-30
|
||||||
|
===================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.4
|
||||||
|
- deps: mime-db@~1.16.0
|
||||||
|
|
||||||
|
1.2.11 / 2015-07-16
|
||||||
|
===================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.3
|
||||||
|
- deps: mime-db@~1.15.0
|
||||||
|
|
||||||
|
1.2.10 / 2015-07-01
|
||||||
|
===================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.2
|
||||||
|
- deps: mime-db@~1.14.0
|
||||||
|
|
||||||
|
1.2.9 / 2015-06-08
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.1
|
||||||
|
- perf: fix deopt during mapping
|
||||||
|
|
||||||
|
1.2.8 / 2015-06-07
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.1.0
|
||||||
|
- deps: mime-db@~1.13.0
|
||||||
|
* perf: avoid argument reassignment & argument slice
|
||||||
|
* perf: avoid negotiator recursive construction
|
||||||
|
* perf: enable strict mode
|
||||||
|
* perf: remove unnecessary bitwise operator
|
||||||
|
|
||||||
|
1.2.7 / 2015-05-10
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: negotiator@0.5.3
|
||||||
|
- Fix media type parameter matching to be case-insensitive
|
||||||
|
|
||||||
|
1.2.6 / 2015-05-07
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.11
|
||||||
|
- deps: mime-db@~1.9.1
|
||||||
|
* deps: negotiator@0.5.2
|
||||||
|
- Fix comparing media types with quoted values
|
||||||
|
- Fix splitting media types with quoted commas
|
||||||
|
|
||||||
|
1.2.5 / 2015-03-13
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.10
|
||||||
|
- deps: mime-db@~1.8.0
|
||||||
|
|
||||||
|
1.2.4 / 2015-02-14
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Support Node.js 0.6
|
||||||
|
* deps: mime-types@~2.0.9
|
||||||
|
- deps: mime-db@~1.7.0
|
||||||
|
* deps: negotiator@0.5.1
|
||||||
|
- Fix preference sorting to be stable for long acceptable lists
|
||||||
|
|
||||||
|
1.2.3 / 2015-01-31
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.8
|
||||||
|
- deps: mime-db@~1.6.0
|
||||||
|
|
||||||
|
1.2.2 / 2014-12-30
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.7
|
||||||
|
- deps: mime-db@~1.5.0
|
||||||
|
|
||||||
|
1.2.1 / 2014-12-30
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.5
|
||||||
|
- deps: mime-db@~1.3.1
|
||||||
|
|
||||||
|
1.2.0 / 2014-12-19
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: negotiator@0.5.0
|
||||||
|
- Fix list return order when large accepted list
|
||||||
|
- Fix missing identity encoding when q=0 exists
|
||||||
|
- Remove dynamic building of Negotiator class
|
||||||
|
|
||||||
|
1.1.4 / 2014-12-10
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.4
|
||||||
|
- deps: mime-db@~1.3.0
|
||||||
|
|
||||||
|
1.1.3 / 2014-11-09
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.3
|
||||||
|
- deps: mime-db@~1.2.0
|
||||||
|
|
||||||
|
1.1.2 / 2014-10-14
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: negotiator@0.4.9
|
||||||
|
- Fix error when media type has invalid parameter
|
||||||
|
|
||||||
|
1.1.1 / 2014-09-28
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: mime-types@~2.0.2
|
||||||
|
- deps: mime-db@~1.1.0
|
||||||
|
* deps: negotiator@0.4.8
|
||||||
|
- Fix all negotiations to be case-insensitive
|
||||||
|
- Stable sort preferences of same quality according to client order
|
||||||
|
|
||||||
|
1.1.0 / 2014-09-02
|
||||||
|
==================
|
||||||
|
|
||||||
|
* update `mime-types`
|
||||||
|
|
||||||
|
1.0.7 / 2014-07-04
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Fix wrong type returned from `type` when match after unknown extension
|
||||||
|
|
||||||
|
1.0.6 / 2014-06-24
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: negotiator@0.4.7
|
||||||
|
|
||||||
|
1.0.5 / 2014-06-20
|
||||||
|
==================
|
||||||
|
|
||||||
|
* fix crash when unknown extension given
|
||||||
|
|
||||||
|
1.0.4 / 2014-06-19
|
||||||
|
==================
|
||||||
|
|
||||||
|
* use `mime-types`
|
||||||
|
|
||||||
|
1.0.3 / 2014-06-11
|
||||||
|
==================
|
||||||
|
|
||||||
|
* deps: negotiator@0.4.6
|
||||||
|
- Order by specificity when quality is the same
|
||||||
|
|
||||||
|
1.0.2 / 2014-05-29
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Fix interpretation when header not in request
|
||||||
|
* deps: pin negotiator@0.4.5
|
||||||
|
|
||||||
|
1.0.1 / 2014-01-18
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Identity encoding isn't always acceptable
|
||||||
|
* deps: negotiator@~0.4.0
|
||||||
|
|
||||||
|
1.0.0 / 2013-12-27
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Genesis
|
||||||
23
node_modules/accepts/LICENSE
generated
vendored
Normal file
23
node_modules/accepts/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
(The MIT License)
|
||||||
|
|
||||||
|
Copyright (c) 2014 Jonathan Ong <me@jongleberry.com>
|
||||||
|
Copyright (c) 2015 Douglas Christopher Wilson <doug@somethingdoug.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
'Software'), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
||||||
|
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
||||||
|
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
||||||
|
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
140
node_modules/accepts/README.md
generated
vendored
Normal file
140
node_modules/accepts/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,140 @@
|
||||||
|
# accepts
|
||||||
|
|
||||||
|
[![NPM Version][npm-version-image]][npm-url]
|
||||||
|
[![NPM Downloads][npm-downloads-image]][npm-url]
|
||||||
|
[![Node.js Version][node-version-image]][node-version-url]
|
||||||
|
[![Build Status][github-actions-ci-image]][github-actions-ci-url]
|
||||||
|
[![Test Coverage][coveralls-image]][coveralls-url]
|
||||||
|
|
||||||
|
Higher level content negotiation based on [negotiator](https://www.npmjs.com/package/negotiator).
|
||||||
|
Extracted from [koa](https://www.npmjs.com/package/koa) for general use.
|
||||||
|
|
||||||
|
In addition to negotiator, it allows:
|
||||||
|
|
||||||
|
- Allows types as an array or arguments list, ie `(['text/html', 'application/json'])`
|
||||||
|
as well as `('text/html', 'application/json')`.
|
||||||
|
- Allows type shorthands such as `json`.
|
||||||
|
- Returns `false` when no types match
|
||||||
|
- Treats non-existent headers as `*`
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
This is a [Node.js](https://nodejs.org/en/) module available through the
|
||||||
|
[npm registry](https://www.npmjs.com/). Installation is done using the
|
||||||
|
[`npm install` command](https://docs.npmjs.com/getting-started/installing-npm-packages-locally):
|
||||||
|
|
||||||
|
```sh
|
||||||
|
$ npm install accepts
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
```js
|
||||||
|
var accepts = require('accepts')
|
||||||
|
```
|
||||||
|
|
||||||
|
### accepts(req)
|
||||||
|
|
||||||
|
Create a new `Accepts` object for the given `req`.
|
||||||
|
|
||||||
|
#### .charset(charsets)
|
||||||
|
|
||||||
|
Return the first accepted charset. If nothing in `charsets` is accepted,
|
||||||
|
then `false` is returned.
|
||||||
|
|
||||||
|
#### .charsets()
|
||||||
|
|
||||||
|
Return the charsets that the request accepts, in the order of the client's
|
||||||
|
preference (most preferred first).
|
||||||
|
|
||||||
|
#### .encoding(encodings)
|
||||||
|
|
||||||
|
Return the first accepted encoding. If nothing in `encodings` is accepted,
|
||||||
|
then `false` is returned.
|
||||||
|
|
||||||
|
#### .encodings()
|
||||||
|
|
||||||
|
Return the encodings that the request accepts, in the order of the client's
|
||||||
|
preference (most preferred first).
|
||||||
|
|
||||||
|
#### .language(languages)
|
||||||
|
|
||||||
|
Return the first accepted language. If nothing in `languages` is accepted,
|
||||||
|
then `false` is returned.
|
||||||
|
|
||||||
|
#### .languages()
|
||||||
|
|
||||||
|
Return the languages that the request accepts, in the order of the client's
|
||||||
|
preference (most preferred first).
|
||||||
|
|
||||||
|
#### .type(types)
|
||||||
|
|
||||||
|
Return the first accepted type (and it is returned as the same text as what
|
||||||
|
appears in the `types` array). If nothing in `types` is accepted, then `false`
|
||||||
|
is returned.
|
||||||
|
|
||||||
|
The `types` array can contain full MIME types or file extensions. Any value
|
||||||
|
that is not a full MIME type is passed to `require('mime-types').lookup`.
|
||||||
|
|
||||||
|
#### .types()
|
||||||
|
|
||||||
|
Return the types that the request accepts, in the order of the client's
|
||||||
|
preference (most preferred first).
|
||||||
|
|
||||||
|
## Examples
|
||||||
|
|
||||||
|
### Simple type negotiation
|
||||||
|
|
||||||
|
This simple example shows how to use `accepts` to return a different typed
|
||||||
|
respond body based on what the client wants to accept. The server lists it's
|
||||||
|
preferences in order and will get back the best match between the client and
|
||||||
|
server.
|
||||||
|
|
||||||
|
```js
|
||||||
|
var accepts = require('accepts')
|
||||||
|
var http = require('http')
|
||||||
|
|
||||||
|
function app (req, res) {
|
||||||
|
var accept = accepts(req)
|
||||||
|
|
||||||
|
// the order of this list is significant; should be server preferred order
|
||||||
|
switch (accept.type(['json', 'html'])) {
|
||||||
|
case 'json':
|
||||||
|
res.setHeader('Content-Type', 'application/json')
|
||||||
|
res.write('{"hello":"world!"}')
|
||||||
|
break
|
||||||
|
case 'html':
|
||||||
|
res.setHeader('Content-Type', 'text/html')
|
||||||
|
res.write('<b>hello, world!</b>')
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
// the fallback is text/plain, so no need to specify it above
|
||||||
|
res.setHeader('Content-Type', 'text/plain')
|
||||||
|
res.write('hello, world!')
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
res.end()
|
||||||
|
}
|
||||||
|
|
||||||
|
http.createServer(app).listen(3000)
|
||||||
|
```
|
||||||
|
|
||||||
|
You can test this out with the cURL program:
|
||||||
|
```sh
|
||||||
|
curl -I -H'Accept: text/html' http://localhost:3000/
|
||||||
|
```
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
[MIT](LICENSE)
|
||||||
|
|
||||||
|
[coveralls-image]: https://badgen.net/coveralls/c/github/jshttp/accepts/master
|
||||||
|
[coveralls-url]: https://coveralls.io/r/jshttp/accepts?branch=master
|
||||||
|
[github-actions-ci-image]: https://badgen.net/github/checks/jshttp/accepts/master?label=ci
|
||||||
|
[github-actions-ci-url]: https://github.com/jshttp/accepts/actions/workflows/ci.yml
|
||||||
|
[node-version-image]: https://badgen.net/npm/node/accepts
|
||||||
|
[node-version-url]: https://nodejs.org/en/download
|
||||||
|
[npm-downloads-image]: https://badgen.net/npm/dm/accepts
|
||||||
|
[npm-url]: https://npmjs.org/package/accepts
|
||||||
|
[npm-version-image]: https://badgen.net/npm/v/accepts
|
||||||
238
node_modules/accepts/index.js
generated
vendored
Normal file
238
node_modules/accepts/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1,238 @@
|
||||||
|
/*!
|
||||||
|
* accepts
|
||||||
|
* Copyright(c) 2014 Jonathan Ong
|
||||||
|
* Copyright(c) 2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Negotiator = require('negotiator')
|
||||||
|
var mime = require('mime-types')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = Accepts
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new Accepts object for the given req.
|
||||||
|
*
|
||||||
|
* @param {object} req
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
function Accepts (req) {
|
||||||
|
if (!(this instanceof Accepts)) {
|
||||||
|
return new Accepts(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
this.headers = req.headers
|
||||||
|
this.negotiator = new Negotiator(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if the given `type(s)` is acceptable, returning
|
||||||
|
* the best match when true, otherwise `undefined`, in which
|
||||||
|
* case you should respond with 406 "Not Acceptable".
|
||||||
|
*
|
||||||
|
* The `type` value may be a single mime type string
|
||||||
|
* such as "application/json", the extension name
|
||||||
|
* such as "json" or an array `["json", "html", "text/plain"]`. When a list
|
||||||
|
* or array is given the _best_ match, if any is returned.
|
||||||
|
*
|
||||||
|
* Examples:
|
||||||
|
*
|
||||||
|
* // Accept: text/html
|
||||||
|
* this.types('html');
|
||||||
|
* // => "html"
|
||||||
|
*
|
||||||
|
* // Accept: text/*, application/json
|
||||||
|
* this.types('html');
|
||||||
|
* // => "html"
|
||||||
|
* this.types('text/html');
|
||||||
|
* // => "text/html"
|
||||||
|
* this.types('json', 'text');
|
||||||
|
* // => "json"
|
||||||
|
* this.types('application/json');
|
||||||
|
* // => "application/json"
|
||||||
|
*
|
||||||
|
* // Accept: text/*, application/json
|
||||||
|
* this.types('image/png');
|
||||||
|
* this.types('png');
|
||||||
|
* // => undefined
|
||||||
|
*
|
||||||
|
* // Accept: text/*;q=.5, application/json
|
||||||
|
* this.types(['html', 'json']);
|
||||||
|
* this.types('html', 'json');
|
||||||
|
* // => "json"
|
||||||
|
*
|
||||||
|
* @param {String|Array} types...
|
||||||
|
* @return {String|Array|Boolean}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
Accepts.prototype.type =
|
||||||
|
Accepts.prototype.types = function (types_) {
|
||||||
|
var types = types_
|
||||||
|
|
||||||
|
// support flattened arguments
|
||||||
|
if (types && !Array.isArray(types)) {
|
||||||
|
types = new Array(arguments.length)
|
||||||
|
for (var i = 0; i < types.length; i++) {
|
||||||
|
types[i] = arguments[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// no types, return all requested types
|
||||||
|
if (!types || types.length === 0) {
|
||||||
|
return this.negotiator.mediaTypes()
|
||||||
|
}
|
||||||
|
|
||||||
|
// no accept header, return first given type
|
||||||
|
if (!this.headers.accept) {
|
||||||
|
return types[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
var mimes = types.map(extToMime)
|
||||||
|
var accepts = this.negotiator.mediaTypes(mimes.filter(validMime))
|
||||||
|
var first = accepts[0]
|
||||||
|
|
||||||
|
return first
|
||||||
|
? types[mimes.indexOf(first)]
|
||||||
|
: false
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return accepted encodings or best fit based on `encodings`.
|
||||||
|
*
|
||||||
|
* Given `Accept-Encoding: gzip, deflate`
|
||||||
|
* an array sorted by quality is returned:
|
||||||
|
*
|
||||||
|
* ['gzip', 'deflate']
|
||||||
|
*
|
||||||
|
* @param {String|Array} encodings...
|
||||||
|
* @return {String|Array}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
Accepts.prototype.encoding =
|
||||||
|
Accepts.prototype.encodings = function (encodings_) {
|
||||||
|
var encodings = encodings_
|
||||||
|
|
||||||
|
// support flattened arguments
|
||||||
|
if (encodings && !Array.isArray(encodings)) {
|
||||||
|
encodings = new Array(arguments.length)
|
||||||
|
for (var i = 0; i < encodings.length; i++) {
|
||||||
|
encodings[i] = arguments[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// no encodings, return all requested encodings
|
||||||
|
if (!encodings || encodings.length === 0) {
|
||||||
|
return this.negotiator.encodings()
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.negotiator.encodings(encodings)[0] || false
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return accepted charsets or best fit based on `charsets`.
|
||||||
|
*
|
||||||
|
* Given `Accept-Charset: utf-8, iso-8859-1;q=0.2, utf-7;q=0.5`
|
||||||
|
* an array sorted by quality is returned:
|
||||||
|
*
|
||||||
|
* ['utf-8', 'utf-7', 'iso-8859-1']
|
||||||
|
*
|
||||||
|
* @param {String|Array} charsets...
|
||||||
|
* @return {String|Array}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
Accepts.prototype.charset =
|
||||||
|
Accepts.prototype.charsets = function (charsets_) {
|
||||||
|
var charsets = charsets_
|
||||||
|
|
||||||
|
// support flattened arguments
|
||||||
|
if (charsets && !Array.isArray(charsets)) {
|
||||||
|
charsets = new Array(arguments.length)
|
||||||
|
for (var i = 0; i < charsets.length; i++) {
|
||||||
|
charsets[i] = arguments[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// no charsets, return all requested charsets
|
||||||
|
if (!charsets || charsets.length === 0) {
|
||||||
|
return this.negotiator.charsets()
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.negotiator.charsets(charsets)[0] || false
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return accepted languages or best fit based on `langs`.
|
||||||
|
*
|
||||||
|
* Given `Accept-Language: en;q=0.8, es, pt`
|
||||||
|
* an array sorted by quality is returned:
|
||||||
|
*
|
||||||
|
* ['es', 'pt', 'en']
|
||||||
|
*
|
||||||
|
* @param {String|Array} langs...
|
||||||
|
* @return {Array|String}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
Accepts.prototype.lang =
|
||||||
|
Accepts.prototype.langs =
|
||||||
|
Accepts.prototype.language =
|
||||||
|
Accepts.prototype.languages = function (languages_) {
|
||||||
|
var languages = languages_
|
||||||
|
|
||||||
|
// support flattened arguments
|
||||||
|
if (languages && !Array.isArray(languages)) {
|
||||||
|
languages = new Array(arguments.length)
|
||||||
|
for (var i = 0; i < languages.length; i++) {
|
||||||
|
languages[i] = arguments[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// no languages, return all requested languages
|
||||||
|
if (!languages || languages.length === 0) {
|
||||||
|
return this.negotiator.languages()
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.negotiator.languages(languages)[0] || false
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert extnames to mime.
|
||||||
|
*
|
||||||
|
* @param {String} type
|
||||||
|
* @return {String}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
function extToMime (type) {
|
||||||
|
return type.indexOf('/') === -1
|
||||||
|
? mime.lookup(type)
|
||||||
|
: type
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if mime is valid.
|
||||||
|
*
|
||||||
|
* @param {String} type
|
||||||
|
* @return {Boolean}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
function validMime (type) {
|
||||||
|
return typeof type === 'string'
|
||||||
|
}
|
||||||
47
node_modules/accepts/package.json
generated
vendored
Normal file
47
node_modules/accepts/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
{
|
||||||
|
"name": "accepts",
|
||||||
|
"description": "Higher-level content negotiation",
|
||||||
|
"version": "2.0.0",
|
||||||
|
"contributors": [
|
||||||
|
"Douglas Christopher Wilson <doug@somethingdoug.com>",
|
||||||
|
"Jonathan Ong <me@jongleberry.com> (http://jongleberry.com)"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"repository": "jshttp/accepts",
|
||||||
|
"dependencies": {
|
||||||
|
"mime-types": "^3.0.0",
|
||||||
|
"negotiator": "^1.0.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"deep-equal": "1.0.1",
|
||||||
|
"eslint": "7.32.0",
|
||||||
|
"eslint-config-standard": "14.1.1",
|
||||||
|
"eslint-plugin-import": "2.25.4",
|
||||||
|
"eslint-plugin-markdown": "2.2.1",
|
||||||
|
"eslint-plugin-node": "11.1.0",
|
||||||
|
"eslint-plugin-promise": "4.3.1",
|
||||||
|
"eslint-plugin-standard": "4.1.0",
|
||||||
|
"mocha": "9.2.0",
|
||||||
|
"nyc": "15.1.0"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"LICENSE",
|
||||||
|
"HISTORY.md",
|
||||||
|
"index.js"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.6"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"lint": "eslint .",
|
||||||
|
"test": "mocha --reporter spec --check-leaks --bail test/",
|
||||||
|
"test-ci": "nyc --reporter=lcov --reporter=text npm test",
|
||||||
|
"test-cov": "nyc --reporter=html --reporter=text npm test"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"content",
|
||||||
|
"negotiation",
|
||||||
|
"accept",
|
||||||
|
"accepts"
|
||||||
|
]
|
||||||
|
}
|
||||||
1
node_modules/append-field/.npmignore
generated
vendored
Normal file
1
node_modules/append-field/.npmignore
generated
vendored
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
node_modules/
|
||||||
21
node_modules/append-field/LICENSE
generated
vendored
Normal file
21
node_modules/append-field/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,21 @@
|
||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2015 Linus Unnebäck
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
44
node_modules/append-field/README.md
generated
vendored
Normal file
44
node_modules/append-field/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,44 @@
|
||||||
|
# `append-field`
|
||||||
|
|
||||||
|
A [W3C HTML JSON forms spec](http://www.w3.org/TR/html-json-forms/) compliant
|
||||||
|
field appender (for lack of a better name). Useful for people implementing
|
||||||
|
`application/x-www-form-urlencoded` and `multipart/form-data` parsers.
|
||||||
|
|
||||||
|
It works best on objects created with `Object.create(null)`. Otherwise it might
|
||||||
|
conflict with variables from the prototype (e.g. `hasOwnProperty`).
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm install --save append-field
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
var appendField = require('append-field')
|
||||||
|
var obj = Object.create(null)
|
||||||
|
|
||||||
|
appendField(obj, 'pets[0][species]', 'Dahut')
|
||||||
|
appendField(obj, 'pets[0][name]', 'Hypatia')
|
||||||
|
appendField(obj, 'pets[1][species]', 'Felis Stultus')
|
||||||
|
appendField(obj, 'pets[1][name]', 'Billie')
|
||||||
|
|
||||||
|
console.log(obj)
|
||||||
|
```
|
||||||
|
|
||||||
|
```text
|
||||||
|
{ pets:
|
||||||
|
[ { species: 'Dahut', name: 'Hypatia' },
|
||||||
|
{ species: 'Felis Stultus', name: 'Billie' } ] }
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### `appendField(store, key, value)`
|
||||||
|
|
||||||
|
Adds the field named `key` with the value `value` to the object `store`.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
MIT
|
||||||
12
node_modules/append-field/index.js
generated
vendored
Normal file
12
node_modules/append-field/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
var parsePath = require('./lib/parse-path')
|
||||||
|
var setValue = require('./lib/set-value')
|
||||||
|
|
||||||
|
function appendField (store, key, value) {
|
||||||
|
var steps = parsePath(key)
|
||||||
|
|
||||||
|
steps.reduce(function (context, step) {
|
||||||
|
return setValue(context, step, context[step.key], value)
|
||||||
|
}, store)
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = appendField
|
||||||
53
node_modules/append-field/lib/parse-path.js
generated
vendored
Normal file
53
node_modules/append-field/lib/parse-path.js
generated
vendored
Normal file
|
|
@ -0,0 +1,53 @@
|
||||||
|
var reFirstKey = /^[^\[]*/
|
||||||
|
var reDigitPath = /^\[(\d+)\]/
|
||||||
|
var reNormalPath = /^\[([^\]]+)\]/
|
||||||
|
|
||||||
|
function parsePath (key) {
|
||||||
|
function failure () {
|
||||||
|
return [{ type: 'object', key: key, last: true }]
|
||||||
|
}
|
||||||
|
|
||||||
|
var firstKey = reFirstKey.exec(key)[0]
|
||||||
|
if (!firstKey) return failure()
|
||||||
|
|
||||||
|
var len = key.length
|
||||||
|
var pos = firstKey.length
|
||||||
|
var tail = { type: 'object', key: firstKey }
|
||||||
|
var steps = [tail]
|
||||||
|
|
||||||
|
while (pos < len) {
|
||||||
|
var m
|
||||||
|
|
||||||
|
if (key[pos] === '[' && key[pos + 1] === ']') {
|
||||||
|
pos += 2
|
||||||
|
tail.append = true
|
||||||
|
if (pos !== len) return failure()
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
m = reDigitPath.exec(key.substring(pos))
|
||||||
|
if (m !== null) {
|
||||||
|
pos += m[0].length
|
||||||
|
tail.nextType = 'array'
|
||||||
|
tail = { type: 'array', key: parseInt(m[1], 10) }
|
||||||
|
steps.push(tail)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
m = reNormalPath.exec(key.substring(pos))
|
||||||
|
if (m !== null) {
|
||||||
|
pos += m[0].length
|
||||||
|
tail.nextType = 'object'
|
||||||
|
tail = { type: 'object', key: m[1] }
|
||||||
|
steps.push(tail)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
return failure()
|
||||||
|
}
|
||||||
|
|
||||||
|
tail.last = true
|
||||||
|
return steps
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = parsePath
|
||||||
64
node_modules/append-field/lib/set-value.js
generated
vendored
Normal file
64
node_modules/append-field/lib/set-value.js
generated
vendored
Normal file
|
|
@ -0,0 +1,64 @@
|
||||||
|
function valueType (value) {
|
||||||
|
if (value === undefined) return 'undefined'
|
||||||
|
if (Array.isArray(value)) return 'array'
|
||||||
|
if (typeof value === 'object') return 'object'
|
||||||
|
return 'scalar'
|
||||||
|
}
|
||||||
|
|
||||||
|
function setLastValue (context, step, currentValue, entryValue) {
|
||||||
|
switch (valueType(currentValue)) {
|
||||||
|
case 'undefined':
|
||||||
|
if (step.append) {
|
||||||
|
context[step.key] = [entryValue]
|
||||||
|
} else {
|
||||||
|
context[step.key] = entryValue
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case 'array':
|
||||||
|
context[step.key].push(entryValue)
|
||||||
|
break
|
||||||
|
case 'object':
|
||||||
|
return setLastValue(currentValue, { type: 'object', key: '', last: true }, currentValue[''], entryValue)
|
||||||
|
case 'scalar':
|
||||||
|
context[step.key] = [context[step.key], entryValue]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
return context
|
||||||
|
}
|
||||||
|
|
||||||
|
function setValue (context, step, currentValue, entryValue) {
|
||||||
|
if (step.last) return setLastValue(context, step, currentValue, entryValue)
|
||||||
|
|
||||||
|
var obj
|
||||||
|
switch (valueType(currentValue)) {
|
||||||
|
case 'undefined':
|
||||||
|
if (step.nextType === 'array') {
|
||||||
|
context[step.key] = []
|
||||||
|
} else {
|
||||||
|
context[step.key] = Object.create(null)
|
||||||
|
}
|
||||||
|
return context[step.key]
|
||||||
|
case 'object':
|
||||||
|
return context[step.key]
|
||||||
|
case 'array':
|
||||||
|
if (step.nextType === 'array') {
|
||||||
|
return currentValue
|
||||||
|
}
|
||||||
|
|
||||||
|
obj = Object.create(null)
|
||||||
|
context[step.key] = obj
|
||||||
|
currentValue.forEach(function (item, i) {
|
||||||
|
if (item !== undefined) obj['' + i] = item
|
||||||
|
})
|
||||||
|
|
||||||
|
return obj
|
||||||
|
case 'scalar':
|
||||||
|
obj = Object.create(null)
|
||||||
|
obj[''] = currentValue
|
||||||
|
context[step.key] = obj
|
||||||
|
return obj
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = setValue
|
||||||
19
node_modules/append-field/package.json
generated
vendored
Normal file
19
node_modules/append-field/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
{
|
||||||
|
"name": "append-field",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"license": "MIT",
|
||||||
|
"author": "Linus Unnebäck <linus@folkdatorn.se>",
|
||||||
|
"main": "index.js",
|
||||||
|
"devDependencies": {
|
||||||
|
"mocha": "^2.2.4",
|
||||||
|
"standard": "^6.0.5",
|
||||||
|
"testdata-w3c-json-form": "^0.2.0"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"test": "standard && mocha"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "http://github.com/LinusU/node-append-field.git"
|
||||||
|
}
|
||||||
|
}
|
||||||
19
node_modules/append-field/test/forms.js
generated
vendored
Normal file
19
node_modules/append-field/test/forms.js
generated
vendored
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
/* eslint-env mocha */
|
||||||
|
|
||||||
|
var assert = require('assert')
|
||||||
|
var appendField = require('../')
|
||||||
|
var testData = require('testdata-w3c-json-form')
|
||||||
|
|
||||||
|
describe('Append Field', function () {
|
||||||
|
for (var test of testData) {
|
||||||
|
it('handles ' + test.name, function () {
|
||||||
|
var store = Object.create(null)
|
||||||
|
|
||||||
|
for (var field of test.fields) {
|
||||||
|
appendField(store, field.key, field.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.deepEqual(store, test.expected)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
23
node_modules/body-parser/LICENSE
generated
vendored
Normal file
23
node_modules/body-parser/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
(The MIT License)
|
||||||
|
|
||||||
|
Copyright (c) 2014 Jonathan Ong <me@jongleberry.com>
|
||||||
|
Copyright (c) 2014-2015 Douglas Christopher Wilson <doug@somethingdoug.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
'Software'), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
||||||
|
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
||||||
|
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
||||||
|
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
494
node_modules/body-parser/README.md
generated
vendored
Normal file
494
node_modules/body-parser/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,494 @@
|
||||||
|
# body-parser
|
||||||
|
|
||||||
|
[![NPM Version][npm-version-image]][npm-url]
|
||||||
|
[![NPM Downloads][npm-downloads-image]][npm-url]
|
||||||
|
[![Build Status][ci-image]][ci-url]
|
||||||
|
[![Test Coverage][coveralls-image]][coveralls-url]
|
||||||
|
[![OpenSSF Scorecard Badge][ossf-scorecard-badge]][ossf-scorecard-visualizer]
|
||||||
|
|
||||||
|
Node.js body parsing middleware.
|
||||||
|
|
||||||
|
Parse incoming request bodies in a middleware before your handlers, available
|
||||||
|
under the `req.body` property.
|
||||||
|
|
||||||
|
**Note** As `req.body`'s shape is based on user-controlled input, all
|
||||||
|
properties and values in this object are untrusted and should be validated
|
||||||
|
before trusting. For example, `req.body.foo.toString()` may fail in multiple
|
||||||
|
ways, for example the `foo` property may not be there or may not be a string,
|
||||||
|
and `toString` may not be a function and instead a string or other user input.
|
||||||
|
|
||||||
|
[Learn about the anatomy of an HTTP transaction in Node.js](https://nodejs.org/en/learn/http/anatomy-of-an-http-transaction).
|
||||||
|
|
||||||
|
_This does not handle multipart bodies_, due to their complex and typically
|
||||||
|
large nature. For multipart bodies, you may be interested in the following
|
||||||
|
modules:
|
||||||
|
|
||||||
|
* [busboy](https://www.npmjs.com/package/busboy#readme) and
|
||||||
|
[connect-busboy](https://www.npmjs.com/package/connect-busboy#readme)
|
||||||
|
* [multiparty](https://www.npmjs.com/package/multiparty#readme) and
|
||||||
|
[connect-multiparty](https://www.npmjs.com/package/connect-multiparty#readme)
|
||||||
|
* [formidable](https://www.npmjs.com/package/formidable#readme)
|
||||||
|
* [multer](https://www.npmjs.com/package/multer#readme)
|
||||||
|
|
||||||
|
This module provides the following parsers:
|
||||||
|
|
||||||
|
* [JSON body parser](#bodyparserjsonoptions)
|
||||||
|
* [Raw body parser](#bodyparserrawoptions)
|
||||||
|
* [Text body parser](#bodyparsertextoptions)
|
||||||
|
* [URL-encoded form body parser](#bodyparserurlencodedoptions)
|
||||||
|
|
||||||
|
Other body parsers you might be interested in:
|
||||||
|
|
||||||
|
- [body](https://www.npmjs.com/package/body#readme)
|
||||||
|
- [co-body](https://www.npmjs.com/package/co-body#readme)
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
```sh
|
||||||
|
$ npm install body-parser
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
```js
|
||||||
|
const bodyParser = require('body-parser')
|
||||||
|
```
|
||||||
|
|
||||||
|
The `bodyParser` object exposes various factories to create middlewares. All
|
||||||
|
middlewares will populate the `req.body` property with the parsed body when
|
||||||
|
the `Content-Type` request header matches the `type` option.
|
||||||
|
|
||||||
|
The various errors returned by this module are described in the
|
||||||
|
[errors section](#errors).
|
||||||
|
|
||||||
|
### bodyParser.json([options])
|
||||||
|
|
||||||
|
Returns middleware that only parses `json` and only looks at requests where
|
||||||
|
the `Content-Type` header matches the `type` option. This parser accepts any
|
||||||
|
Unicode encoding of the body and supports automatic inflation of `gzip`,
|
||||||
|
`br` (brotli) and `deflate` encodings.
|
||||||
|
|
||||||
|
A new `body` object containing the parsed data is populated on the `request`
|
||||||
|
object after the middleware (i.e. `req.body`).
|
||||||
|
|
||||||
|
#### Options
|
||||||
|
|
||||||
|
The `json` function takes an optional `options` object that may contain any of
|
||||||
|
the following keys:
|
||||||
|
|
||||||
|
##### defaultCharset
|
||||||
|
|
||||||
|
Specify the default character set for the json content if the charset is not
|
||||||
|
specified in the `Content-Type` header of the request. Defaults to `utf-8`.
|
||||||
|
|
||||||
|
##### inflate
|
||||||
|
|
||||||
|
When set to `true`, then deflated (compressed) bodies will be inflated; when
|
||||||
|
`false`, deflated bodies are rejected. Defaults to `true`.
|
||||||
|
|
||||||
|
##### limit
|
||||||
|
|
||||||
|
Controls the maximum request body size. If this is a number, then the value
|
||||||
|
specifies the number of bytes; if it is a string, the value is passed to the
|
||||||
|
[bytes](https://www.npmjs.com/package/bytes) library for parsing. Defaults
|
||||||
|
to `'100kb'`.
|
||||||
|
|
||||||
|
##### reviver
|
||||||
|
|
||||||
|
The `reviver` option is passed directly to `JSON.parse` as the second
|
||||||
|
argument. You can find more information on this argument
|
||||||
|
[in the MDN documentation about JSON.parse](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/JSON/parse#Example.3A_Using_the_reviver_parameter).
|
||||||
|
|
||||||
|
##### strict
|
||||||
|
|
||||||
|
When set to `true`, will only accept arrays and objects; when `false` will
|
||||||
|
accept anything `JSON.parse` accepts. Defaults to `true`.
|
||||||
|
|
||||||
|
##### type
|
||||||
|
|
||||||
|
The `type` option is used to determine what media type the middleware will
|
||||||
|
parse. This option can be a string, array of strings, or a function. If not a
|
||||||
|
function, `type` option is passed directly to the
|
||||||
|
[type-is](https://www.npmjs.com/package/type-is#readme) library and this can
|
||||||
|
be an extension name (like `json`), a mime type (like `application/json`), or
|
||||||
|
a mime type with a wildcard (like `*/*` or `*/json`). If a function, the `type`
|
||||||
|
option is called as `fn(req)` and the request is parsed if it returns a truthy
|
||||||
|
value. Defaults to `application/json`.
|
||||||
|
|
||||||
|
##### verify
|
||||||
|
|
||||||
|
The `verify` option, if supplied, is called as `verify(req, res, buf, encoding)`,
|
||||||
|
where `buf` is a `Buffer` of the raw request body and `encoding` is the
|
||||||
|
encoding of the request. The parsing can be aborted by throwing an error.
|
||||||
|
|
||||||
|
### bodyParser.raw([options])
|
||||||
|
|
||||||
|
Returns middleware that parses all bodies as a `Buffer` and only looks at
|
||||||
|
requests where the `Content-Type` header matches the `type` option. This
|
||||||
|
parser supports automatic inflation of `gzip`, `br` (brotli) and `deflate`
|
||||||
|
encodings.
|
||||||
|
|
||||||
|
A new `body` object containing the parsed data is populated on the `request`
|
||||||
|
object after the middleware (i.e. `req.body`). This will be a `Buffer` object
|
||||||
|
of the body.
|
||||||
|
|
||||||
|
#### Options
|
||||||
|
|
||||||
|
The `raw` function takes an optional `options` object that may contain any of
|
||||||
|
the following keys:
|
||||||
|
|
||||||
|
##### inflate
|
||||||
|
|
||||||
|
When set to `true`, then deflated (compressed) bodies will be inflated; when
|
||||||
|
`false`, deflated bodies are rejected. Defaults to `true`.
|
||||||
|
|
||||||
|
##### limit
|
||||||
|
|
||||||
|
Controls the maximum request body size. If this is a number, then the value
|
||||||
|
specifies the number of bytes; if it is a string, the value is passed to the
|
||||||
|
[bytes](https://www.npmjs.com/package/bytes) library for parsing. Defaults
|
||||||
|
to `'100kb'`.
|
||||||
|
|
||||||
|
##### type
|
||||||
|
|
||||||
|
The `type` option is used to determine what media type the middleware will
|
||||||
|
parse. This option can be a string, array of strings, or a function.
|
||||||
|
If not a function, `type` option is passed directly to the
|
||||||
|
[type-is](https://www.npmjs.com/package/type-is#readme) library and this
|
||||||
|
can be an extension name (like `bin`), a mime type (like
|
||||||
|
`application/octet-stream`), or a mime type with a wildcard (like `*/*` or
|
||||||
|
`application/*`). If a function, the `type` option is called as `fn(req)`
|
||||||
|
and the request is parsed if it returns a truthy value. Defaults to
|
||||||
|
`application/octet-stream`.
|
||||||
|
|
||||||
|
##### verify
|
||||||
|
|
||||||
|
The `verify` option, if supplied, is called as `verify(req, res, buf, encoding)`,
|
||||||
|
where `buf` is a `Buffer` of the raw request body and `encoding` is the
|
||||||
|
encoding of the request. The parsing can be aborted by throwing an error.
|
||||||
|
|
||||||
|
### bodyParser.text([options])
|
||||||
|
|
||||||
|
Returns middleware that parses all bodies as a string and only looks at
|
||||||
|
requests where the `Content-Type` header matches the `type` option. This
|
||||||
|
parser supports automatic inflation of `gzip`, `br` (brotli) and `deflate`
|
||||||
|
encodings.
|
||||||
|
|
||||||
|
A new `body` string containing the parsed data is populated on the `request`
|
||||||
|
object after the middleware (i.e. `req.body`). This will be a string of the
|
||||||
|
body.
|
||||||
|
|
||||||
|
#### Options
|
||||||
|
|
||||||
|
The `text` function takes an optional `options` object that may contain any of
|
||||||
|
the following keys:
|
||||||
|
|
||||||
|
##### defaultCharset
|
||||||
|
|
||||||
|
Specify the default character set for the text content if the charset is not
|
||||||
|
specified in the `Content-Type` header of the request. Defaults to `utf-8`.
|
||||||
|
|
||||||
|
##### inflate
|
||||||
|
|
||||||
|
When set to `true`, then deflated (compressed) bodies will be inflated; when
|
||||||
|
`false`, deflated bodies are rejected. Defaults to `true`.
|
||||||
|
|
||||||
|
##### limit
|
||||||
|
|
||||||
|
Controls the maximum request body size. If this is a number, then the value
|
||||||
|
specifies the number of bytes; if it is a string, the value is passed to the
|
||||||
|
[bytes](https://www.npmjs.com/package/bytes) library for parsing. Defaults
|
||||||
|
to `'100kb'`.
|
||||||
|
|
||||||
|
##### type
|
||||||
|
|
||||||
|
The `type` option is used to determine what media type the middleware will
|
||||||
|
parse. This option can be a string, array of strings, or a function. If not
|
||||||
|
a function, `type` option is passed directly to the
|
||||||
|
[type-is](https://www.npmjs.com/package/type-is#readme) library and this can
|
||||||
|
be an extension name (like `txt`), a mime type (like `text/plain`), or a mime
|
||||||
|
type with a wildcard (like `*/*` or `text/*`). If a function, the `type`
|
||||||
|
option is called as `fn(req)` and the request is parsed if it returns a
|
||||||
|
truthy value. Defaults to `text/plain`.
|
||||||
|
|
||||||
|
##### verify
|
||||||
|
|
||||||
|
The `verify` option, if supplied, is called as `verify(req, res, buf, encoding)`,
|
||||||
|
where `buf` is a `Buffer` of the raw request body and `encoding` is the
|
||||||
|
encoding of the request. The parsing can be aborted by throwing an error.
|
||||||
|
|
||||||
|
### bodyParser.urlencoded([options])
|
||||||
|
|
||||||
|
Returns middleware that only parses `urlencoded` bodies and only looks at
|
||||||
|
requests where the `Content-Type` header matches the `type` option. This
|
||||||
|
parser accepts only UTF-8 and ISO-8859-1 encodings of the body and supports
|
||||||
|
automatic inflation of `gzip`, `br` (brotli) and `deflate` encodings.
|
||||||
|
|
||||||
|
A new `body` object containing the parsed data is populated on the `request`
|
||||||
|
object after the middleware (i.e. `req.body`). This object will contain
|
||||||
|
key-value pairs, where the value can be a string or array (when `extended` is
|
||||||
|
`false`), or any type (when `extended` is `true`).
|
||||||
|
|
||||||
|
#### Options
|
||||||
|
|
||||||
|
The `urlencoded` function takes an optional `options` object that may contain
|
||||||
|
any of the following keys:
|
||||||
|
|
||||||
|
##### extended
|
||||||
|
|
||||||
|
The "extended" syntax allows for rich objects and arrays to be encoded into the
|
||||||
|
URL-encoded format, allowing for a JSON-like experience with URL-encoded. For
|
||||||
|
more information, please [see the qs
|
||||||
|
library](https://www.npmjs.com/package/qs#readme).
|
||||||
|
|
||||||
|
Defaults to `false`.
|
||||||
|
|
||||||
|
##### inflate
|
||||||
|
|
||||||
|
When set to `true`, then deflated (compressed) bodies will be inflated; when
|
||||||
|
`false`, deflated bodies are rejected. Defaults to `true`.
|
||||||
|
|
||||||
|
##### limit
|
||||||
|
|
||||||
|
Controls the maximum request body size. If this is a number, then the value
|
||||||
|
specifies the number of bytes; if it is a string, the value is passed to the
|
||||||
|
[bytes](https://www.npmjs.com/package/bytes) library for parsing. Defaults
|
||||||
|
to `'100kb'`.
|
||||||
|
|
||||||
|
##### parameterLimit
|
||||||
|
|
||||||
|
The `parameterLimit` option controls the maximum number of parameters that
|
||||||
|
are allowed in the URL-encoded data. If a request contains more parameters
|
||||||
|
than this value, a 413 will be returned to the client. Defaults to `1000`.
|
||||||
|
|
||||||
|
##### type
|
||||||
|
|
||||||
|
The `type` option is used to determine what media type the middleware will
|
||||||
|
parse. This option can be a string, array of strings, or a function. If not
|
||||||
|
a function, `type` option is passed directly to the
|
||||||
|
[type-is](https://www.npmjs.com/package/type-is#readme) library and this can
|
||||||
|
be an extension name (like `urlencoded`), a mime type (like
|
||||||
|
`application/x-www-form-urlencoded`), or a mime type with a wildcard (like
|
||||||
|
`*/x-www-form-urlencoded`). If a function, the `type` option is called as
|
||||||
|
`fn(req)` and the request is parsed if it returns a truthy value. Defaults
|
||||||
|
to `application/x-www-form-urlencoded`.
|
||||||
|
|
||||||
|
##### verify
|
||||||
|
|
||||||
|
The `verify` option, if supplied, is called as `verify(req, res, buf, encoding)`,
|
||||||
|
where `buf` is a `Buffer` of the raw request body and `encoding` is the
|
||||||
|
encoding of the request. The parsing can be aborted by throwing an error.
|
||||||
|
|
||||||
|
##### defaultCharset
|
||||||
|
|
||||||
|
The default charset to parse as, if not specified in content-type. Must be
|
||||||
|
either `utf-8` or `iso-8859-1`. Defaults to `utf-8`.
|
||||||
|
|
||||||
|
##### charsetSentinel
|
||||||
|
|
||||||
|
Whether to let the value of the `utf8` parameter take precedence as the charset
|
||||||
|
selector. It requires the form to contain a parameter named `utf8` with a value
|
||||||
|
of `✓`. Defaults to `false`.
|
||||||
|
|
||||||
|
##### interpretNumericEntities
|
||||||
|
|
||||||
|
Whether to decode numeric entities such as `☺` when parsing an iso-8859-1
|
||||||
|
form. Defaults to `false`.
|
||||||
|
|
||||||
|
|
||||||
|
##### depth
|
||||||
|
|
||||||
|
The `depth` option is used to configure the maximum depth of the `qs` library when `extended` is `true`. This allows you to limit the amount of keys that are parsed and can be useful to prevent certain types of abuse. Defaults to `32`. It is recommended to keep this value as low as possible.
|
||||||
|
|
||||||
|
## Errors
|
||||||
|
|
||||||
|
The middlewares provided by this module create errors using the
|
||||||
|
[`http-errors` module](https://www.npmjs.com/package/http-errors). The errors
|
||||||
|
will typically have a `status`/`statusCode` property that contains the suggested
|
||||||
|
HTTP response code, an `expose` property to determine if the `message` property
|
||||||
|
should be displayed to the client, a `type` property to determine the type of
|
||||||
|
error without matching against the `message`, and a `body` property containing
|
||||||
|
the read body, if available.
|
||||||
|
|
||||||
|
The following are the common errors created, though any error can come through
|
||||||
|
for various reasons.
|
||||||
|
|
||||||
|
### content encoding unsupported
|
||||||
|
|
||||||
|
This error will occur when the request had a `Content-Encoding` header that
|
||||||
|
contained an encoding but the "inflation" option was set to `false`. The
|
||||||
|
`status` property is set to `415`, the `type` property is set to
|
||||||
|
`'encoding.unsupported'`, and the `charset` property will be set to the
|
||||||
|
encoding that is unsupported.
|
||||||
|
|
||||||
|
### entity parse failed
|
||||||
|
|
||||||
|
This error will occur when the request contained an entity that could not be
|
||||||
|
parsed by the middleware. The `status` property is set to `400`, the `type`
|
||||||
|
property is set to `'entity.parse.failed'`, and the `body` property is set to
|
||||||
|
the entity value that failed parsing.
|
||||||
|
|
||||||
|
### entity verify failed
|
||||||
|
|
||||||
|
This error will occur when the request contained an entity that could not be
|
||||||
|
failed verification by the defined `verify` option. The `status` property is
|
||||||
|
set to `403`, the `type` property is set to `'entity.verify.failed'`, and the
|
||||||
|
`body` property is set to the entity value that failed verification.
|
||||||
|
|
||||||
|
### request aborted
|
||||||
|
|
||||||
|
This error will occur when the request is aborted by the client before reading
|
||||||
|
the body has finished. The `received` property will be set to the number of
|
||||||
|
bytes received before the request was aborted and the `expected` property is
|
||||||
|
set to the number of expected bytes. The `status` property is set to `400`
|
||||||
|
and `type` property is set to `'request.aborted'`.
|
||||||
|
|
||||||
|
### request entity too large
|
||||||
|
|
||||||
|
This error will occur when the request body's size is larger than the "limit"
|
||||||
|
option. The `limit` property will be set to the byte limit and the `length`
|
||||||
|
property will be set to the request body's length. The `status` property is
|
||||||
|
set to `413` and the `type` property is set to `'entity.too.large'`.
|
||||||
|
|
||||||
|
### request size did not match content length
|
||||||
|
|
||||||
|
This error will occur when the request's length did not match the length from
|
||||||
|
the `Content-Length` header. This typically occurs when the request is malformed,
|
||||||
|
typically when the `Content-Length` header was calculated based on characters
|
||||||
|
instead of bytes. The `status` property is set to `400` and the `type` property
|
||||||
|
is set to `'request.size.invalid'`.
|
||||||
|
|
||||||
|
### stream encoding should not be set
|
||||||
|
|
||||||
|
This error will occur when something called the `req.setEncoding` method prior
|
||||||
|
to this middleware. This module operates directly on bytes only and you cannot
|
||||||
|
call `req.setEncoding` when using this module. The `status` property is set to
|
||||||
|
`500` and the `type` property is set to `'stream.encoding.set'`.
|
||||||
|
|
||||||
|
### stream is not readable
|
||||||
|
|
||||||
|
This error will occur when the request is no longer readable when this middleware
|
||||||
|
attempts to read it. This typically means something other than a middleware from
|
||||||
|
this module read the request body already and the middleware was also configured to
|
||||||
|
read the same request. The `status` property is set to `500` and the `type`
|
||||||
|
property is set to `'stream.not.readable'`.
|
||||||
|
|
||||||
|
### too many parameters
|
||||||
|
|
||||||
|
This error will occur when the content of the request exceeds the configured
|
||||||
|
`parameterLimit` for the `urlencoded` parser. The `status` property is set to
|
||||||
|
`413` and the `type` property is set to `'parameters.too.many'`.
|
||||||
|
|
||||||
|
### unsupported charset "BOGUS"
|
||||||
|
|
||||||
|
This error will occur when the request had a charset parameter in the
|
||||||
|
`Content-Type` header, but the `iconv-lite` module does not support it OR the
|
||||||
|
parser does not support it. The charset is contained in the message as well
|
||||||
|
as in the `charset` property. The `status` property is set to `415`, the
|
||||||
|
`type` property is set to `'charset.unsupported'`, and the `charset` property
|
||||||
|
is set to the charset that is unsupported.
|
||||||
|
|
||||||
|
### unsupported content encoding "bogus"
|
||||||
|
|
||||||
|
This error will occur when the request had a `Content-Encoding` header that
|
||||||
|
contained an unsupported encoding. The encoding is contained in the message
|
||||||
|
as well as in the `encoding` property. The `status` property is set to `415`,
|
||||||
|
the `type` property is set to `'encoding.unsupported'`, and the `encoding`
|
||||||
|
property is set to the encoding that is unsupported.
|
||||||
|
|
||||||
|
### The input exceeded the depth
|
||||||
|
|
||||||
|
This error occurs when using `bodyParser.urlencoded` with the `extended` property set to `true` and the input exceeds the configured `depth` option. The `status` property is set to `400`. It is recommended to review the `depth` option and evaluate if it requires a higher value. When the `depth` option is set to `32` (default value), the error will not be thrown.
|
||||||
|
|
||||||
|
## Examples
|
||||||
|
|
||||||
|
### Express/Connect top-level generic
|
||||||
|
|
||||||
|
This example demonstrates adding a generic JSON and URL-encoded parser as a
|
||||||
|
top-level middleware, which will parse the bodies of all incoming requests.
|
||||||
|
This is the simplest setup.
|
||||||
|
|
||||||
|
```js
|
||||||
|
const express = require('express')
|
||||||
|
const bodyParser = require('body-parser')
|
||||||
|
|
||||||
|
const app = express()
|
||||||
|
|
||||||
|
// parse application/x-www-form-urlencoded
|
||||||
|
app.use(bodyParser.urlencoded())
|
||||||
|
|
||||||
|
// parse application/json
|
||||||
|
app.use(bodyParser.json())
|
||||||
|
|
||||||
|
app.use(function (req, res) {
|
||||||
|
res.setHeader('Content-Type', 'text/plain')
|
||||||
|
res.write('you posted:\n')
|
||||||
|
res.end(String(JSON.stringify(req.body, null, 2)))
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
### Express route-specific
|
||||||
|
|
||||||
|
This example demonstrates adding body parsers specifically to the routes that
|
||||||
|
need them. In general, this is the most recommended way to use body-parser with
|
||||||
|
Express.
|
||||||
|
|
||||||
|
```js
|
||||||
|
const express = require('express')
|
||||||
|
const bodyParser = require('body-parser')
|
||||||
|
|
||||||
|
const app = express()
|
||||||
|
|
||||||
|
// create application/json parser
|
||||||
|
const jsonParser = bodyParser.json()
|
||||||
|
|
||||||
|
// create application/x-www-form-urlencoded parser
|
||||||
|
const urlencodedParser = bodyParser.urlencoded()
|
||||||
|
|
||||||
|
// POST /login gets urlencoded bodies
|
||||||
|
app.post('/login', urlencodedParser, function (req, res) {
|
||||||
|
if (!req.body || !req.body.username) res.sendStatus(400)
|
||||||
|
res.send('welcome, ' + req.body.username)
|
||||||
|
})
|
||||||
|
|
||||||
|
// POST /api/users gets JSON bodies
|
||||||
|
app.post('/api/users', jsonParser, function (req, res) {
|
||||||
|
if (!req.body) res.sendStatus(400)
|
||||||
|
// create user in req.body
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
### Change accepted type for parsers
|
||||||
|
|
||||||
|
All the parsers accept a `type` option which allows you to change the
|
||||||
|
`Content-Type` that the middleware will parse.
|
||||||
|
|
||||||
|
```js
|
||||||
|
const express = require('express')
|
||||||
|
const bodyParser = require('body-parser')
|
||||||
|
|
||||||
|
const app = express()
|
||||||
|
|
||||||
|
// parse various different custom JSON types as JSON
|
||||||
|
app.use(bodyParser.json({ type: 'application/*+json' }))
|
||||||
|
|
||||||
|
// parse some custom thing into a Buffer
|
||||||
|
app.use(bodyParser.raw({ type: 'application/vnd.custom-type' }))
|
||||||
|
|
||||||
|
// parse an HTML body into a string
|
||||||
|
app.use(bodyParser.text({ type: 'text/html' }))
|
||||||
|
```
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
[MIT](LICENSE)
|
||||||
|
|
||||||
|
[ci-image]: https://img.shields.io/github/actions/workflow/status/expressjs/body-parser/ci.yml?branch=master&label=ci
|
||||||
|
[ci-url]: https://github.com/expressjs/body-parser/actions/workflows/ci.yml
|
||||||
|
[coveralls-image]: https://img.shields.io/coverallsCoverage/github/expressjs/body-parser?branch=master
|
||||||
|
[coveralls-url]: https://coveralls.io/r/expressjs/body-parser?branch=master
|
||||||
|
[npm-downloads-image]: https://img.shields.io/npm/dm/body-parser
|
||||||
|
[npm-url]: https://npmjs.com/package/body-parser
|
||||||
|
[npm-version-image]: https://img.shields.io/npm/v/body-parser
|
||||||
|
[ossf-scorecard-badge]: https://api.scorecard.dev/projects/github.com/expressjs/body-parser/badge
|
||||||
|
[ossf-scorecard-visualizer]: https://ossf.github.io/scorecard-visualizer/#/projects/github.com/expressjs/body-parser
|
||||||
71
node_modules/body-parser/index.js
generated
vendored
Normal file
71
node_modules/body-parser/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1,71 @@
|
||||||
|
/*!
|
||||||
|
* body-parser
|
||||||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} Parsers
|
||||||
|
* @property {Function} json JSON parser
|
||||||
|
* @property {Function} raw Raw parser
|
||||||
|
* @property {Function} text Text parser
|
||||||
|
* @property {Function} urlencoded URL-encoded parser
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
* @type {Function & Parsers}
|
||||||
|
*/
|
||||||
|
exports = module.exports = bodyParser
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON parser.
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
Object.defineProperty(exports, 'json', {
|
||||||
|
configurable: true,
|
||||||
|
enumerable: true,
|
||||||
|
get: () => require('./lib/types/json')
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Raw parser.
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
Object.defineProperty(exports, 'raw', {
|
||||||
|
configurable: true,
|
||||||
|
enumerable: true,
|
||||||
|
get: () => require('./lib/types/raw')
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text parser.
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
Object.defineProperty(exports, 'text', {
|
||||||
|
configurable: true,
|
||||||
|
enumerable: true,
|
||||||
|
get: () => require('./lib/types/text')
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* URL-encoded parser.
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
Object.defineProperty(exports, 'urlencoded', {
|
||||||
|
configurable: true,
|
||||||
|
enumerable: true,
|
||||||
|
get: () => require('./lib/types/urlencoded')
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a middleware to parse json and urlencoded bodies.
|
||||||
|
*
|
||||||
|
* @deprecated
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
function bodyParser () {
|
||||||
|
throw new Error('The bodyParser() generic has been split into individual middleware to use instead.')
|
||||||
|
}
|
||||||
247
node_modules/body-parser/lib/read.js
generated
vendored
Normal file
247
node_modules/body-parser/lib/read.js
generated
vendored
Normal file
|
|
@ -0,0 +1,247 @@
|
||||||
|
/*!
|
||||||
|
* body-parser
|
||||||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
var createError = require('http-errors')
|
||||||
|
var getBody = require('raw-body')
|
||||||
|
var iconv = require('iconv-lite')
|
||||||
|
var onFinished = require('on-finished')
|
||||||
|
var zlib = require('node:zlib')
|
||||||
|
var hasBody = require('type-is').hasBody
|
||||||
|
var { getCharset } = require('./utils')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = read
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read a request into a buffer and parse.
|
||||||
|
*
|
||||||
|
* @param {Object} req
|
||||||
|
* @param {Object} res
|
||||||
|
* @param {Function} next
|
||||||
|
* @param {Function} parse
|
||||||
|
* @param {Function} debug
|
||||||
|
* @param {Object} options
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function read (req, res, next, parse, debug, options) {
|
||||||
|
if (onFinished.isFinished(req)) {
|
||||||
|
debug('body already parsed')
|
||||||
|
next()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!('body' in req)) {
|
||||||
|
req.body = undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
// skip requests without bodies
|
||||||
|
if (!hasBody(req)) {
|
||||||
|
debug('skip empty body')
|
||||||
|
next()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
debug('content-type %j', req.headers['content-type'])
|
||||||
|
|
||||||
|
// determine if request should be parsed
|
||||||
|
if (!options.shouldParse(req)) {
|
||||||
|
debug('skip parsing')
|
||||||
|
next()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var encoding = null
|
||||||
|
if (options?.skipCharset !== true) {
|
||||||
|
encoding = getCharset(req) || options.defaultCharset
|
||||||
|
|
||||||
|
// validate charset
|
||||||
|
if (!!options?.isValidCharset && !options.isValidCharset(encoding)) {
|
||||||
|
debug('invalid charset')
|
||||||
|
next(createError(415, 'unsupported charset "' + encoding.toUpperCase() + '"', {
|
||||||
|
charset: encoding,
|
||||||
|
type: 'charset.unsupported'
|
||||||
|
}))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var length
|
||||||
|
var opts = options
|
||||||
|
var stream
|
||||||
|
|
||||||
|
// read options
|
||||||
|
var verify = opts.verify
|
||||||
|
|
||||||
|
try {
|
||||||
|
// get the content stream
|
||||||
|
stream = contentstream(req, debug, opts.inflate)
|
||||||
|
length = stream.length
|
||||||
|
stream.length = undefined
|
||||||
|
} catch (err) {
|
||||||
|
return next(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// set raw-body options
|
||||||
|
opts.length = length
|
||||||
|
opts.encoding = verify
|
||||||
|
? null
|
||||||
|
: encoding
|
||||||
|
|
||||||
|
// assert charset is supported
|
||||||
|
if (opts.encoding === null && encoding !== null && !iconv.encodingExists(encoding)) {
|
||||||
|
return next(createError(415, 'unsupported charset "' + encoding.toUpperCase() + '"', {
|
||||||
|
charset: encoding.toLowerCase(),
|
||||||
|
type: 'charset.unsupported'
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
// read body
|
||||||
|
debug('read body')
|
||||||
|
getBody(stream, opts, function (error, body) {
|
||||||
|
if (error) {
|
||||||
|
var _error
|
||||||
|
|
||||||
|
if (error.type === 'encoding.unsupported') {
|
||||||
|
// echo back charset
|
||||||
|
_error = createError(415, 'unsupported charset "' + encoding.toUpperCase() + '"', {
|
||||||
|
charset: encoding.toLowerCase(),
|
||||||
|
type: 'charset.unsupported'
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// set status code on error
|
||||||
|
_error = createError(400, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// unpipe from stream and destroy
|
||||||
|
if (stream !== req) {
|
||||||
|
req.unpipe()
|
||||||
|
stream.destroy()
|
||||||
|
}
|
||||||
|
|
||||||
|
// read off entire request
|
||||||
|
dump(req, function onfinished () {
|
||||||
|
next(createError(400, _error))
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// verify
|
||||||
|
if (verify) {
|
||||||
|
try {
|
||||||
|
debug('verify body')
|
||||||
|
verify(req, res, body, encoding)
|
||||||
|
} catch (err) {
|
||||||
|
next(createError(403, err, {
|
||||||
|
body: body,
|
||||||
|
type: err.type || 'entity.verify.failed'
|
||||||
|
}))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// parse
|
||||||
|
var str = body
|
||||||
|
try {
|
||||||
|
debug('parse body')
|
||||||
|
str = typeof body !== 'string' && encoding !== null
|
||||||
|
? iconv.decode(body, encoding)
|
||||||
|
: body
|
||||||
|
req.body = parse(str, encoding)
|
||||||
|
} catch (err) {
|
||||||
|
next(createError(400, err, {
|
||||||
|
body: str,
|
||||||
|
type: err.type || 'entity.parse.failed'
|
||||||
|
}))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
next()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the content stream of the request.
|
||||||
|
*
|
||||||
|
* @param {Object} req
|
||||||
|
* @param {Function} debug
|
||||||
|
* @param {boolean} inflate
|
||||||
|
* @returns {Object}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function contentstream (req, debug, inflate) {
|
||||||
|
var encoding = (req.headers['content-encoding'] || 'identity').toLowerCase()
|
||||||
|
var length = req.headers['content-length']
|
||||||
|
|
||||||
|
debug('content-encoding "%s"', encoding)
|
||||||
|
|
||||||
|
if (inflate === false && encoding !== 'identity') {
|
||||||
|
throw createError(415, 'content encoding unsupported', {
|
||||||
|
encoding: encoding,
|
||||||
|
type: 'encoding.unsupported'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (encoding === 'identity') {
|
||||||
|
req.length = length
|
||||||
|
return req
|
||||||
|
}
|
||||||
|
|
||||||
|
var stream = createDecompressionStream(encoding, debug)
|
||||||
|
req.pipe(stream)
|
||||||
|
return stream
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a decompression stream for the given encoding.
|
||||||
|
* @param {string} encoding
|
||||||
|
* @param {Function} debug
|
||||||
|
* @returns {Object}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function createDecompressionStream (encoding, debug) {
|
||||||
|
switch (encoding) {
|
||||||
|
case 'deflate':
|
||||||
|
debug('inflate body')
|
||||||
|
return zlib.createInflate()
|
||||||
|
case 'gzip':
|
||||||
|
debug('gunzip body')
|
||||||
|
return zlib.createGunzip()
|
||||||
|
case 'br':
|
||||||
|
debug('brotli decompress body')
|
||||||
|
return zlib.createBrotliDecompress()
|
||||||
|
default:
|
||||||
|
throw createError(415, 'unsupported content encoding "' + encoding + '"', {
|
||||||
|
encoding: encoding,
|
||||||
|
type: 'encoding.unsupported'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dump the contents of a request.
|
||||||
|
*
|
||||||
|
* @param {Object} req
|
||||||
|
* @param {Function} callback
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function dump (req, callback) {
|
||||||
|
if (onFinished.isFinished(req)) {
|
||||||
|
callback(null)
|
||||||
|
} else {
|
||||||
|
onFinished(req, callback)
|
||||||
|
req.resume()
|
||||||
|
}
|
||||||
|
}
|
||||||
158
node_modules/body-parser/lib/types/json.js
generated
vendored
Normal file
158
node_modules/body-parser/lib/types/json.js
generated
vendored
Normal file
|
|
@ -0,0 +1,158 @@
|
||||||
|
/*!
|
||||||
|
* body-parser
|
||||||
|
* Copyright(c) 2014 Jonathan Ong
|
||||||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
var debug = require('debug')('body-parser:json')
|
||||||
|
var read = require('../read')
|
||||||
|
var { normalizeOptions } = require('../utils')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = json
|
||||||
|
|
||||||
|
/**
|
||||||
|
* RegExp to match the first non-space in a string.
|
||||||
|
*
|
||||||
|
* Allowed whitespace is defined in RFC 7159:
|
||||||
|
*
|
||||||
|
* ws = *(
|
||||||
|
* %x20 / ; Space
|
||||||
|
* %x09 / ; Horizontal tab
|
||||||
|
* %x0A / ; Line feed or New line
|
||||||
|
* %x0D ) ; Carriage return
|
||||||
|
*/
|
||||||
|
var FIRST_CHAR_REGEXP = /^[\x20\x09\x0a\x0d]*([^\x20\x09\x0a\x0d])/ // eslint-disable-line no-control-regex
|
||||||
|
|
||||||
|
var JSON_SYNTAX_CHAR = '#'
|
||||||
|
var JSON_SYNTAX_REGEXP = /#+/g
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a middleware to parse JSON bodies.
|
||||||
|
*
|
||||||
|
* @param {Object} [options]
|
||||||
|
* @returns {Function}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
function json (options) {
|
||||||
|
const normalizedOptions = normalizeOptions(options, 'application/json')
|
||||||
|
|
||||||
|
var reviver = options?.reviver
|
||||||
|
var strict = options?.strict !== false
|
||||||
|
|
||||||
|
function parse (body) {
|
||||||
|
if (body.length === 0) {
|
||||||
|
// special-case empty json body, as it's a common client-side mistake
|
||||||
|
// TODO: maybe make this configurable or part of "strict" option
|
||||||
|
return {}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strict) {
|
||||||
|
var first = firstchar(body)
|
||||||
|
|
||||||
|
if (first !== '{' && first !== '[') {
|
||||||
|
debug('strict violation')
|
||||||
|
throw createStrictSyntaxError(body, first)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
debug('parse json')
|
||||||
|
return JSON.parse(body, reviver)
|
||||||
|
} catch (e) {
|
||||||
|
throw normalizeJsonSyntaxError(e, {
|
||||||
|
message: e.message,
|
||||||
|
stack: e.stack
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const readOptions = {
|
||||||
|
...normalizedOptions,
|
||||||
|
// assert charset per RFC 7159 sec 8.1
|
||||||
|
isValidCharset: (charset) => charset.slice(0, 4) === 'utf-'
|
||||||
|
}
|
||||||
|
|
||||||
|
return function jsonParser (req, res, next) {
|
||||||
|
read(req, res, next, parse, debug, readOptions)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create strict violation syntax error matching native error.
|
||||||
|
*
|
||||||
|
* @param {string} str
|
||||||
|
* @param {string} char
|
||||||
|
* @returns {Error}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function createStrictSyntaxError (str, char) {
|
||||||
|
var index = str.indexOf(char)
|
||||||
|
var partial = ''
|
||||||
|
|
||||||
|
if (index !== -1) {
|
||||||
|
partial = str.substring(0, index) + JSON_SYNTAX_CHAR.repeat(str.length - index)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSON.parse(partial); /* istanbul ignore next */ throw new SyntaxError('strict violation')
|
||||||
|
} catch (e) {
|
||||||
|
return normalizeJsonSyntaxError(e, {
|
||||||
|
message: e.message.replace(JSON_SYNTAX_REGEXP, function (placeholder) {
|
||||||
|
return str.substring(index, index + placeholder.length)
|
||||||
|
}),
|
||||||
|
stack: e.stack
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the first non-whitespace character in a string.
|
||||||
|
*
|
||||||
|
* @param {string} str
|
||||||
|
* @returns {string|undefined}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function firstchar (str) {
|
||||||
|
var match = FIRST_CHAR_REGEXP.exec(str)
|
||||||
|
|
||||||
|
return match
|
||||||
|
? match[1]
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalize a SyntaxError for JSON.parse.
|
||||||
|
*
|
||||||
|
* @param {SyntaxError} error
|
||||||
|
* @param {Object} obj
|
||||||
|
* @returns {SyntaxError}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function normalizeJsonSyntaxError (error, obj) {
|
||||||
|
var keys = Object.getOwnPropertyNames(error)
|
||||||
|
|
||||||
|
for (var i = 0; i < keys.length; i++) {
|
||||||
|
var key = keys[i]
|
||||||
|
if (key !== 'stack' && key !== 'message') {
|
||||||
|
delete error[key]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// replace stack before message for Node.js 0.10 and below
|
||||||
|
error.stack = obj.stack.replace(error.message, obj.message)
|
||||||
|
error.message = obj.message
|
||||||
|
|
||||||
|
return error
|
||||||
|
}
|
||||||
42
node_modules/body-parser/lib/types/raw.js
generated
vendored
Normal file
42
node_modules/body-parser/lib/types/raw.js
generated
vendored
Normal file
|
|
@ -0,0 +1,42 @@
|
||||||
|
/*!
|
||||||
|
* body-parser
|
||||||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
*/
|
||||||
|
|
||||||
|
var debug = require('debug')('body-parser:raw')
|
||||||
|
var read = require('../read')
|
||||||
|
var { normalizeOptions, passthrough } = require('../utils')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = raw
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a middleware to parse raw bodies.
|
||||||
|
*
|
||||||
|
* @param {Object} [options]
|
||||||
|
* @returns {Function}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
function raw (options) {
|
||||||
|
const normalizedOptions = normalizeOptions(options, 'application/octet-stream')
|
||||||
|
|
||||||
|
const readOptions = {
|
||||||
|
...normalizedOptions,
|
||||||
|
// Skip charset validation and parse the body as is
|
||||||
|
skipCharset: true
|
||||||
|
}
|
||||||
|
|
||||||
|
return function rawParser (req, res, next) {
|
||||||
|
read(req, res, next, passthrough, debug, readOptions)
|
||||||
|
}
|
||||||
|
}
|
||||||
36
node_modules/body-parser/lib/types/text.js
generated
vendored
Normal file
36
node_modules/body-parser/lib/types/text.js
generated
vendored
Normal file
|
|
@ -0,0 +1,36 @@
|
||||||
|
/*!
|
||||||
|
* body-parser
|
||||||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
*/
|
||||||
|
|
||||||
|
var debug = require('debug')('body-parser:text')
|
||||||
|
var read = require('../read')
|
||||||
|
var { normalizeOptions, passthrough } = require('../utils')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = text
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a middleware to parse text bodies.
|
||||||
|
*
|
||||||
|
* @param {Object} [options]
|
||||||
|
* @returns {Function}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
function text (options) {
|
||||||
|
const normalizedOptions = normalizeOptions(options, 'text/plain')
|
||||||
|
|
||||||
|
return function textParser (req, res, next) {
|
||||||
|
read(req, res, next, passthrough, debug, normalizedOptions)
|
||||||
|
}
|
||||||
|
}
|
||||||
142
node_modules/body-parser/lib/types/urlencoded.js
generated
vendored
Normal file
142
node_modules/body-parser/lib/types/urlencoded.js
generated
vendored
Normal file
|
|
@ -0,0 +1,142 @@
|
||||||
|
/*!
|
||||||
|
* body-parser
|
||||||
|
* Copyright(c) 2014 Jonathan Ong
|
||||||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
var createError = require('http-errors')
|
||||||
|
var debug = require('debug')('body-parser:urlencoded')
|
||||||
|
var read = require('../read')
|
||||||
|
var qs = require('qs')
|
||||||
|
var { normalizeOptions } = require('../utils')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = urlencoded
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a middleware to parse urlencoded bodies.
|
||||||
|
*
|
||||||
|
* @param {Object} [options]
|
||||||
|
* @returns {Function}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
function urlencoded (options) {
|
||||||
|
const normalizedOptions = normalizeOptions(options, 'application/x-www-form-urlencoded')
|
||||||
|
|
||||||
|
if (normalizedOptions.defaultCharset !== 'utf-8' && normalizedOptions.defaultCharset !== 'iso-8859-1') {
|
||||||
|
throw new TypeError('option defaultCharset must be either utf-8 or iso-8859-1')
|
||||||
|
}
|
||||||
|
|
||||||
|
// create the appropriate query parser
|
||||||
|
var queryparse = createQueryParser(options)
|
||||||
|
|
||||||
|
function parse (body, encoding) {
|
||||||
|
return body.length
|
||||||
|
? queryparse(body, encoding)
|
||||||
|
: {}
|
||||||
|
}
|
||||||
|
|
||||||
|
const readOptions = {
|
||||||
|
...normalizedOptions,
|
||||||
|
// assert charset
|
||||||
|
isValidCharset: (charset) => charset === 'utf-8' || charset === 'iso-8859-1'
|
||||||
|
}
|
||||||
|
|
||||||
|
return function urlencodedParser (req, res, next) {
|
||||||
|
read(req, res, next, parse, debug, readOptions)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the extended query parser.
|
||||||
|
*
|
||||||
|
* @param {Object} options
|
||||||
|
* @returns {Function}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function createQueryParser (options) {
|
||||||
|
var extended = Boolean(options?.extended)
|
||||||
|
var parameterLimit = options?.parameterLimit !== undefined
|
||||||
|
? options?.parameterLimit
|
||||||
|
: 1000
|
||||||
|
var charsetSentinel = options?.charsetSentinel
|
||||||
|
var interpretNumericEntities = options?.interpretNumericEntities
|
||||||
|
var depth = extended ? (options?.depth !== undefined ? options?.depth : 32) : 0
|
||||||
|
|
||||||
|
if (isNaN(parameterLimit) || parameterLimit < 1) {
|
||||||
|
throw new TypeError('option parameterLimit must be a positive number')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isNaN(depth) || depth < 0) {
|
||||||
|
throw new TypeError('option depth must be a zero or a positive number')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isFinite(parameterLimit)) {
|
||||||
|
parameterLimit = parameterLimit | 0
|
||||||
|
}
|
||||||
|
|
||||||
|
return function queryparse (body, encoding) {
|
||||||
|
var paramCount = parameterCount(body, parameterLimit)
|
||||||
|
|
||||||
|
if (paramCount === undefined) {
|
||||||
|
debug('too many parameters')
|
||||||
|
throw createError(413, 'too many parameters', {
|
||||||
|
type: 'parameters.too.many'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
var arrayLimit = extended ? Math.max(100, paramCount) : paramCount
|
||||||
|
|
||||||
|
debug('parse ' + (extended ? 'extended ' : '') + 'urlencoding')
|
||||||
|
try {
|
||||||
|
return qs.parse(body, {
|
||||||
|
allowPrototypes: true,
|
||||||
|
arrayLimit: arrayLimit,
|
||||||
|
depth: depth,
|
||||||
|
charsetSentinel: charsetSentinel,
|
||||||
|
interpretNumericEntities: interpretNumericEntities,
|
||||||
|
charset: encoding,
|
||||||
|
parameterLimit: parameterLimit,
|
||||||
|
strictDepth: true
|
||||||
|
})
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof RangeError) {
|
||||||
|
throw createError(400, 'The input exceeded the depth', {
|
||||||
|
type: 'querystring.parse.rangeError'
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
throw err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Count the number of parameters, stopping once limit reached
|
||||||
|
*
|
||||||
|
* @param {string} body
|
||||||
|
* @param {number} limit
|
||||||
|
* @returns {number|undefined} Returns undefined if limit exceeded
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function parameterCount (body, limit) {
|
||||||
|
let count = 0
|
||||||
|
let index = -1
|
||||||
|
do {
|
||||||
|
count++
|
||||||
|
if (count > limit) return undefined // Early exit if limit exceeded
|
||||||
|
index = body.indexOf('&', index + 1)
|
||||||
|
} while (index !== -1)
|
||||||
|
return count
|
||||||
|
}
|
||||||
98
node_modules/body-parser/lib/utils.js
generated
vendored
Normal file
98
node_modules/body-parser/lib/utils.js
generated
vendored
Normal file
|
|
@ -0,0 +1,98 @@
|
||||||
|
'use strict'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module dependencies.
|
||||||
|
*/
|
||||||
|
|
||||||
|
var bytes = require('bytes')
|
||||||
|
var contentType = require('content-type')
|
||||||
|
var typeis = require('type-is')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
*/
|
||||||
|
module.exports = {
|
||||||
|
getCharset,
|
||||||
|
normalizeOptions,
|
||||||
|
passthrough
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the charset of a request.
|
||||||
|
*
|
||||||
|
* @param {Object} req
|
||||||
|
* @returns {string | undefined}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function getCharset (req) {
|
||||||
|
try {
|
||||||
|
return (contentType.parse(req).parameters.charset || '').toLowerCase()
|
||||||
|
} catch {
|
||||||
|
return undefined
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the simple type checker.
|
||||||
|
*
|
||||||
|
* @param {string | string[]} type
|
||||||
|
* @returns {Function}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function typeChecker (type) {
|
||||||
|
return function checkType (req) {
|
||||||
|
return Boolean(typeis(req, type))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalizes the common options for all parsers.
|
||||||
|
*
|
||||||
|
* @param {Object} options options to normalize
|
||||||
|
* @param {string | string[] | Function} defaultType default content type(s) or a function to determine it
|
||||||
|
* @returns {Object}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function normalizeOptions (options, defaultType) {
|
||||||
|
if (!defaultType) {
|
||||||
|
// Parsers must define a default content type
|
||||||
|
throw new TypeError('defaultType must be provided')
|
||||||
|
}
|
||||||
|
|
||||||
|
var inflate = options?.inflate !== false
|
||||||
|
var limit = typeof options?.limit !== 'number'
|
||||||
|
? bytes.parse(options?.limit || '100kb')
|
||||||
|
: options?.limit
|
||||||
|
var type = options?.type || defaultType
|
||||||
|
var verify = options?.verify || false
|
||||||
|
var defaultCharset = options?.defaultCharset || 'utf-8'
|
||||||
|
|
||||||
|
if (verify !== false && typeof verify !== 'function') {
|
||||||
|
throw new TypeError('option verify must be function')
|
||||||
|
}
|
||||||
|
|
||||||
|
// create the appropriate type checking function
|
||||||
|
var shouldParse = typeof type !== 'function'
|
||||||
|
? typeChecker(type)
|
||||||
|
: type
|
||||||
|
|
||||||
|
return {
|
||||||
|
inflate,
|
||||||
|
limit,
|
||||||
|
verify,
|
||||||
|
defaultCharset,
|
||||||
|
shouldParse
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Passthrough function that returns input unchanged.
|
||||||
|
* Used by parsers that don't need to transform the data.
|
||||||
|
*
|
||||||
|
* @param {*} value
|
||||||
|
* @returns {*}
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
function passthrough (value) {
|
||||||
|
return value
|
||||||
|
}
|
||||||
52
node_modules/body-parser/package.json
generated
vendored
Normal file
52
node_modules/body-parser/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,52 @@
|
||||||
|
{
|
||||||
|
"name": "body-parser",
|
||||||
|
"description": "Node.js body parsing middleware",
|
||||||
|
"version": "2.2.2",
|
||||||
|
"contributors": [
|
||||||
|
"Douglas Christopher Wilson <doug@somethingdoug.com>",
|
||||||
|
"Jonathan Ong <me@jongleberry.com> (http://jongleberry.com)"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"repository": "expressjs/body-parser",
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/express"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"bytes": "^3.1.2",
|
||||||
|
"content-type": "^1.0.5",
|
||||||
|
"debug": "^4.4.3",
|
||||||
|
"http-errors": "^2.0.0",
|
||||||
|
"iconv-lite": "^0.7.0",
|
||||||
|
"on-finished": "^2.4.1",
|
||||||
|
"qs": "^6.14.1",
|
||||||
|
"raw-body": "^3.0.1",
|
||||||
|
"type-is": "^2.0.1"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"eslint": "^8.57.1",
|
||||||
|
"eslint-config-standard": "^14.1.1",
|
||||||
|
"eslint-plugin-import": "^2.31.0",
|
||||||
|
"eslint-plugin-markdown": "^3.0.1",
|
||||||
|
"eslint-plugin-node": "^11.1.0",
|
||||||
|
"eslint-plugin-promise": "^6.6.0",
|
||||||
|
"eslint-plugin-standard": "^4.1.0",
|
||||||
|
"mocha": "^11.1.0",
|
||||||
|
"nyc": "^17.1.0",
|
||||||
|
"supertest": "^7.0.0"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"lib/",
|
||||||
|
"LICENSE",
|
||||||
|
"index.js"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"lint": "eslint .",
|
||||||
|
"test": "mocha --reporter spec --check-leaks test/",
|
||||||
|
"test-ci": "nyc --reporter=lcovonly --reporter=text npm test",
|
||||||
|
"test-cov": "nyc --reporter=html --reporter=text npm test"
|
||||||
|
}
|
||||||
|
}
|
||||||
21
node_modules/buffer-from/LICENSE
generated
vendored
Normal file
21
node_modules/buffer-from/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,21 @@
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2016, 2018 Linus Unnebäck
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
72
node_modules/buffer-from/index.js
generated
vendored
Normal file
72
node_modules/buffer-from/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1,72 @@
|
||||||
|
/* eslint-disable node/no-deprecated-api */
|
||||||
|
|
||||||
|
var toString = Object.prototype.toString
|
||||||
|
|
||||||
|
var isModern = (
|
||||||
|
typeof Buffer !== 'undefined' &&
|
||||||
|
typeof Buffer.alloc === 'function' &&
|
||||||
|
typeof Buffer.allocUnsafe === 'function' &&
|
||||||
|
typeof Buffer.from === 'function'
|
||||||
|
)
|
||||||
|
|
||||||
|
function isArrayBuffer (input) {
|
||||||
|
return toString.call(input).slice(8, -1) === 'ArrayBuffer'
|
||||||
|
}
|
||||||
|
|
||||||
|
function fromArrayBuffer (obj, byteOffset, length) {
|
||||||
|
byteOffset >>>= 0
|
||||||
|
|
||||||
|
var maxLength = obj.byteLength - byteOffset
|
||||||
|
|
||||||
|
if (maxLength < 0) {
|
||||||
|
throw new RangeError("'offset' is out of bounds")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length === undefined) {
|
||||||
|
length = maxLength
|
||||||
|
} else {
|
||||||
|
length >>>= 0
|
||||||
|
|
||||||
|
if (length > maxLength) {
|
||||||
|
throw new RangeError("'length' is out of bounds")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isModern
|
||||||
|
? Buffer.from(obj.slice(byteOffset, byteOffset + length))
|
||||||
|
: new Buffer(new Uint8Array(obj.slice(byteOffset, byteOffset + length)))
|
||||||
|
}
|
||||||
|
|
||||||
|
function fromString (string, encoding) {
|
||||||
|
if (typeof encoding !== 'string' || encoding === '') {
|
||||||
|
encoding = 'utf8'
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Buffer.isEncoding(encoding)) {
|
||||||
|
throw new TypeError('"encoding" must be a valid string encoding')
|
||||||
|
}
|
||||||
|
|
||||||
|
return isModern
|
||||||
|
? Buffer.from(string, encoding)
|
||||||
|
: new Buffer(string, encoding)
|
||||||
|
}
|
||||||
|
|
||||||
|
function bufferFrom (value, encodingOrOffset, length) {
|
||||||
|
if (typeof value === 'number') {
|
||||||
|
throw new TypeError('"value" argument must not be a number')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isArrayBuffer(value)) {
|
||||||
|
return fromArrayBuffer(value, encodingOrOffset, length)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === 'string') {
|
||||||
|
return fromString(value, encodingOrOffset)
|
||||||
|
}
|
||||||
|
|
||||||
|
return isModern
|
||||||
|
? Buffer.from(value)
|
||||||
|
: new Buffer(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = bufferFrom
|
||||||
19
node_modules/buffer-from/package.json
generated
vendored
Normal file
19
node_modules/buffer-from/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
{
|
||||||
|
"name": "buffer-from",
|
||||||
|
"version": "1.1.2",
|
||||||
|
"license": "MIT",
|
||||||
|
"repository": "LinusU/buffer-from",
|
||||||
|
"files": [
|
||||||
|
"index.js"
|
||||||
|
],
|
||||||
|
"scripts": {
|
||||||
|
"test": "standard && node test"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"standard": "^12.0.1"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"buffer",
|
||||||
|
"buffer from"
|
||||||
|
]
|
||||||
|
}
|
||||||
69
node_modules/buffer-from/readme.md
generated
vendored
Normal file
69
node_modules/buffer-from/readme.md
generated
vendored
Normal file
|
|
@ -0,0 +1,69 @@
|
||||||
|
# Buffer From
|
||||||
|
|
||||||
|
A [ponyfill](https://ponyfill.com) for `Buffer.from`, uses native implementation if available.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm install --save buffer-from
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
```js
|
||||||
|
const bufferFrom = require('buffer-from')
|
||||||
|
|
||||||
|
console.log(bufferFrom([1, 2, 3, 4]))
|
||||||
|
//=> <Buffer 01 02 03 04>
|
||||||
|
|
||||||
|
const arr = new Uint8Array([1, 2, 3, 4])
|
||||||
|
console.log(bufferFrom(arr.buffer, 1, 2))
|
||||||
|
//=> <Buffer 02 03>
|
||||||
|
|
||||||
|
console.log(bufferFrom('test', 'utf8'))
|
||||||
|
//=> <Buffer 74 65 73 74>
|
||||||
|
|
||||||
|
const buf = bufferFrom('test')
|
||||||
|
console.log(bufferFrom(buf))
|
||||||
|
//=> <Buffer 74 65 73 74>
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### bufferFrom(array)
|
||||||
|
|
||||||
|
- `array` <Array>
|
||||||
|
|
||||||
|
Allocates a new `Buffer` using an `array` of octets.
|
||||||
|
|
||||||
|
### bufferFrom(arrayBuffer[, byteOffset[, length]])
|
||||||
|
|
||||||
|
- `arrayBuffer` <ArrayBuffer> The `.buffer` property of a TypedArray or ArrayBuffer
|
||||||
|
- `byteOffset` <Integer> Where to start copying from `arrayBuffer`. **Default:** `0`
|
||||||
|
- `length` <Integer> How many bytes to copy from `arrayBuffer`. **Default:** `arrayBuffer.length - byteOffset`
|
||||||
|
|
||||||
|
When passed a reference to the `.buffer` property of a TypedArray instance, the
|
||||||
|
newly created `Buffer` will share the same allocated memory as the TypedArray.
|
||||||
|
|
||||||
|
The optional `byteOffset` and `length` arguments specify a memory range within
|
||||||
|
the `arrayBuffer` that will be shared by the `Buffer`.
|
||||||
|
|
||||||
|
### bufferFrom(buffer)
|
||||||
|
|
||||||
|
- `buffer` <Buffer> An existing `Buffer` to copy data from
|
||||||
|
|
||||||
|
Copies the passed `buffer` data onto a new `Buffer` instance.
|
||||||
|
|
||||||
|
### bufferFrom(string[, encoding])
|
||||||
|
|
||||||
|
- `string` <String> A string to encode.
|
||||||
|
- `encoding` <String> The encoding of `string`. **Default:** `'utf8'`
|
||||||
|
|
||||||
|
Creates a new `Buffer` containing the given JavaScript string `string`. If
|
||||||
|
provided, the `encoding` parameter identifies the character encoding of
|
||||||
|
`string`.
|
||||||
|
|
||||||
|
## See also
|
||||||
|
|
||||||
|
- [buffer-alloc](https://github.com/LinusU/buffer-alloc) A ponyfill for `Buffer.alloc`
|
||||||
|
- [buffer-alloc-unsafe](https://github.com/LinusU/buffer-alloc-unsafe) A ponyfill for `Buffer.allocUnsafe`
|
||||||
5
node_modules/busboy/.eslintrc.js
generated
vendored
Normal file
5
node_modules/busboy/.eslintrc.js
generated
vendored
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
extends: '@mscdex/eslint-config',
|
||||||
|
};
|
||||||
24
node_modules/busboy/.github/workflows/ci.yml
generated
vendored
Normal file
24
node_modules/busboy/.github/workflows/ci.yml
generated
vendored
Normal file
|
|
@ -0,0 +1,24 @@
|
||||||
|
name: CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
push:
|
||||||
|
branches: [ master ]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
tests-linux:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
node-version: [10.16.0, 10.x, 12.x, 14.x, 16.x]
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Use Node.js ${{ matrix.node-version }}
|
||||||
|
uses: actions/setup-node@v1
|
||||||
|
with:
|
||||||
|
node-version: ${{ matrix.node-version }}
|
||||||
|
- name: Install module
|
||||||
|
run: npm install
|
||||||
|
- name: Run tests
|
||||||
|
run: npm test
|
||||||
23
node_modules/busboy/.github/workflows/lint.yml
generated
vendored
Normal file
23
node_modules/busboy/.github/workflows/lint.yml
generated
vendored
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
name: lint
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
push:
|
||||||
|
branches: [ master ]
|
||||||
|
|
||||||
|
env:
|
||||||
|
NODE_VERSION: 16.x
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
lint-js:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Use Node.js ${{ env.NODE_VERSION }}
|
||||||
|
uses: actions/setup-node@v1
|
||||||
|
with:
|
||||||
|
node-version: ${{ env.NODE_VERSION }}
|
||||||
|
- name: Install ESLint + ESLint configs/plugins
|
||||||
|
run: npm install --only=dev
|
||||||
|
- name: Lint files
|
||||||
|
run: npm run lint
|
||||||
19
node_modules/busboy/LICENSE
generated
vendored
Normal file
19
node_modules/busboy/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
Copyright Brian White. All rights reserved.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to
|
||||||
|
deal in the Software without restriction, including without limitation the
|
||||||
|
rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
||||||
|
sell copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
||||||
|
IN THE SOFTWARE.
|
||||||
191
node_modules/busboy/README.md
generated
vendored
Normal file
191
node_modules/busboy/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,191 @@
|
||||||
|
# Description
|
||||||
|
|
||||||
|
A node.js module for parsing incoming HTML form data.
|
||||||
|
|
||||||
|
Changes (breaking or otherwise) in v1.0.0 can be found [here](https://github.com/mscdex/busboy/issues/266).
|
||||||
|
|
||||||
|
# Requirements
|
||||||
|
|
||||||
|
* [node.js](http://nodejs.org/) -- v10.16.0 or newer
|
||||||
|
|
||||||
|
|
||||||
|
# Install
|
||||||
|
|
||||||
|
npm install busboy
|
||||||
|
|
||||||
|
|
||||||
|
# Examples
|
||||||
|
|
||||||
|
* Parsing (multipart) with default options:
|
||||||
|
|
||||||
|
```js
|
||||||
|
const http = require('http');
|
||||||
|
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
http.createServer((req, res) => {
|
||||||
|
if (req.method === 'POST') {
|
||||||
|
console.log('POST request');
|
||||||
|
const bb = busboy({ headers: req.headers });
|
||||||
|
bb.on('file', (name, file, info) => {
|
||||||
|
const { filename, encoding, mimeType } = info;
|
||||||
|
console.log(
|
||||||
|
`File [${name}]: filename: %j, encoding: %j, mimeType: %j`,
|
||||||
|
filename,
|
||||||
|
encoding,
|
||||||
|
mimeType
|
||||||
|
);
|
||||||
|
file.on('data', (data) => {
|
||||||
|
console.log(`File [${name}] got ${data.length} bytes`);
|
||||||
|
}).on('close', () => {
|
||||||
|
console.log(`File [${name}] done`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
bb.on('field', (name, val, info) => {
|
||||||
|
console.log(`Field [${name}]: value: %j`, val);
|
||||||
|
});
|
||||||
|
bb.on('close', () => {
|
||||||
|
console.log('Done parsing form!');
|
||||||
|
res.writeHead(303, { Connection: 'close', Location: '/' });
|
||||||
|
res.end();
|
||||||
|
});
|
||||||
|
req.pipe(bb);
|
||||||
|
} else if (req.method === 'GET') {
|
||||||
|
res.writeHead(200, { Connection: 'close' });
|
||||||
|
res.end(`
|
||||||
|
<html>
|
||||||
|
<head></head>
|
||||||
|
<body>
|
||||||
|
<form method="POST" enctype="multipart/form-data">
|
||||||
|
<input type="file" name="filefield"><br />
|
||||||
|
<input type="text" name="textfield"><br />
|
||||||
|
<input type="submit">
|
||||||
|
</form>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`);
|
||||||
|
}
|
||||||
|
}).listen(8000, () => {
|
||||||
|
console.log('Listening for requests');
|
||||||
|
});
|
||||||
|
|
||||||
|
// Example output:
|
||||||
|
//
|
||||||
|
// Listening for requests
|
||||||
|
// < ... form submitted ... >
|
||||||
|
// POST request
|
||||||
|
// File [filefield]: filename: "logo.jpg", encoding: "binary", mime: "image/jpeg"
|
||||||
|
// File [filefield] got 11912 bytes
|
||||||
|
// Field [textfield]: value: "testing! :-)"
|
||||||
|
// File [filefield] done
|
||||||
|
// Done parsing form!
|
||||||
|
```
|
||||||
|
|
||||||
|
* Save all incoming files to disk:
|
||||||
|
|
||||||
|
```js
|
||||||
|
const { randomFillSync } = require('crypto');
|
||||||
|
const fs = require('fs');
|
||||||
|
const http = require('http');
|
||||||
|
const os = require('os');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
const random = (() => {
|
||||||
|
const buf = Buffer.alloc(16);
|
||||||
|
return () => randomFillSync(buf).toString('hex');
|
||||||
|
})();
|
||||||
|
|
||||||
|
http.createServer((req, res) => {
|
||||||
|
if (req.method === 'POST') {
|
||||||
|
const bb = busboy({ headers: req.headers });
|
||||||
|
bb.on('file', (name, file, info) => {
|
||||||
|
const saveTo = path.join(os.tmpdir(), `busboy-upload-${random()}`);
|
||||||
|
file.pipe(fs.createWriteStream(saveTo));
|
||||||
|
});
|
||||||
|
bb.on('close', () => {
|
||||||
|
res.writeHead(200, { 'Connection': 'close' });
|
||||||
|
res.end(`That's all folks!`);
|
||||||
|
});
|
||||||
|
req.pipe(bb);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
res.writeHead(404);
|
||||||
|
res.end();
|
||||||
|
}).listen(8000, () => {
|
||||||
|
console.log('Listening for requests');
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
# API
|
||||||
|
|
||||||
|
## Exports
|
||||||
|
|
||||||
|
`busboy` exports a single function:
|
||||||
|
|
||||||
|
**( _function_ )**(< _object_ >config) - Creates and returns a new _Writable_ form parser stream.
|
||||||
|
|
||||||
|
* Valid `config` properties:
|
||||||
|
|
||||||
|
* **headers** - _object_ - These are the HTTP headers of the incoming request, which are used by individual parsers.
|
||||||
|
|
||||||
|
* **highWaterMark** - _integer_ - highWaterMark to use for the parser stream. **Default:** node's _stream.Writable_ default.
|
||||||
|
|
||||||
|
* **fileHwm** - _integer_ - highWaterMark to use for individual file streams. **Default:** node's _stream.Readable_ default.
|
||||||
|
|
||||||
|
* **defCharset** - _string_ - Default character set to use when one isn't defined. **Default:** `'utf8'`.
|
||||||
|
|
||||||
|
* **defParamCharset** - _string_ - For multipart forms, the default character set to use for values of part header parameters (e.g. filename) that are not extended parameters (that contain an explicit charset). **Default:** `'latin1'`.
|
||||||
|
|
||||||
|
* **preservePath** - _boolean_ - If paths in filenames from file parts in a `'multipart/form-data'` request shall be preserved. **Default:** `false`.
|
||||||
|
|
||||||
|
* **limits** - _object_ - Various limits on incoming data. Valid properties are:
|
||||||
|
|
||||||
|
* **fieldNameSize** - _integer_ - Max field name size (in bytes). **Default:** `100`.
|
||||||
|
|
||||||
|
* **fieldSize** - _integer_ - Max field value size (in bytes). **Default:** `1048576` (1MB).
|
||||||
|
|
||||||
|
* **fields** - _integer_ - Max number of non-file fields. **Default:** `Infinity`.
|
||||||
|
|
||||||
|
* **fileSize** - _integer_ - For multipart forms, the max file size (in bytes). **Default:** `Infinity`.
|
||||||
|
|
||||||
|
* **files** - _integer_ - For multipart forms, the max number of file fields. **Default:** `Infinity`.
|
||||||
|
|
||||||
|
* **parts** - _integer_ - For multipart forms, the max number of parts (fields + files). **Default:** `Infinity`.
|
||||||
|
|
||||||
|
* **headerPairs** - _integer_ - For multipart forms, the max number of header key-value pairs to parse. **Default:** `2000` (same as node's http module).
|
||||||
|
|
||||||
|
This function can throw exceptions if there is something wrong with the values in `config`. For example, if the Content-Type in `headers` is missing entirely, is not a supported type, or is missing the boundary for `'multipart/form-data'` requests.
|
||||||
|
|
||||||
|
## (Special) Parser stream events
|
||||||
|
|
||||||
|
* **file**(< _string_ >name, < _Readable_ >stream, < _object_ >info) - Emitted for each new file found. `name` contains the form field name. `stream` is a _Readable_ stream containing the file's data. No transformations/conversions (e.g. base64 to raw binary) are done on the file's data. `info` contains the following properties:
|
||||||
|
|
||||||
|
* `filename` - _string_ - If supplied, this contains the file's filename. **WARNING:** You should almost _never_ use this value as-is (especially if you are using `preservePath: true` in your `config`) as it could contain malicious input. You are better off generating your own (safe) filenames, or at the very least using a hash of the filename.
|
||||||
|
|
||||||
|
* `encoding` - _string_ - The file's `'Content-Transfer-Encoding'` value.
|
||||||
|
|
||||||
|
* `mimeType` - _string_ - The file's `'Content-Type'` value.
|
||||||
|
|
||||||
|
**Note:** If you listen for this event, you should always consume the `stream` whether you care about its contents or not (you can simply do `stream.resume();` if you want to discard/skip the contents), otherwise the `'finish'`/`'close'` event will never fire on the busboy parser stream.
|
||||||
|
However, if you aren't accepting files, you can either simply not listen for the `'file'` event at all or set `limits.files` to `0`, and any/all files will be automatically skipped (these skipped files will still count towards any configured `limits.files` and `limits.parts` limits though).
|
||||||
|
|
||||||
|
**Note:** If a configured `limits.fileSize` limit was reached for a file, `stream` will both have a boolean property `truncated` set to `true` (best checked at the end of the stream) and emit a `'limit'` event to notify you when this happens.
|
||||||
|
|
||||||
|
* **field**(< _string_ >name, < _string_ >value, < _object_ >info) - Emitted for each new non-file field found. `name` contains the form field name. `value` contains the string value of the field. `info` contains the following properties:
|
||||||
|
|
||||||
|
* `nameTruncated` - _boolean_ - Whether `name` was truncated or not (due to a configured `limits.fieldNameSize` limit)
|
||||||
|
|
||||||
|
* `valueTruncated` - _boolean_ - Whether `value` was truncated or not (due to a configured `limits.fieldSize` limit)
|
||||||
|
|
||||||
|
* `encoding` - _string_ - The field's `'Content-Transfer-Encoding'` value.
|
||||||
|
|
||||||
|
* `mimeType` - _string_ - The field's `'Content-Type'` value.
|
||||||
|
|
||||||
|
* **partsLimit**() - Emitted when the configured `limits.parts` limit has been reached. No more `'file'` or `'field'` events will be emitted.
|
||||||
|
|
||||||
|
* **filesLimit**() - Emitted when the configured `limits.files` limit has been reached. No more `'file'` events will be emitted.
|
||||||
|
|
||||||
|
* **fieldsLimit**() - Emitted when the configured `limits.fields` limit has been reached. No more `'field'` events will be emitted.
|
||||||
149
node_modules/busboy/bench/bench-multipart-fields-100mb-big.js
generated
vendored
Normal file
149
node_modules/busboy/bench/bench-multipart-fields-100mb-big.js
generated
vendored
Normal file
|
|
@ -0,0 +1,149 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
function createMultipartBuffers(boundary, sizes) {
|
||||||
|
const bufs = [];
|
||||||
|
for (let i = 0; i < sizes.length; ++i) {
|
||||||
|
const mb = sizes[i] * 1024 * 1024;
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}`,
|
||||||
|
`content-disposition: form-data; name="field${i + 1}"`,
|
||||||
|
'',
|
||||||
|
'0'.repeat(mb),
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
}
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}--`,
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
return bufs;
|
||||||
|
}
|
||||||
|
|
||||||
|
const boundary = '-----------------------------168072824752491622650073';
|
||||||
|
const buffers = createMultipartBuffers(boundary, [
|
||||||
|
10,
|
||||||
|
10,
|
||||||
|
10,
|
||||||
|
20,
|
||||||
|
50,
|
||||||
|
]);
|
||||||
|
const calls = {
|
||||||
|
partBegin: 0,
|
||||||
|
headerField: 0,
|
||||||
|
headerValue: 0,
|
||||||
|
headerEnd: 0,
|
||||||
|
headersEnd: 0,
|
||||||
|
partData: 0,
|
||||||
|
partEnd: 0,
|
||||||
|
end: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
const moduleName = process.argv[2];
|
||||||
|
switch (moduleName) {
|
||||||
|
case 'busboy': {
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
const parser = busboy({
|
||||||
|
limits: {
|
||||||
|
fieldSizeLimit: Infinity,
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
parser.on('field', (name, val, info) => {
|
||||||
|
++calls.partBegin;
|
||||||
|
++calls.partData;
|
||||||
|
++calls.partEnd;
|
||||||
|
}).on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable': {
|
||||||
|
const { MultipartParser } = require('formidable');
|
||||||
|
|
||||||
|
const parser = new MultipartParser();
|
||||||
|
parser.initWithBoundary(boundary);
|
||||||
|
parser.on('data', ({ name }) => {
|
||||||
|
++calls[name];
|
||||||
|
if (name === 'end')
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'multiparty': {
|
||||||
|
const { Readable } = require('stream');
|
||||||
|
|
||||||
|
const { Form } = require('multiparty');
|
||||||
|
|
||||||
|
const form = new Form({
|
||||||
|
maxFieldsSize: Infinity,
|
||||||
|
maxFields: Infinity,
|
||||||
|
maxFilesSize: Infinity,
|
||||||
|
autoFields: false,
|
||||||
|
autoFiles: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
const req = new Readable({ read: () => {} });
|
||||||
|
req.headers = {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
function hijack(name, fn) {
|
||||||
|
const oldFn = form[name];
|
||||||
|
form[name] = function() {
|
||||||
|
fn();
|
||||||
|
return oldFn.apply(this, arguments);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
hijack('onParseHeaderField', () => {
|
||||||
|
++calls.headerField;
|
||||||
|
});
|
||||||
|
hijack('onParseHeaderValue', () => {
|
||||||
|
++calls.headerValue;
|
||||||
|
});
|
||||||
|
hijack('onParsePartBegin', () => {
|
||||||
|
++calls.partBegin;
|
||||||
|
});
|
||||||
|
hijack('onParsePartData', () => {
|
||||||
|
++calls.partData;
|
||||||
|
});
|
||||||
|
hijack('onParsePartEnd', () => {
|
||||||
|
++calls.partEnd;
|
||||||
|
});
|
||||||
|
|
||||||
|
form.on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
}).on('part', (p) => p.resume());
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
form.parse(req);
|
||||||
|
for (const buf of buffers)
|
||||||
|
req.push(buf);
|
||||||
|
req.push(null);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (moduleName === undefined)
|
||||||
|
console.error('Missing parser module name');
|
||||||
|
else
|
||||||
|
console.error(`Invalid parser module name: ${moduleName}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
143
node_modules/busboy/bench/bench-multipart-fields-100mb-small.js
generated
vendored
Normal file
143
node_modules/busboy/bench/bench-multipart-fields-100mb-small.js
generated
vendored
Normal file
|
|
@ -0,0 +1,143 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
function createMultipartBuffers(boundary, sizes) {
|
||||||
|
const bufs = [];
|
||||||
|
for (let i = 0; i < sizes.length; ++i) {
|
||||||
|
const mb = sizes[i] * 1024 * 1024;
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}`,
|
||||||
|
`content-disposition: form-data; name="field${i + 1}"`,
|
||||||
|
'',
|
||||||
|
'0'.repeat(mb),
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
}
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}--`,
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
return bufs;
|
||||||
|
}
|
||||||
|
|
||||||
|
const boundary = '-----------------------------168072824752491622650073';
|
||||||
|
const buffers = createMultipartBuffers(boundary, (new Array(100)).fill(1));
|
||||||
|
const calls = {
|
||||||
|
partBegin: 0,
|
||||||
|
headerField: 0,
|
||||||
|
headerValue: 0,
|
||||||
|
headerEnd: 0,
|
||||||
|
headersEnd: 0,
|
||||||
|
partData: 0,
|
||||||
|
partEnd: 0,
|
||||||
|
end: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
const moduleName = process.argv[2];
|
||||||
|
switch (moduleName) {
|
||||||
|
case 'busboy': {
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
const parser = busboy({
|
||||||
|
limits: {
|
||||||
|
fieldSizeLimit: Infinity,
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
parser.on('field', (name, val, info) => {
|
||||||
|
++calls.partBegin;
|
||||||
|
++calls.partData;
|
||||||
|
++calls.partEnd;
|
||||||
|
}).on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable': {
|
||||||
|
const { MultipartParser } = require('formidable');
|
||||||
|
|
||||||
|
const parser = new MultipartParser();
|
||||||
|
parser.initWithBoundary(boundary);
|
||||||
|
parser.on('data', ({ name }) => {
|
||||||
|
++calls[name];
|
||||||
|
if (name === 'end')
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'multiparty': {
|
||||||
|
const { Readable } = require('stream');
|
||||||
|
|
||||||
|
const { Form } = require('multiparty');
|
||||||
|
|
||||||
|
const form = new Form({
|
||||||
|
maxFieldsSize: Infinity,
|
||||||
|
maxFields: Infinity,
|
||||||
|
maxFilesSize: Infinity,
|
||||||
|
autoFields: false,
|
||||||
|
autoFiles: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
const req = new Readable({ read: () => {} });
|
||||||
|
req.headers = {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
function hijack(name, fn) {
|
||||||
|
const oldFn = form[name];
|
||||||
|
form[name] = function() {
|
||||||
|
fn();
|
||||||
|
return oldFn.apply(this, arguments);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
hijack('onParseHeaderField', () => {
|
||||||
|
++calls.headerField;
|
||||||
|
});
|
||||||
|
hijack('onParseHeaderValue', () => {
|
||||||
|
++calls.headerValue;
|
||||||
|
});
|
||||||
|
hijack('onParsePartBegin', () => {
|
||||||
|
++calls.partBegin;
|
||||||
|
});
|
||||||
|
hijack('onParsePartData', () => {
|
||||||
|
++calls.partData;
|
||||||
|
});
|
||||||
|
hijack('onParsePartEnd', () => {
|
||||||
|
++calls.partEnd;
|
||||||
|
});
|
||||||
|
|
||||||
|
form.on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
}).on('part', (p) => p.resume());
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
form.parse(req);
|
||||||
|
for (const buf of buffers)
|
||||||
|
req.push(buf);
|
||||||
|
req.push(null);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (moduleName === undefined)
|
||||||
|
console.error('Missing parser module name');
|
||||||
|
else
|
||||||
|
console.error(`Invalid parser module name: ${moduleName}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
154
node_modules/busboy/bench/bench-multipart-files-100mb-big.js
generated
vendored
Normal file
154
node_modules/busboy/bench/bench-multipart-files-100mb-big.js
generated
vendored
Normal file
|
|
@ -0,0 +1,154 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
function createMultipartBuffers(boundary, sizes) {
|
||||||
|
const bufs = [];
|
||||||
|
for (let i = 0; i < sizes.length; ++i) {
|
||||||
|
const mb = sizes[i] * 1024 * 1024;
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}`,
|
||||||
|
`content-disposition: form-data; name="file${i + 1}"; `
|
||||||
|
+ `filename="random${i + 1}.bin"`,
|
||||||
|
'content-type: application/octet-stream',
|
||||||
|
'',
|
||||||
|
'0'.repeat(mb),
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
}
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}--`,
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
return bufs;
|
||||||
|
}
|
||||||
|
|
||||||
|
const boundary = '-----------------------------168072824752491622650073';
|
||||||
|
const buffers = createMultipartBuffers(boundary, [
|
||||||
|
10,
|
||||||
|
10,
|
||||||
|
10,
|
||||||
|
20,
|
||||||
|
50,
|
||||||
|
]);
|
||||||
|
const calls = {
|
||||||
|
partBegin: 0,
|
||||||
|
headerField: 0,
|
||||||
|
headerValue: 0,
|
||||||
|
headerEnd: 0,
|
||||||
|
headersEnd: 0,
|
||||||
|
partData: 0,
|
||||||
|
partEnd: 0,
|
||||||
|
end: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
const moduleName = process.argv[2];
|
||||||
|
switch (moduleName) {
|
||||||
|
case 'busboy': {
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
const parser = busboy({
|
||||||
|
limits: {
|
||||||
|
fieldSizeLimit: Infinity,
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
parser.on('file', (name, stream, info) => {
|
||||||
|
++calls.partBegin;
|
||||||
|
stream.on('data', (chunk) => {
|
||||||
|
++calls.partData;
|
||||||
|
}).on('end', () => {
|
||||||
|
++calls.partEnd;
|
||||||
|
});
|
||||||
|
}).on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable': {
|
||||||
|
const { MultipartParser } = require('formidable');
|
||||||
|
|
||||||
|
const parser = new MultipartParser();
|
||||||
|
parser.initWithBoundary(boundary);
|
||||||
|
parser.on('data', ({ name }) => {
|
||||||
|
++calls[name];
|
||||||
|
if (name === 'end')
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'multiparty': {
|
||||||
|
const { Readable } = require('stream');
|
||||||
|
|
||||||
|
const { Form } = require('multiparty');
|
||||||
|
|
||||||
|
const form = new Form({
|
||||||
|
maxFieldsSize: Infinity,
|
||||||
|
maxFields: Infinity,
|
||||||
|
maxFilesSize: Infinity,
|
||||||
|
autoFields: false,
|
||||||
|
autoFiles: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
const req = new Readable({ read: () => {} });
|
||||||
|
req.headers = {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
function hijack(name, fn) {
|
||||||
|
const oldFn = form[name];
|
||||||
|
form[name] = function() {
|
||||||
|
fn();
|
||||||
|
return oldFn.apply(this, arguments);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
hijack('onParseHeaderField', () => {
|
||||||
|
++calls.headerField;
|
||||||
|
});
|
||||||
|
hijack('onParseHeaderValue', () => {
|
||||||
|
++calls.headerValue;
|
||||||
|
});
|
||||||
|
hijack('onParsePartBegin', () => {
|
||||||
|
++calls.partBegin;
|
||||||
|
});
|
||||||
|
hijack('onParsePartData', () => {
|
||||||
|
++calls.partData;
|
||||||
|
});
|
||||||
|
hijack('onParsePartEnd', () => {
|
||||||
|
++calls.partEnd;
|
||||||
|
});
|
||||||
|
|
||||||
|
form.on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
}).on('part', (p) => p.resume());
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
form.parse(req);
|
||||||
|
for (const buf of buffers)
|
||||||
|
req.push(buf);
|
||||||
|
req.push(null);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (moduleName === undefined)
|
||||||
|
console.error('Missing parser module name');
|
||||||
|
else
|
||||||
|
console.error(`Invalid parser module name: ${moduleName}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
148
node_modules/busboy/bench/bench-multipart-files-100mb-small.js
generated
vendored
Normal file
148
node_modules/busboy/bench/bench-multipart-files-100mb-small.js
generated
vendored
Normal file
|
|
@ -0,0 +1,148 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
function createMultipartBuffers(boundary, sizes) {
|
||||||
|
const bufs = [];
|
||||||
|
for (let i = 0; i < sizes.length; ++i) {
|
||||||
|
const mb = sizes[i] * 1024 * 1024;
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}`,
|
||||||
|
`content-disposition: form-data; name="file${i + 1}"; `
|
||||||
|
+ `filename="random${i + 1}.bin"`,
|
||||||
|
'content-type: application/octet-stream',
|
||||||
|
'',
|
||||||
|
'0'.repeat(mb),
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
}
|
||||||
|
bufs.push(Buffer.from([
|
||||||
|
`--${boundary}--`,
|
||||||
|
'',
|
||||||
|
].join('\r\n')));
|
||||||
|
return bufs;
|
||||||
|
}
|
||||||
|
|
||||||
|
const boundary = '-----------------------------168072824752491622650073';
|
||||||
|
const buffers = createMultipartBuffers(boundary, (new Array(100)).fill(1));
|
||||||
|
const calls = {
|
||||||
|
partBegin: 0,
|
||||||
|
headerField: 0,
|
||||||
|
headerValue: 0,
|
||||||
|
headerEnd: 0,
|
||||||
|
headersEnd: 0,
|
||||||
|
partData: 0,
|
||||||
|
partEnd: 0,
|
||||||
|
end: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
const moduleName = process.argv[2];
|
||||||
|
switch (moduleName) {
|
||||||
|
case 'busboy': {
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
const parser = busboy({
|
||||||
|
limits: {
|
||||||
|
fieldSizeLimit: Infinity,
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
parser.on('file', (name, stream, info) => {
|
||||||
|
++calls.partBegin;
|
||||||
|
stream.on('data', (chunk) => {
|
||||||
|
++calls.partData;
|
||||||
|
}).on('end', () => {
|
||||||
|
++calls.partEnd;
|
||||||
|
});
|
||||||
|
}).on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable': {
|
||||||
|
const { MultipartParser } = require('formidable');
|
||||||
|
|
||||||
|
const parser = new MultipartParser();
|
||||||
|
parser.initWithBoundary(boundary);
|
||||||
|
parser.on('data', ({ name }) => {
|
||||||
|
++calls[name];
|
||||||
|
if (name === 'end')
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'multiparty': {
|
||||||
|
const { Readable } = require('stream');
|
||||||
|
|
||||||
|
const { Form } = require('multiparty');
|
||||||
|
|
||||||
|
const form = new Form({
|
||||||
|
maxFieldsSize: Infinity,
|
||||||
|
maxFields: Infinity,
|
||||||
|
maxFilesSize: Infinity,
|
||||||
|
autoFields: false,
|
||||||
|
autoFiles: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
const req = new Readable({ read: () => {} });
|
||||||
|
req.headers = {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
function hijack(name, fn) {
|
||||||
|
const oldFn = form[name];
|
||||||
|
form[name] = function() {
|
||||||
|
fn();
|
||||||
|
return oldFn.apply(this, arguments);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
hijack('onParseHeaderField', () => {
|
||||||
|
++calls.headerField;
|
||||||
|
});
|
||||||
|
hijack('onParseHeaderValue', () => {
|
||||||
|
++calls.headerValue;
|
||||||
|
});
|
||||||
|
hijack('onParsePartBegin', () => {
|
||||||
|
++calls.partBegin;
|
||||||
|
});
|
||||||
|
hijack('onParsePartData', () => {
|
||||||
|
++calls.partData;
|
||||||
|
});
|
||||||
|
hijack('onParsePartEnd', () => {
|
||||||
|
++calls.partEnd;
|
||||||
|
});
|
||||||
|
|
||||||
|
form.on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
}).on('part', (p) => p.resume());
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
form.parse(req);
|
||||||
|
for (const buf of buffers)
|
||||||
|
req.push(buf);
|
||||||
|
req.push(null);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (moduleName === undefined)
|
||||||
|
console.error('Missing parser module name');
|
||||||
|
else
|
||||||
|
console.error(`Invalid parser module name: ${moduleName}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
101
node_modules/busboy/bench/bench-urlencoded-fields-100pairs-small.js
generated
vendored
Normal file
101
node_modules/busboy/bench/bench-urlencoded-fields-100pairs-small.js
generated
vendored
Normal file
|
|
@ -0,0 +1,101 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const buffers = [
|
||||||
|
Buffer.from(
|
||||||
|
(new Array(100)).fill('').map((_, i) => `key${i}=value${i}`).join('&')
|
||||||
|
),
|
||||||
|
];
|
||||||
|
const calls = {
|
||||||
|
field: 0,
|
||||||
|
end: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
let n = 3e3;
|
||||||
|
|
||||||
|
const moduleName = process.argv[2];
|
||||||
|
switch (moduleName) {
|
||||||
|
case 'busboy': {
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
(function next() {
|
||||||
|
const parser = busboy({
|
||||||
|
limits: {
|
||||||
|
fieldSizeLimit: Infinity,
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'content-type': 'application/x-www-form-urlencoded; charset=utf-8',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
parser.on('field', (name, val, info) => {
|
||||||
|
++calls.field;
|
||||||
|
}).on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
if (--n === 0)
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
else
|
||||||
|
process.nextTick(next);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
parser.end();
|
||||||
|
})();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable': {
|
||||||
|
const QuerystringParser =
|
||||||
|
require('formidable/src/parsers/Querystring.js');
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
(function next() {
|
||||||
|
const parser = new QuerystringParser();
|
||||||
|
parser.on('data', (obj) => {
|
||||||
|
++calls.field;
|
||||||
|
}).on('end', () => {
|
||||||
|
++calls.end;
|
||||||
|
if (--n === 0)
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
else
|
||||||
|
process.nextTick(next);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
parser.end();
|
||||||
|
})();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable-streaming': {
|
||||||
|
const QuerystringParser =
|
||||||
|
require('formidable/src/parsers/StreamingQuerystring.js');
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
(function next() {
|
||||||
|
const parser = new QuerystringParser();
|
||||||
|
parser.on('data', (obj) => {
|
||||||
|
++calls.field;
|
||||||
|
}).on('end', () => {
|
||||||
|
++calls.end;
|
||||||
|
if (--n === 0)
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
else
|
||||||
|
process.nextTick(next);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
parser.end();
|
||||||
|
})();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (moduleName === undefined)
|
||||||
|
console.error('Missing parser module name');
|
||||||
|
else
|
||||||
|
console.error(`Invalid parser module name: ${moduleName}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
84
node_modules/busboy/bench/bench-urlencoded-fields-900pairs-small-alt.js
generated
vendored
Normal file
84
node_modules/busboy/bench/bench-urlencoded-fields-900pairs-small-alt.js
generated
vendored
Normal file
|
|
@ -0,0 +1,84 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const buffers = [
|
||||||
|
Buffer.from(
|
||||||
|
(new Array(900)).fill('').map((_, i) => `key${i}=value${i}`).join('&')
|
||||||
|
),
|
||||||
|
];
|
||||||
|
const calls = {
|
||||||
|
field: 0,
|
||||||
|
end: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
const moduleName = process.argv[2];
|
||||||
|
switch (moduleName) {
|
||||||
|
case 'busboy': {
|
||||||
|
const busboy = require('busboy');
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
const parser = busboy({
|
||||||
|
limits: {
|
||||||
|
fieldSizeLimit: Infinity,
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'content-type': 'application/x-www-form-urlencoded; charset=utf-8',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
parser.on('field', (name, val, info) => {
|
||||||
|
++calls.field;
|
||||||
|
}).on('close', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
parser.end();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable': {
|
||||||
|
const QuerystringParser =
|
||||||
|
require('formidable/src/parsers/Querystring.js');
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
const parser = new QuerystringParser();
|
||||||
|
parser.on('data', (obj) => {
|
||||||
|
++calls.field;
|
||||||
|
}).on('end', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
parser.end();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 'formidable-streaming': {
|
||||||
|
const QuerystringParser =
|
||||||
|
require('formidable/src/parsers/StreamingQuerystring.js');
|
||||||
|
|
||||||
|
console.time(moduleName);
|
||||||
|
const parser = new QuerystringParser();
|
||||||
|
parser.on('data', (obj) => {
|
||||||
|
++calls.field;
|
||||||
|
}).on('end', () => {
|
||||||
|
++calls.end;
|
||||||
|
console.timeEnd(moduleName);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const buf of buffers)
|
||||||
|
parser.write(buf);
|
||||||
|
parser.end();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (moduleName === undefined)
|
||||||
|
console.error('Missing parser module name');
|
||||||
|
else
|
||||||
|
console.error(`Invalid parser module name: ${moduleName}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
57
node_modules/busboy/lib/index.js
generated
vendored
Normal file
57
node_modules/busboy/lib/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1,57 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const { parseContentType } = require('./utils.js');
|
||||||
|
|
||||||
|
function getInstance(cfg) {
|
||||||
|
const headers = cfg.headers;
|
||||||
|
const conType = parseContentType(headers['content-type']);
|
||||||
|
if (!conType)
|
||||||
|
throw new Error('Malformed content type');
|
||||||
|
|
||||||
|
for (const type of TYPES) {
|
||||||
|
const matched = type.detect(conType);
|
||||||
|
if (!matched)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const instanceCfg = {
|
||||||
|
limits: cfg.limits,
|
||||||
|
headers,
|
||||||
|
conType,
|
||||||
|
highWaterMark: undefined,
|
||||||
|
fileHwm: undefined,
|
||||||
|
defCharset: undefined,
|
||||||
|
defParamCharset: undefined,
|
||||||
|
preservePath: false,
|
||||||
|
};
|
||||||
|
if (cfg.highWaterMark)
|
||||||
|
instanceCfg.highWaterMark = cfg.highWaterMark;
|
||||||
|
if (cfg.fileHwm)
|
||||||
|
instanceCfg.fileHwm = cfg.fileHwm;
|
||||||
|
instanceCfg.defCharset = cfg.defCharset;
|
||||||
|
instanceCfg.defParamCharset = cfg.defParamCharset;
|
||||||
|
instanceCfg.preservePath = cfg.preservePath;
|
||||||
|
return new type(instanceCfg);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error(`Unsupported content type: ${headers['content-type']}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: types are explicitly listed here for easier bundling
|
||||||
|
// See: https://github.com/mscdex/busboy/issues/121
|
||||||
|
const TYPES = [
|
||||||
|
require('./types/multipart'),
|
||||||
|
require('./types/urlencoded'),
|
||||||
|
].filter(function(typemod) { return typeof typemod.detect === 'function'; });
|
||||||
|
|
||||||
|
module.exports = (cfg) => {
|
||||||
|
if (typeof cfg !== 'object' || cfg === null)
|
||||||
|
cfg = {};
|
||||||
|
|
||||||
|
if (typeof cfg.headers !== 'object'
|
||||||
|
|| cfg.headers === null
|
||||||
|
|| typeof cfg.headers['content-type'] !== 'string') {
|
||||||
|
throw new Error('Missing Content-Type');
|
||||||
|
}
|
||||||
|
|
||||||
|
return getInstance(cfg);
|
||||||
|
};
|
||||||
653
node_modules/busboy/lib/types/multipart.js
generated
vendored
Normal file
653
node_modules/busboy/lib/types/multipart.js
generated
vendored
Normal file
|
|
@ -0,0 +1,653 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const { Readable, Writable } = require('stream');
|
||||||
|
|
||||||
|
const StreamSearch = require('streamsearch');
|
||||||
|
|
||||||
|
const {
|
||||||
|
basename,
|
||||||
|
convertToUTF8,
|
||||||
|
getDecoder,
|
||||||
|
parseContentType,
|
||||||
|
parseDisposition,
|
||||||
|
} = require('../utils.js');
|
||||||
|
|
||||||
|
const BUF_CRLF = Buffer.from('\r\n');
|
||||||
|
const BUF_CR = Buffer.from('\r');
|
||||||
|
const BUF_DASH = Buffer.from('-');
|
||||||
|
|
||||||
|
function noop() {}
|
||||||
|
|
||||||
|
const MAX_HEADER_PAIRS = 2000; // From node
|
||||||
|
const MAX_HEADER_SIZE = 16 * 1024; // From node (its default value)
|
||||||
|
|
||||||
|
const HPARSER_NAME = 0;
|
||||||
|
const HPARSER_PRE_OWS = 1;
|
||||||
|
const HPARSER_VALUE = 2;
|
||||||
|
class HeaderParser {
|
||||||
|
constructor(cb) {
|
||||||
|
this.header = Object.create(null);
|
||||||
|
this.pairCount = 0;
|
||||||
|
this.byteCount = 0;
|
||||||
|
this.state = HPARSER_NAME;
|
||||||
|
this.name = '';
|
||||||
|
this.value = '';
|
||||||
|
this.crlf = 0;
|
||||||
|
this.cb = cb;
|
||||||
|
}
|
||||||
|
|
||||||
|
reset() {
|
||||||
|
this.header = Object.create(null);
|
||||||
|
this.pairCount = 0;
|
||||||
|
this.byteCount = 0;
|
||||||
|
this.state = HPARSER_NAME;
|
||||||
|
this.name = '';
|
||||||
|
this.value = '';
|
||||||
|
this.crlf = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
push(chunk, pos, end) {
|
||||||
|
let start = pos;
|
||||||
|
while (pos < end) {
|
||||||
|
switch (this.state) {
|
||||||
|
case HPARSER_NAME: {
|
||||||
|
let done = false;
|
||||||
|
for (; pos < end; ++pos) {
|
||||||
|
if (this.byteCount === MAX_HEADER_SIZE)
|
||||||
|
return -1;
|
||||||
|
++this.byteCount;
|
||||||
|
const code = chunk[pos];
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
if (code !== 58/* ':' */)
|
||||||
|
return -1;
|
||||||
|
this.name += chunk.latin1Slice(start, pos);
|
||||||
|
if (this.name.length === 0)
|
||||||
|
return -1;
|
||||||
|
++pos;
|
||||||
|
done = true;
|
||||||
|
this.state = HPARSER_PRE_OWS;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!done) {
|
||||||
|
this.name += chunk.latin1Slice(start, pos);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// FALLTHROUGH
|
||||||
|
}
|
||||||
|
case HPARSER_PRE_OWS: {
|
||||||
|
// Skip optional whitespace
|
||||||
|
let done = false;
|
||||||
|
for (; pos < end; ++pos) {
|
||||||
|
if (this.byteCount === MAX_HEADER_SIZE)
|
||||||
|
return -1;
|
||||||
|
++this.byteCount;
|
||||||
|
const code = chunk[pos];
|
||||||
|
if (code !== 32/* ' ' */ && code !== 9/* '\t' */) {
|
||||||
|
start = pos;
|
||||||
|
done = true;
|
||||||
|
this.state = HPARSER_VALUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!done)
|
||||||
|
break;
|
||||||
|
// FALLTHROUGH
|
||||||
|
}
|
||||||
|
case HPARSER_VALUE:
|
||||||
|
switch (this.crlf) {
|
||||||
|
case 0: // Nothing yet
|
||||||
|
for (; pos < end; ++pos) {
|
||||||
|
if (this.byteCount === MAX_HEADER_SIZE)
|
||||||
|
return -1;
|
||||||
|
++this.byteCount;
|
||||||
|
const code = chunk[pos];
|
||||||
|
if (FIELD_VCHAR[code] !== 1) {
|
||||||
|
if (code !== 13/* '\r' */)
|
||||||
|
return -1;
|
||||||
|
++this.crlf;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.value += chunk.latin1Slice(start, pos++);
|
||||||
|
break;
|
||||||
|
case 1: // Received CR
|
||||||
|
if (this.byteCount === MAX_HEADER_SIZE)
|
||||||
|
return -1;
|
||||||
|
++this.byteCount;
|
||||||
|
if (chunk[pos++] !== 10/* '\n' */)
|
||||||
|
return -1;
|
||||||
|
++this.crlf;
|
||||||
|
break;
|
||||||
|
case 2: { // Received CR LF
|
||||||
|
if (this.byteCount === MAX_HEADER_SIZE)
|
||||||
|
return -1;
|
||||||
|
++this.byteCount;
|
||||||
|
const code = chunk[pos];
|
||||||
|
if (code === 32/* ' ' */ || code === 9/* '\t' */) {
|
||||||
|
// Folded value
|
||||||
|
start = pos;
|
||||||
|
this.crlf = 0;
|
||||||
|
} else {
|
||||||
|
if (++this.pairCount < MAX_HEADER_PAIRS) {
|
||||||
|
this.name = this.name.toLowerCase();
|
||||||
|
if (this.header[this.name] === undefined)
|
||||||
|
this.header[this.name] = [this.value];
|
||||||
|
else
|
||||||
|
this.header[this.name].push(this.value);
|
||||||
|
}
|
||||||
|
if (code === 13/* '\r' */) {
|
||||||
|
++this.crlf;
|
||||||
|
++pos;
|
||||||
|
} else {
|
||||||
|
// Assume start of next header field name
|
||||||
|
start = pos;
|
||||||
|
this.crlf = 0;
|
||||||
|
this.state = HPARSER_NAME;
|
||||||
|
this.name = '';
|
||||||
|
this.value = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 3: { // Received CR LF CR
|
||||||
|
if (this.byteCount === MAX_HEADER_SIZE)
|
||||||
|
return -1;
|
||||||
|
++this.byteCount;
|
||||||
|
if (chunk[pos++] !== 10/* '\n' */)
|
||||||
|
return -1;
|
||||||
|
// End of header
|
||||||
|
const header = this.header;
|
||||||
|
this.reset();
|
||||||
|
this.cb(header);
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class FileStream extends Readable {
|
||||||
|
constructor(opts, owner) {
|
||||||
|
super(opts);
|
||||||
|
this.truncated = false;
|
||||||
|
this._readcb = null;
|
||||||
|
this.once('end', () => {
|
||||||
|
// We need to make sure that we call any outstanding _writecb() that is
|
||||||
|
// associated with this file so that processing of the rest of the form
|
||||||
|
// can continue. This may not happen if the file stream ends right after
|
||||||
|
// backpressure kicks in, so we force it here.
|
||||||
|
this._read();
|
||||||
|
if (--owner._fileEndsLeft === 0 && owner._finalcb) {
|
||||||
|
const cb = owner._finalcb;
|
||||||
|
owner._finalcb = null;
|
||||||
|
// Make sure other 'end' event handlers get a chance to be executed
|
||||||
|
// before busboy's 'finish' event is emitted
|
||||||
|
process.nextTick(cb);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
_read(n) {
|
||||||
|
const cb = this._readcb;
|
||||||
|
if (cb) {
|
||||||
|
this._readcb = null;
|
||||||
|
cb();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const ignoreData = {
|
||||||
|
push: (chunk, pos) => {},
|
||||||
|
destroy: () => {},
|
||||||
|
};
|
||||||
|
|
||||||
|
function callAndUnsetCb(self, err) {
|
||||||
|
const cb = self._writecb;
|
||||||
|
self._writecb = null;
|
||||||
|
if (err)
|
||||||
|
self.destroy(err);
|
||||||
|
else if (cb)
|
||||||
|
cb();
|
||||||
|
}
|
||||||
|
|
||||||
|
function nullDecoder(val, hint) {
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
class Multipart extends Writable {
|
||||||
|
constructor(cfg) {
|
||||||
|
const streamOpts = {
|
||||||
|
autoDestroy: true,
|
||||||
|
emitClose: true,
|
||||||
|
highWaterMark: (typeof cfg.highWaterMark === 'number'
|
||||||
|
? cfg.highWaterMark
|
||||||
|
: undefined),
|
||||||
|
};
|
||||||
|
super(streamOpts);
|
||||||
|
|
||||||
|
if (!cfg.conType.params || typeof cfg.conType.params.boundary !== 'string')
|
||||||
|
throw new Error('Multipart: Boundary not found');
|
||||||
|
|
||||||
|
const boundary = cfg.conType.params.boundary;
|
||||||
|
const paramDecoder = (typeof cfg.defParamCharset === 'string'
|
||||||
|
&& cfg.defParamCharset
|
||||||
|
? getDecoder(cfg.defParamCharset)
|
||||||
|
: nullDecoder);
|
||||||
|
const defCharset = (cfg.defCharset || 'utf8');
|
||||||
|
const preservePath = cfg.preservePath;
|
||||||
|
const fileOpts = {
|
||||||
|
autoDestroy: true,
|
||||||
|
emitClose: true,
|
||||||
|
highWaterMark: (typeof cfg.fileHwm === 'number'
|
||||||
|
? cfg.fileHwm
|
||||||
|
: undefined),
|
||||||
|
};
|
||||||
|
|
||||||
|
const limits = cfg.limits;
|
||||||
|
const fieldSizeLimit = (limits && typeof limits.fieldSize === 'number'
|
||||||
|
? limits.fieldSize
|
||||||
|
: 1 * 1024 * 1024);
|
||||||
|
const fileSizeLimit = (limits && typeof limits.fileSize === 'number'
|
||||||
|
? limits.fileSize
|
||||||
|
: Infinity);
|
||||||
|
const filesLimit = (limits && typeof limits.files === 'number'
|
||||||
|
? limits.files
|
||||||
|
: Infinity);
|
||||||
|
const fieldsLimit = (limits && typeof limits.fields === 'number'
|
||||||
|
? limits.fields
|
||||||
|
: Infinity);
|
||||||
|
const partsLimit = (limits && typeof limits.parts === 'number'
|
||||||
|
? limits.parts
|
||||||
|
: Infinity);
|
||||||
|
|
||||||
|
let parts = -1; // Account for initial boundary
|
||||||
|
let fields = 0;
|
||||||
|
let files = 0;
|
||||||
|
let skipPart = false;
|
||||||
|
|
||||||
|
this._fileEndsLeft = 0;
|
||||||
|
this._fileStream = undefined;
|
||||||
|
this._complete = false;
|
||||||
|
let fileSize = 0;
|
||||||
|
|
||||||
|
let field;
|
||||||
|
let fieldSize = 0;
|
||||||
|
let partCharset;
|
||||||
|
let partEncoding;
|
||||||
|
let partType;
|
||||||
|
let partName;
|
||||||
|
let partTruncated = false;
|
||||||
|
|
||||||
|
let hitFilesLimit = false;
|
||||||
|
let hitFieldsLimit = false;
|
||||||
|
|
||||||
|
this._hparser = null;
|
||||||
|
const hparser = new HeaderParser((header) => {
|
||||||
|
this._hparser = null;
|
||||||
|
skipPart = false;
|
||||||
|
|
||||||
|
partType = 'text/plain';
|
||||||
|
partCharset = defCharset;
|
||||||
|
partEncoding = '7bit';
|
||||||
|
partName = undefined;
|
||||||
|
partTruncated = false;
|
||||||
|
|
||||||
|
let filename;
|
||||||
|
if (!header['content-disposition']) {
|
||||||
|
skipPart = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const disp = parseDisposition(header['content-disposition'][0],
|
||||||
|
paramDecoder);
|
||||||
|
if (!disp || disp.type !== 'form-data') {
|
||||||
|
skipPart = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (disp.params) {
|
||||||
|
if (disp.params.name)
|
||||||
|
partName = disp.params.name;
|
||||||
|
|
||||||
|
if (disp.params['filename*'])
|
||||||
|
filename = disp.params['filename*'];
|
||||||
|
else if (disp.params.filename)
|
||||||
|
filename = disp.params.filename;
|
||||||
|
|
||||||
|
if (filename !== undefined && !preservePath)
|
||||||
|
filename = basename(filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (header['content-type']) {
|
||||||
|
const conType = parseContentType(header['content-type'][0]);
|
||||||
|
if (conType) {
|
||||||
|
partType = `${conType.type}/${conType.subtype}`;
|
||||||
|
if (conType.params && typeof conType.params.charset === 'string')
|
||||||
|
partCharset = conType.params.charset.toLowerCase();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (header['content-transfer-encoding'])
|
||||||
|
partEncoding = header['content-transfer-encoding'][0].toLowerCase();
|
||||||
|
|
||||||
|
if (partType === 'application/octet-stream' || filename !== undefined) {
|
||||||
|
// File
|
||||||
|
|
||||||
|
if (files === filesLimit) {
|
||||||
|
if (!hitFilesLimit) {
|
||||||
|
hitFilesLimit = true;
|
||||||
|
this.emit('filesLimit');
|
||||||
|
}
|
||||||
|
skipPart = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
++files;
|
||||||
|
|
||||||
|
if (this.listenerCount('file') === 0) {
|
||||||
|
skipPart = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fileSize = 0;
|
||||||
|
this._fileStream = new FileStream(fileOpts, this);
|
||||||
|
++this._fileEndsLeft;
|
||||||
|
this.emit(
|
||||||
|
'file',
|
||||||
|
partName,
|
||||||
|
this._fileStream,
|
||||||
|
{ filename,
|
||||||
|
encoding: partEncoding,
|
||||||
|
mimeType: partType }
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// Non-file
|
||||||
|
|
||||||
|
if (fields === fieldsLimit) {
|
||||||
|
if (!hitFieldsLimit) {
|
||||||
|
hitFieldsLimit = true;
|
||||||
|
this.emit('fieldsLimit');
|
||||||
|
}
|
||||||
|
skipPart = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
++fields;
|
||||||
|
|
||||||
|
if (this.listenerCount('field') === 0) {
|
||||||
|
skipPart = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
field = [];
|
||||||
|
fieldSize = 0;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let matchPostBoundary = 0;
|
||||||
|
const ssCb = (isMatch, data, start, end, isDataSafe) => {
|
||||||
|
retrydata:
|
||||||
|
while (data) {
|
||||||
|
if (this._hparser !== null) {
|
||||||
|
const ret = this._hparser.push(data, start, end);
|
||||||
|
if (ret === -1) {
|
||||||
|
this._hparser = null;
|
||||||
|
hparser.reset();
|
||||||
|
this.emit('error', new Error('Malformed part header'));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
start = ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (start === end)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (matchPostBoundary !== 0) {
|
||||||
|
if (matchPostBoundary === 1) {
|
||||||
|
switch (data[start]) {
|
||||||
|
case 45: // '-'
|
||||||
|
// Try matching '--' after boundary
|
||||||
|
matchPostBoundary = 2;
|
||||||
|
++start;
|
||||||
|
break;
|
||||||
|
case 13: // '\r'
|
||||||
|
// Try matching CR LF before header
|
||||||
|
matchPostBoundary = 3;
|
||||||
|
++start;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
matchPostBoundary = 0;
|
||||||
|
}
|
||||||
|
if (start === end)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (matchPostBoundary === 2) {
|
||||||
|
matchPostBoundary = 0;
|
||||||
|
if (data[start] === 45/* '-' */) {
|
||||||
|
// End of multipart data
|
||||||
|
this._complete = true;
|
||||||
|
this._bparser = ignoreData;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// We saw something other than '-', so put the dash we consumed
|
||||||
|
// "back"
|
||||||
|
const writecb = this._writecb;
|
||||||
|
this._writecb = noop;
|
||||||
|
ssCb(false, BUF_DASH, 0, 1, false);
|
||||||
|
this._writecb = writecb;
|
||||||
|
} else if (matchPostBoundary === 3) {
|
||||||
|
matchPostBoundary = 0;
|
||||||
|
if (data[start] === 10/* '\n' */) {
|
||||||
|
++start;
|
||||||
|
if (parts >= partsLimit)
|
||||||
|
break;
|
||||||
|
// Prepare the header parser
|
||||||
|
this._hparser = hparser;
|
||||||
|
if (start === end)
|
||||||
|
break;
|
||||||
|
// Process the remaining data as a header
|
||||||
|
continue retrydata;
|
||||||
|
} else {
|
||||||
|
// We saw something other than LF, so put the CR we consumed
|
||||||
|
// "back"
|
||||||
|
const writecb = this._writecb;
|
||||||
|
this._writecb = noop;
|
||||||
|
ssCb(false, BUF_CR, 0, 1, false);
|
||||||
|
this._writecb = writecb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!skipPart) {
|
||||||
|
if (this._fileStream) {
|
||||||
|
let chunk;
|
||||||
|
const actualLen = Math.min(end - start, fileSizeLimit - fileSize);
|
||||||
|
if (!isDataSafe) {
|
||||||
|
chunk = Buffer.allocUnsafe(actualLen);
|
||||||
|
data.copy(chunk, 0, start, start + actualLen);
|
||||||
|
} else {
|
||||||
|
chunk = data.slice(start, start + actualLen);
|
||||||
|
}
|
||||||
|
|
||||||
|
fileSize += chunk.length;
|
||||||
|
if (fileSize === fileSizeLimit) {
|
||||||
|
if (chunk.length > 0)
|
||||||
|
this._fileStream.push(chunk);
|
||||||
|
this._fileStream.emit('limit');
|
||||||
|
this._fileStream.truncated = true;
|
||||||
|
skipPart = true;
|
||||||
|
} else if (!this._fileStream.push(chunk)) {
|
||||||
|
if (this._writecb)
|
||||||
|
this._fileStream._readcb = this._writecb;
|
||||||
|
this._writecb = null;
|
||||||
|
}
|
||||||
|
} else if (field !== undefined) {
|
||||||
|
let chunk;
|
||||||
|
const actualLen = Math.min(
|
||||||
|
end - start,
|
||||||
|
fieldSizeLimit - fieldSize
|
||||||
|
);
|
||||||
|
if (!isDataSafe) {
|
||||||
|
chunk = Buffer.allocUnsafe(actualLen);
|
||||||
|
data.copy(chunk, 0, start, start + actualLen);
|
||||||
|
} else {
|
||||||
|
chunk = data.slice(start, start + actualLen);
|
||||||
|
}
|
||||||
|
|
||||||
|
fieldSize += actualLen;
|
||||||
|
field.push(chunk);
|
||||||
|
if (fieldSize === fieldSizeLimit) {
|
||||||
|
skipPart = true;
|
||||||
|
partTruncated = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isMatch) {
|
||||||
|
matchPostBoundary = 1;
|
||||||
|
|
||||||
|
if (this._fileStream) {
|
||||||
|
// End the active file stream if the previous part was a file
|
||||||
|
this._fileStream.push(null);
|
||||||
|
this._fileStream = null;
|
||||||
|
} else if (field !== undefined) {
|
||||||
|
let data;
|
||||||
|
switch (field.length) {
|
||||||
|
case 0:
|
||||||
|
data = '';
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
data = convertToUTF8(field[0], partCharset, 0);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
data = convertToUTF8(
|
||||||
|
Buffer.concat(field, fieldSize),
|
||||||
|
partCharset,
|
||||||
|
0
|
||||||
|
);
|
||||||
|
}
|
||||||
|
field = undefined;
|
||||||
|
fieldSize = 0;
|
||||||
|
this.emit(
|
||||||
|
'field',
|
||||||
|
partName,
|
||||||
|
data,
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: partTruncated,
|
||||||
|
encoding: partEncoding,
|
||||||
|
mimeType: partType }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (++parts === partsLimit)
|
||||||
|
this.emit('partsLimit');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
this._bparser = new StreamSearch(`\r\n--${boundary}`, ssCb);
|
||||||
|
|
||||||
|
this._writecb = null;
|
||||||
|
this._finalcb = null;
|
||||||
|
|
||||||
|
// Just in case there is no preamble
|
||||||
|
this.write(BUF_CRLF);
|
||||||
|
}
|
||||||
|
|
||||||
|
static detect(conType) {
|
||||||
|
return (conType.type === 'multipart' && conType.subtype === 'form-data');
|
||||||
|
}
|
||||||
|
|
||||||
|
_write(chunk, enc, cb) {
|
||||||
|
this._writecb = cb;
|
||||||
|
this._bparser.push(chunk, 0);
|
||||||
|
if (this._writecb)
|
||||||
|
callAndUnsetCb(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
_destroy(err, cb) {
|
||||||
|
this._hparser = null;
|
||||||
|
this._bparser = ignoreData;
|
||||||
|
if (!err)
|
||||||
|
err = checkEndState(this);
|
||||||
|
const fileStream = this._fileStream;
|
||||||
|
if (fileStream) {
|
||||||
|
this._fileStream = null;
|
||||||
|
fileStream.destroy(err);
|
||||||
|
}
|
||||||
|
cb(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
_final(cb) {
|
||||||
|
this._bparser.destroy();
|
||||||
|
if (!this._complete)
|
||||||
|
return cb(new Error('Unexpected end of form'));
|
||||||
|
if (this._fileEndsLeft)
|
||||||
|
this._finalcb = finalcb.bind(null, this, cb);
|
||||||
|
else
|
||||||
|
finalcb(this, cb);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function finalcb(self, cb, err) {
|
||||||
|
if (err)
|
||||||
|
return cb(err);
|
||||||
|
err = checkEndState(self);
|
||||||
|
cb(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkEndState(self) {
|
||||||
|
if (self._hparser)
|
||||||
|
return new Error('Malformed part header');
|
||||||
|
const fileStream = self._fileStream;
|
||||||
|
if (fileStream) {
|
||||||
|
self._fileStream = null;
|
||||||
|
fileStream.destroy(new Error('Unexpected end of file'));
|
||||||
|
}
|
||||||
|
if (!self._complete)
|
||||||
|
return new Error('Unexpected end of form');
|
||||||
|
}
|
||||||
|
|
||||||
|
const TOKEN = [
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
];
|
||||||
|
|
||||||
|
const FIELD_VCHAR = [
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
];
|
||||||
|
|
||||||
|
module.exports = Multipart;
|
||||||
350
node_modules/busboy/lib/types/urlencoded.js
generated
vendored
Normal file
350
node_modules/busboy/lib/types/urlencoded.js
generated
vendored
Normal file
|
|
@ -0,0 +1,350 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const { Writable } = require('stream');
|
||||||
|
|
||||||
|
const { getDecoder } = require('../utils.js');
|
||||||
|
|
||||||
|
class URLEncoded extends Writable {
|
||||||
|
constructor(cfg) {
|
||||||
|
const streamOpts = {
|
||||||
|
autoDestroy: true,
|
||||||
|
emitClose: true,
|
||||||
|
highWaterMark: (typeof cfg.highWaterMark === 'number'
|
||||||
|
? cfg.highWaterMark
|
||||||
|
: undefined),
|
||||||
|
};
|
||||||
|
super(streamOpts);
|
||||||
|
|
||||||
|
let charset = (cfg.defCharset || 'utf8');
|
||||||
|
if (cfg.conType.params && typeof cfg.conType.params.charset === 'string')
|
||||||
|
charset = cfg.conType.params.charset;
|
||||||
|
|
||||||
|
this.charset = charset;
|
||||||
|
|
||||||
|
const limits = cfg.limits;
|
||||||
|
this.fieldSizeLimit = (limits && typeof limits.fieldSize === 'number'
|
||||||
|
? limits.fieldSize
|
||||||
|
: 1 * 1024 * 1024);
|
||||||
|
this.fieldsLimit = (limits && typeof limits.fields === 'number'
|
||||||
|
? limits.fields
|
||||||
|
: Infinity);
|
||||||
|
this.fieldNameSizeLimit = (
|
||||||
|
limits && typeof limits.fieldNameSize === 'number'
|
||||||
|
? limits.fieldNameSize
|
||||||
|
: 100
|
||||||
|
);
|
||||||
|
|
||||||
|
this._inKey = true;
|
||||||
|
this._keyTrunc = false;
|
||||||
|
this._valTrunc = false;
|
||||||
|
this._bytesKey = 0;
|
||||||
|
this._bytesVal = 0;
|
||||||
|
this._fields = 0;
|
||||||
|
this._key = '';
|
||||||
|
this._val = '';
|
||||||
|
this._byte = -2;
|
||||||
|
this._lastPos = 0;
|
||||||
|
this._encode = 0;
|
||||||
|
this._decoder = getDecoder(charset);
|
||||||
|
}
|
||||||
|
|
||||||
|
static detect(conType) {
|
||||||
|
return (conType.type === 'application'
|
||||||
|
&& conType.subtype === 'x-www-form-urlencoded');
|
||||||
|
}
|
||||||
|
|
||||||
|
_write(chunk, enc, cb) {
|
||||||
|
if (this._fields >= this.fieldsLimit)
|
||||||
|
return cb();
|
||||||
|
|
||||||
|
let i = 0;
|
||||||
|
const len = chunk.length;
|
||||||
|
this._lastPos = 0;
|
||||||
|
|
||||||
|
// Check if we last ended mid-percent-encoded byte
|
||||||
|
if (this._byte !== -2) {
|
||||||
|
i = readPctEnc(this, chunk, i, len);
|
||||||
|
if (i === -1)
|
||||||
|
return cb(new Error('Malformed urlencoded form'));
|
||||||
|
if (i >= len)
|
||||||
|
return cb();
|
||||||
|
if (this._inKey)
|
||||||
|
++this._bytesKey;
|
||||||
|
else
|
||||||
|
++this._bytesVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
main:
|
||||||
|
while (i < len) {
|
||||||
|
if (this._inKey) {
|
||||||
|
// Parsing key
|
||||||
|
|
||||||
|
i = skipKeyBytes(this, chunk, i, len);
|
||||||
|
|
||||||
|
while (i < len) {
|
||||||
|
switch (chunk[i]) {
|
||||||
|
case 61: // '='
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._key += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._lastPos = ++i;
|
||||||
|
this._key = this._decoder(this._key, this._encode);
|
||||||
|
this._encode = 0;
|
||||||
|
this._inKey = false;
|
||||||
|
continue main;
|
||||||
|
case 38: // '&'
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._key += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._lastPos = ++i;
|
||||||
|
this._key = this._decoder(this._key, this._encode);
|
||||||
|
this._encode = 0;
|
||||||
|
if (this._bytesKey > 0) {
|
||||||
|
this.emit(
|
||||||
|
'field',
|
||||||
|
this._key,
|
||||||
|
'',
|
||||||
|
{ nameTruncated: this._keyTrunc,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: this.charset,
|
||||||
|
mimeType: 'text/plain' }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
this._key = '';
|
||||||
|
this._val = '';
|
||||||
|
this._keyTrunc = false;
|
||||||
|
this._valTrunc = false;
|
||||||
|
this._bytesKey = 0;
|
||||||
|
this._bytesVal = 0;
|
||||||
|
if (++this._fields >= this.fieldsLimit) {
|
||||||
|
this.emit('fieldsLimit');
|
||||||
|
return cb();
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
case 43: // '+'
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._key += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._key += ' ';
|
||||||
|
this._lastPos = i + 1;
|
||||||
|
break;
|
||||||
|
case 37: // '%'
|
||||||
|
if (this._encode === 0)
|
||||||
|
this._encode = 1;
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._key += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._lastPos = i + 1;
|
||||||
|
this._byte = -1;
|
||||||
|
i = readPctEnc(this, chunk, i + 1, len);
|
||||||
|
if (i === -1)
|
||||||
|
return cb(new Error('Malformed urlencoded form'));
|
||||||
|
if (i >= len)
|
||||||
|
return cb();
|
||||||
|
++this._bytesKey;
|
||||||
|
i = skipKeyBytes(this, chunk, i, len);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
++i;
|
||||||
|
++this._bytesKey;
|
||||||
|
i = skipKeyBytes(this, chunk, i, len);
|
||||||
|
}
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._key += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
} else {
|
||||||
|
// Parsing value
|
||||||
|
|
||||||
|
i = skipValBytes(this, chunk, i, len);
|
||||||
|
|
||||||
|
while (i < len) {
|
||||||
|
switch (chunk[i]) {
|
||||||
|
case 38: // '&'
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._val += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._lastPos = ++i;
|
||||||
|
this._inKey = true;
|
||||||
|
this._val = this._decoder(this._val, this._encode);
|
||||||
|
this._encode = 0;
|
||||||
|
if (this._bytesKey > 0 || this._bytesVal > 0) {
|
||||||
|
this.emit(
|
||||||
|
'field',
|
||||||
|
this._key,
|
||||||
|
this._val,
|
||||||
|
{ nameTruncated: this._keyTrunc,
|
||||||
|
valueTruncated: this._valTrunc,
|
||||||
|
encoding: this.charset,
|
||||||
|
mimeType: 'text/plain' }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
this._key = '';
|
||||||
|
this._val = '';
|
||||||
|
this._keyTrunc = false;
|
||||||
|
this._valTrunc = false;
|
||||||
|
this._bytesKey = 0;
|
||||||
|
this._bytesVal = 0;
|
||||||
|
if (++this._fields >= this.fieldsLimit) {
|
||||||
|
this.emit('fieldsLimit');
|
||||||
|
return cb();
|
||||||
|
}
|
||||||
|
continue main;
|
||||||
|
case 43: // '+'
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._val += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._val += ' ';
|
||||||
|
this._lastPos = i + 1;
|
||||||
|
break;
|
||||||
|
case 37: // '%'
|
||||||
|
if (this._encode === 0)
|
||||||
|
this._encode = 1;
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._val += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
this._lastPos = i + 1;
|
||||||
|
this._byte = -1;
|
||||||
|
i = readPctEnc(this, chunk, i + 1, len);
|
||||||
|
if (i === -1)
|
||||||
|
return cb(new Error('Malformed urlencoded form'));
|
||||||
|
if (i >= len)
|
||||||
|
return cb();
|
||||||
|
++this._bytesVal;
|
||||||
|
i = skipValBytes(this, chunk, i, len);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
++i;
|
||||||
|
++this._bytesVal;
|
||||||
|
i = skipValBytes(this, chunk, i, len);
|
||||||
|
}
|
||||||
|
if (this._lastPos < i)
|
||||||
|
this._val += chunk.latin1Slice(this._lastPos, i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cb();
|
||||||
|
}
|
||||||
|
|
||||||
|
_final(cb) {
|
||||||
|
if (this._byte !== -2)
|
||||||
|
return cb(new Error('Malformed urlencoded form'));
|
||||||
|
if (!this._inKey || this._bytesKey > 0 || this._bytesVal > 0) {
|
||||||
|
if (this._inKey)
|
||||||
|
this._key = this._decoder(this._key, this._encode);
|
||||||
|
else
|
||||||
|
this._val = this._decoder(this._val, this._encode);
|
||||||
|
this.emit(
|
||||||
|
'field',
|
||||||
|
this._key,
|
||||||
|
this._val,
|
||||||
|
{ nameTruncated: this._keyTrunc,
|
||||||
|
valueTruncated: this._valTrunc,
|
||||||
|
encoding: this.charset,
|
||||||
|
mimeType: 'text/plain' }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
cb();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function readPctEnc(self, chunk, pos, len) {
|
||||||
|
if (pos >= len)
|
||||||
|
return len;
|
||||||
|
|
||||||
|
if (self._byte === -1) {
|
||||||
|
// We saw a '%' but no hex characters yet
|
||||||
|
const hexUpper = HEX_VALUES[chunk[pos++]];
|
||||||
|
if (hexUpper === -1)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (hexUpper >= 8)
|
||||||
|
self._encode = 2; // Indicate high bits detected
|
||||||
|
|
||||||
|
if (pos < len) {
|
||||||
|
// Both hex characters are in this chunk
|
||||||
|
const hexLower = HEX_VALUES[chunk[pos++]];
|
||||||
|
if (hexLower === -1)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (self._inKey)
|
||||||
|
self._key += String.fromCharCode((hexUpper << 4) + hexLower);
|
||||||
|
else
|
||||||
|
self._val += String.fromCharCode((hexUpper << 4) + hexLower);
|
||||||
|
|
||||||
|
self._byte = -2;
|
||||||
|
self._lastPos = pos;
|
||||||
|
} else {
|
||||||
|
// Only one hex character was available in this chunk
|
||||||
|
self._byte = hexUpper;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// We saw only one hex character so far
|
||||||
|
const hexLower = HEX_VALUES[chunk[pos++]];
|
||||||
|
if (hexLower === -1)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (self._inKey)
|
||||||
|
self._key += String.fromCharCode((self._byte << 4) + hexLower);
|
||||||
|
else
|
||||||
|
self._val += String.fromCharCode((self._byte << 4) + hexLower);
|
||||||
|
|
||||||
|
self._byte = -2;
|
||||||
|
self._lastPos = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
function skipKeyBytes(self, chunk, pos, len) {
|
||||||
|
// Skip bytes if we've truncated
|
||||||
|
if (self._bytesKey > self.fieldNameSizeLimit) {
|
||||||
|
if (!self._keyTrunc) {
|
||||||
|
if (self._lastPos < pos)
|
||||||
|
self._key += chunk.latin1Slice(self._lastPos, pos - 1);
|
||||||
|
}
|
||||||
|
self._keyTrunc = true;
|
||||||
|
for (; pos < len; ++pos) {
|
||||||
|
const code = chunk[pos];
|
||||||
|
if (code === 61/* '=' */ || code === 38/* '&' */)
|
||||||
|
break;
|
||||||
|
++self._bytesKey;
|
||||||
|
}
|
||||||
|
self._lastPos = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
function skipValBytes(self, chunk, pos, len) {
|
||||||
|
// Skip bytes if we've truncated
|
||||||
|
if (self._bytesVal > self.fieldSizeLimit) {
|
||||||
|
if (!self._valTrunc) {
|
||||||
|
if (self._lastPos < pos)
|
||||||
|
self._val += chunk.latin1Slice(self._lastPos, pos - 1);
|
||||||
|
}
|
||||||
|
self._valTrunc = true;
|
||||||
|
for (; pos < len; ++pos) {
|
||||||
|
if (chunk[pos] === 38/* '&' */)
|
||||||
|
break;
|
||||||
|
++self._bytesVal;
|
||||||
|
}
|
||||||
|
self._lastPos = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* eslint-disable no-multi-spaces */
|
||||||
|
const HEX_VALUES = [
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
];
|
||||||
|
/* eslint-enable no-multi-spaces */
|
||||||
|
|
||||||
|
module.exports = URLEncoded;
|
||||||
596
node_modules/busboy/lib/utils.js
generated
vendored
Normal file
596
node_modules/busboy/lib/utils.js
generated
vendored
Normal file
|
|
@ -0,0 +1,596 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
function parseContentType(str) {
|
||||||
|
if (str.length === 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const params = Object.create(null);
|
||||||
|
let i = 0;
|
||||||
|
|
||||||
|
// Parse type
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
if (code !== 47/* '/' */ || i === 0)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Check for type without subtype
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const type = str.slice(0, i).toLowerCase();
|
||||||
|
|
||||||
|
// Parse subtype
|
||||||
|
const subtypeStart = ++i;
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
// Make sure we have a subtype
|
||||||
|
if (i === subtypeStart)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (parseContentTypeParams(str, i, params) === undefined)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Make sure we have a subtype
|
||||||
|
if (i === subtypeStart)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const subtype = str.slice(subtypeStart, i).toLowerCase();
|
||||||
|
|
||||||
|
return { type, subtype, params };
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseContentTypeParams(str, i, params) {
|
||||||
|
while (i < str.length) {
|
||||||
|
// Consume whitespace
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code !== 32/* ' ' */ && code !== 9/* '\t' */)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ended on whitespace
|
||||||
|
if (i === str.length)
|
||||||
|
break;
|
||||||
|
|
||||||
|
// Check for malformed parameter
|
||||||
|
if (str.charCodeAt(i++) !== 59/* ';' */)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Consume whitespace
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code !== 32/* ' ' */ && code !== 9/* '\t' */)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ended on whitespace (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
let name;
|
||||||
|
const nameStart = i;
|
||||||
|
// Parse parameter name
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
if (code !== 61/* '=' */)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
name = str.slice(nameStart, i);
|
||||||
|
++i; // Skip over '='
|
||||||
|
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
let value = '';
|
||||||
|
let valueStart;
|
||||||
|
if (str.charCodeAt(i) === 34/* '"' */) {
|
||||||
|
valueStart = ++i;
|
||||||
|
let escaping = false;
|
||||||
|
// Parse quoted value
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code === 92/* '\\' */) {
|
||||||
|
if (escaping) {
|
||||||
|
valueStart = i;
|
||||||
|
escaping = false;
|
||||||
|
} else {
|
||||||
|
value += str.slice(valueStart, i);
|
||||||
|
escaping = true;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (code === 34/* '"' */) {
|
||||||
|
if (escaping) {
|
||||||
|
valueStart = i;
|
||||||
|
escaping = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
value += str.slice(valueStart, i);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (escaping) {
|
||||||
|
valueStart = i - 1;
|
||||||
|
escaping = false;
|
||||||
|
}
|
||||||
|
// Invalid unescaped quoted character (malformed)
|
||||||
|
if (QDTEXT[code] !== 1)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// No end quote (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
++i; // Skip over double quote
|
||||||
|
} else {
|
||||||
|
valueStart = i;
|
||||||
|
// Parse unquoted value
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === valueStart)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
value = str.slice(valueStart, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
name = name.toLowerCase();
|
||||||
|
if (params[name] === undefined)
|
||||||
|
params[name] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return params;
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseDisposition(str, defDecoder) {
|
||||||
|
if (str.length === 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const params = Object.create(null);
|
||||||
|
let i = 0;
|
||||||
|
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
if (parseDispositionParams(str, i, params, defDecoder) === undefined)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const type = str.slice(0, i).toLowerCase();
|
||||||
|
|
||||||
|
return { type, params };
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseDispositionParams(str, i, params, defDecoder) {
|
||||||
|
while (i < str.length) {
|
||||||
|
// Consume whitespace
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code !== 32/* ' ' */ && code !== 9/* '\t' */)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ended on whitespace
|
||||||
|
if (i === str.length)
|
||||||
|
break;
|
||||||
|
|
||||||
|
// Check for malformed parameter
|
||||||
|
if (str.charCodeAt(i++) !== 59/* ';' */)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Consume whitespace
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code !== 32/* ' ' */ && code !== 9/* '\t' */)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ended on whitespace (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
let name;
|
||||||
|
const nameStart = i;
|
||||||
|
// Parse parameter name
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
if (code === 61/* '=' */)
|
||||||
|
break;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
let value = '';
|
||||||
|
let valueStart;
|
||||||
|
let charset;
|
||||||
|
//~ let lang;
|
||||||
|
name = str.slice(nameStart, i);
|
||||||
|
if (name.charCodeAt(name.length - 1) === 42/* '*' */) {
|
||||||
|
// Extended value
|
||||||
|
|
||||||
|
const charsetStart = ++i;
|
||||||
|
// Parse charset name
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (CHARSET[code] !== 1) {
|
||||||
|
if (code !== 39/* '\'' */)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Incomplete charset (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
charset = str.slice(charsetStart, i);
|
||||||
|
++i; // Skip over the '\''
|
||||||
|
|
||||||
|
//~ const langStart = ++i;
|
||||||
|
// Parse language name
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code === 39/* '\'' */)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Incomplete language (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
//~ lang = str.slice(langStart, i);
|
||||||
|
++i; // Skip over the '\''
|
||||||
|
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
valueStart = i;
|
||||||
|
|
||||||
|
let encode = 0;
|
||||||
|
// Parse value
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (EXTENDED_VALUE[code] !== 1) {
|
||||||
|
if (code === 37/* '%' */) {
|
||||||
|
let hexUpper;
|
||||||
|
let hexLower;
|
||||||
|
if (i + 2 < str.length
|
||||||
|
&& (hexUpper = HEX_VALUES[str.charCodeAt(i + 1)]) !== -1
|
||||||
|
&& (hexLower = HEX_VALUES[str.charCodeAt(i + 2)]) !== -1) {
|
||||||
|
const byteVal = (hexUpper << 4) + hexLower;
|
||||||
|
value += str.slice(valueStart, i);
|
||||||
|
value += String.fromCharCode(byteVal);
|
||||||
|
i += 2;
|
||||||
|
valueStart = i + 1;
|
||||||
|
if (byteVal >= 128)
|
||||||
|
encode = 2;
|
||||||
|
else if (encode === 0)
|
||||||
|
encode = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// '%' disallowed in non-percent encoded contexts (malformed)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
value += str.slice(valueStart, i);
|
||||||
|
value = convertToUTF8(value, charset, encode);
|
||||||
|
if (value === undefined)
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
// Non-extended value
|
||||||
|
|
||||||
|
++i; // Skip over '='
|
||||||
|
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (str.charCodeAt(i) === 34/* '"' */) {
|
||||||
|
valueStart = ++i;
|
||||||
|
let escaping = false;
|
||||||
|
// Parse quoted value
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (code === 92/* '\\' */) {
|
||||||
|
if (escaping) {
|
||||||
|
valueStart = i;
|
||||||
|
escaping = false;
|
||||||
|
} else {
|
||||||
|
value += str.slice(valueStart, i);
|
||||||
|
escaping = true;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (code === 34/* '"' */) {
|
||||||
|
if (escaping) {
|
||||||
|
valueStart = i;
|
||||||
|
escaping = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
value += str.slice(valueStart, i);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (escaping) {
|
||||||
|
valueStart = i - 1;
|
||||||
|
escaping = false;
|
||||||
|
}
|
||||||
|
// Invalid unescaped quoted character (malformed)
|
||||||
|
if (QDTEXT[code] !== 1)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// No end quote (malformed)
|
||||||
|
if (i === str.length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
++i; // Skip over double quote
|
||||||
|
} else {
|
||||||
|
valueStart = i;
|
||||||
|
// Parse unquoted value
|
||||||
|
for (; i < str.length; ++i) {
|
||||||
|
const code = str.charCodeAt(i);
|
||||||
|
if (TOKEN[code] !== 1) {
|
||||||
|
// No value (malformed)
|
||||||
|
if (i === valueStart)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
value = str.slice(valueStart, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
value = defDecoder(value, 2);
|
||||||
|
if (value === undefined)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
name = name.toLowerCase();
|
||||||
|
if (params[name] === undefined)
|
||||||
|
params[name] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return params;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDecoder(charset) {
|
||||||
|
let lc;
|
||||||
|
while (true) {
|
||||||
|
switch (charset) {
|
||||||
|
case 'utf-8':
|
||||||
|
case 'utf8':
|
||||||
|
return decoders.utf8;
|
||||||
|
case 'latin1':
|
||||||
|
case 'ascii': // TODO: Make these a separate, strict decoder?
|
||||||
|
case 'us-ascii':
|
||||||
|
case 'iso-8859-1':
|
||||||
|
case 'iso8859-1':
|
||||||
|
case 'iso88591':
|
||||||
|
case 'iso_8859-1':
|
||||||
|
case 'windows-1252':
|
||||||
|
case 'iso_8859-1:1987':
|
||||||
|
case 'cp1252':
|
||||||
|
case 'x-cp1252':
|
||||||
|
return decoders.latin1;
|
||||||
|
case 'utf16le':
|
||||||
|
case 'utf-16le':
|
||||||
|
case 'ucs2':
|
||||||
|
case 'ucs-2':
|
||||||
|
return decoders.utf16le;
|
||||||
|
case 'base64':
|
||||||
|
return decoders.base64;
|
||||||
|
default:
|
||||||
|
if (lc === undefined) {
|
||||||
|
lc = true;
|
||||||
|
charset = charset.toLowerCase();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
return decoders.other.bind(charset);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const decoders = {
|
||||||
|
utf8: (data, hint) => {
|
||||||
|
if (data.length === 0)
|
||||||
|
return '';
|
||||||
|
if (typeof data === 'string') {
|
||||||
|
// If `data` never had any percent-encoded bytes or never had any that
|
||||||
|
// were outside of the ASCII range, then we can safely just return the
|
||||||
|
// input since UTF-8 is ASCII compatible
|
||||||
|
if (hint < 2)
|
||||||
|
return data;
|
||||||
|
|
||||||
|
data = Buffer.from(data, 'latin1');
|
||||||
|
}
|
||||||
|
return data.utf8Slice(0, data.length);
|
||||||
|
},
|
||||||
|
|
||||||
|
latin1: (data, hint) => {
|
||||||
|
if (data.length === 0)
|
||||||
|
return '';
|
||||||
|
if (typeof data === 'string')
|
||||||
|
return data;
|
||||||
|
return data.latin1Slice(0, data.length);
|
||||||
|
},
|
||||||
|
|
||||||
|
utf16le: (data, hint) => {
|
||||||
|
if (data.length === 0)
|
||||||
|
return '';
|
||||||
|
if (typeof data === 'string')
|
||||||
|
data = Buffer.from(data, 'latin1');
|
||||||
|
return data.ucs2Slice(0, data.length);
|
||||||
|
},
|
||||||
|
|
||||||
|
base64: (data, hint) => {
|
||||||
|
if (data.length === 0)
|
||||||
|
return '';
|
||||||
|
if (typeof data === 'string')
|
||||||
|
data = Buffer.from(data, 'latin1');
|
||||||
|
return data.base64Slice(0, data.length);
|
||||||
|
},
|
||||||
|
|
||||||
|
other: (data, hint) => {
|
||||||
|
if (data.length === 0)
|
||||||
|
return '';
|
||||||
|
if (typeof data === 'string')
|
||||||
|
data = Buffer.from(data, 'latin1');
|
||||||
|
try {
|
||||||
|
const decoder = new TextDecoder(this);
|
||||||
|
return decoder.decode(data);
|
||||||
|
} catch {}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
function convertToUTF8(data, charset, hint) {
|
||||||
|
const decode = getDecoder(charset);
|
||||||
|
if (decode)
|
||||||
|
return decode(data, hint);
|
||||||
|
}
|
||||||
|
|
||||||
|
function basename(path) {
|
||||||
|
if (typeof path !== 'string')
|
||||||
|
return '';
|
||||||
|
for (let i = path.length - 1; i >= 0; --i) {
|
||||||
|
switch (path.charCodeAt(i)) {
|
||||||
|
case 0x2F: // '/'
|
||||||
|
case 0x5C: // '\'
|
||||||
|
path = path.slice(i + 1);
|
||||||
|
return (path === '..' || path === '.' ? '' : path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (path === '..' || path === '.' ? '' : path);
|
||||||
|
}
|
||||||
|
|
||||||
|
const TOKEN = [
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
];
|
||||||
|
|
||||||
|
const QDTEXT = [
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
];
|
||||||
|
|
||||||
|
const CHARSET = [
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 0, 1, 1, 1, 1, 0, 0, 0, 0, 1, 0, 1, 0, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
];
|
||||||
|
|
||||||
|
const EXTENDED_VALUE = [
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 0, 1, 1, 0, 1, 0, 0, 0, 0, 1, 0, 1, 1, 0,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||||
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
];
|
||||||
|
|
||||||
|
/* eslint-disable no-multi-spaces */
|
||||||
|
const HEX_VALUES = [
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
];
|
||||||
|
/* eslint-enable no-multi-spaces */
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
basename,
|
||||||
|
convertToUTF8,
|
||||||
|
getDecoder,
|
||||||
|
parseContentType,
|
||||||
|
parseDisposition,
|
||||||
|
};
|
||||||
22
node_modules/busboy/package.json
generated
vendored
Normal file
22
node_modules/busboy/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,22 @@
|
||||||
|
{ "name": "busboy",
|
||||||
|
"version": "1.6.0",
|
||||||
|
"author": "Brian White <mscdex@mscdex.net>",
|
||||||
|
"description": "A streaming parser for HTML form data for node.js",
|
||||||
|
"main": "./lib/index.js",
|
||||||
|
"dependencies": {
|
||||||
|
"streamsearch": "^1.1.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@mscdex/eslint-config": "^1.1.0",
|
||||||
|
"eslint": "^7.32.0"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"test": "node test/test.js",
|
||||||
|
"lint": "eslint --cache --report-unused-disable-directives --ext=.js .eslintrc.js lib test bench",
|
||||||
|
"lint:fix": "npm run lint -- --fix"
|
||||||
|
},
|
||||||
|
"engines": { "node": ">=10.16.0" },
|
||||||
|
"keywords": [ "uploads", "forms", "multipart", "form-data" ],
|
||||||
|
"licenses": [ { "type": "MIT", "url": "http://github.com/mscdex/busboy/raw/master/LICENSE" } ],
|
||||||
|
"repository": { "type": "git", "url": "http://github.com/mscdex/busboy.git" }
|
||||||
|
}
|
||||||
109
node_modules/busboy/test/common.js
generated
vendored
Normal file
109
node_modules/busboy/test/common.js
generated
vendored
Normal file
|
|
@ -0,0 +1,109 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
const { inspect } = require('util');
|
||||||
|
|
||||||
|
const mustCallChecks = [];
|
||||||
|
|
||||||
|
function noop() {}
|
||||||
|
|
||||||
|
function runCallChecks(exitCode) {
|
||||||
|
if (exitCode !== 0) return;
|
||||||
|
|
||||||
|
const failed = mustCallChecks.filter((context) => {
|
||||||
|
if ('minimum' in context) {
|
||||||
|
context.messageSegment = `at least ${context.minimum}`;
|
||||||
|
return context.actual < context.minimum;
|
||||||
|
}
|
||||||
|
context.messageSegment = `exactly ${context.exact}`;
|
||||||
|
return context.actual !== context.exact;
|
||||||
|
});
|
||||||
|
|
||||||
|
failed.forEach((context) => {
|
||||||
|
console.error('Mismatched %s function calls. Expected %s, actual %d.',
|
||||||
|
context.name,
|
||||||
|
context.messageSegment,
|
||||||
|
context.actual);
|
||||||
|
console.error(context.stack.split('\n').slice(2).join('\n'));
|
||||||
|
});
|
||||||
|
|
||||||
|
if (failed.length)
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustCall(fn, exact) {
|
||||||
|
return _mustCallInner(fn, exact, 'exact');
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustCallAtLeast(fn, minimum) {
|
||||||
|
return _mustCallInner(fn, minimum, 'minimum');
|
||||||
|
}
|
||||||
|
|
||||||
|
function _mustCallInner(fn, criteria = 1, field) {
|
||||||
|
if (process._exiting)
|
||||||
|
throw new Error('Cannot use common.mustCall*() in process exit handler');
|
||||||
|
|
||||||
|
if (typeof fn === 'number') {
|
||||||
|
criteria = fn;
|
||||||
|
fn = noop;
|
||||||
|
} else if (fn === undefined) {
|
||||||
|
fn = noop;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof criteria !== 'number')
|
||||||
|
throw new TypeError(`Invalid ${field} value: ${criteria}`);
|
||||||
|
|
||||||
|
const context = {
|
||||||
|
[field]: criteria,
|
||||||
|
actual: 0,
|
||||||
|
stack: inspect(new Error()),
|
||||||
|
name: fn.name || '<anonymous>'
|
||||||
|
};
|
||||||
|
|
||||||
|
// Add the exit listener only once to avoid listener leak warnings
|
||||||
|
if (mustCallChecks.length === 0)
|
||||||
|
process.on('exit', runCallChecks);
|
||||||
|
|
||||||
|
mustCallChecks.push(context);
|
||||||
|
|
||||||
|
function wrapped(...args) {
|
||||||
|
++context.actual;
|
||||||
|
return fn.call(this, ...args);
|
||||||
|
}
|
||||||
|
// TODO: remove origFn?
|
||||||
|
wrapped.origFn = fn;
|
||||||
|
|
||||||
|
return wrapped;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCallSite(top) {
|
||||||
|
const originalStackFormatter = Error.prepareStackTrace;
|
||||||
|
Error.prepareStackTrace = (err, stack) =>
|
||||||
|
`${stack[0].getFileName()}:${stack[0].getLineNumber()}`;
|
||||||
|
const err = new Error();
|
||||||
|
Error.captureStackTrace(err, top);
|
||||||
|
// With the V8 Error API, the stack is not formatted until it is accessed
|
||||||
|
// eslint-disable-next-line no-unused-expressions
|
||||||
|
err.stack;
|
||||||
|
Error.prepareStackTrace = originalStackFormatter;
|
||||||
|
return err.stack;
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustNotCall(msg) {
|
||||||
|
const callSite = getCallSite(mustNotCall);
|
||||||
|
return function mustNotCall(...args) {
|
||||||
|
args = args.map(inspect).join(', ');
|
||||||
|
const argsInfo = (args.length > 0
|
||||||
|
? `\ncalled with arguments: ${args}`
|
||||||
|
: '');
|
||||||
|
assert.fail(
|
||||||
|
`${msg || 'function should not have been called'} at ${callSite}`
|
||||||
|
+ argsInfo);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
mustCall,
|
||||||
|
mustCallAtLeast,
|
||||||
|
mustNotCall,
|
||||||
|
};
|
||||||
94
node_modules/busboy/test/test-types-multipart-charsets.js
generated
vendored
Normal file
94
node_modules/busboy/test/test-types-multipart-charsets.js
generated
vendored
Normal file
|
|
@ -0,0 +1,94 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
const { inspect } = require('util');
|
||||||
|
|
||||||
|
const { mustCall } = require(`${__dirname}/common.js`);
|
||||||
|
|
||||||
|
const busboy = require('..');
|
||||||
|
|
||||||
|
const input = Buffer.from([
|
||||||
|
'-----------------------------paZqsnEHRufoShdX6fh0lUhXBP4k',
|
||||||
|
'Content-Disposition: form-data; '
|
||||||
|
+ 'name="upload_file_0"; filename="テスト.dat"',
|
||||||
|
'Content-Type: application/octet-stream',
|
||||||
|
'',
|
||||||
|
'A'.repeat(1023),
|
||||||
|
'-----------------------------paZqsnEHRufoShdX6fh0lUhXBP4k--'
|
||||||
|
].join('\r\n'));
|
||||||
|
const boundary = '---------------------------paZqsnEHRufoShdX6fh0lUhXBP4k';
|
||||||
|
const expected = [
|
||||||
|
{ type: 'file',
|
||||||
|
name: 'upload_file_0',
|
||||||
|
data: Buffer.from('A'.repeat(1023)),
|
||||||
|
info: {
|
||||||
|
filename: 'テスト.dat',
|
||||||
|
encoding: '7bit',
|
||||||
|
mimeType: 'application/octet-stream',
|
||||||
|
},
|
||||||
|
limited: false,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const bb = busboy({
|
||||||
|
defParamCharset: 'utf8',
|
||||||
|
headers: {
|
||||||
|
'content-type': `multipart/form-data; boundary=${boundary}`,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const results = [];
|
||||||
|
|
||||||
|
bb.on('field', (name, val, info) => {
|
||||||
|
results.push({ type: 'field', name, val, info });
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('file', (name, stream, info) => {
|
||||||
|
const data = [];
|
||||||
|
let nb = 0;
|
||||||
|
const file = {
|
||||||
|
type: 'file',
|
||||||
|
name,
|
||||||
|
data: null,
|
||||||
|
info,
|
||||||
|
limited: false,
|
||||||
|
};
|
||||||
|
results.push(file);
|
||||||
|
stream.on('data', (d) => {
|
||||||
|
data.push(d);
|
||||||
|
nb += d.length;
|
||||||
|
}).on('limit', () => {
|
||||||
|
file.limited = true;
|
||||||
|
}).on('close', () => {
|
||||||
|
file.data = Buffer.concat(data, nb);
|
||||||
|
assert.strictEqual(stream.truncated, file.limited);
|
||||||
|
}).once('error', (err) => {
|
||||||
|
file.err = err.message;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('error', (err) => {
|
||||||
|
results.push({ error: err.message });
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('partsLimit', () => {
|
||||||
|
results.push('partsLimit');
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('filesLimit', () => {
|
||||||
|
results.push('filesLimit');
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('fieldsLimit', () => {
|
||||||
|
results.push('fieldsLimit');
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('close', mustCall(() => {
|
||||||
|
assert.deepStrictEqual(
|
||||||
|
results,
|
||||||
|
expected,
|
||||||
|
'Results mismatch.\n'
|
||||||
|
+ `Parsed: ${inspect(results)}\n`
|
||||||
|
+ `Expected: ${inspect(expected)}`
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
|
||||||
|
bb.end(input);
|
||||||
102
node_modules/busboy/test/test-types-multipart-stream-pause.js
generated
vendored
Normal file
102
node_modules/busboy/test/test-types-multipart-stream-pause.js
generated
vendored
Normal file
|
|
@ -0,0 +1,102 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
const { randomFillSync } = require('crypto');
|
||||||
|
const { inspect } = require('util');
|
||||||
|
|
||||||
|
const busboy = require('..');
|
||||||
|
|
||||||
|
const { mustCall } = require('./common.js');
|
||||||
|
|
||||||
|
const BOUNDARY = 'u2KxIV5yF1y+xUspOQCCZopaVgeV6Jxihv35XQJmuTx8X3sh';
|
||||||
|
|
||||||
|
function formDataSection(key, value) {
|
||||||
|
return Buffer.from(
|
||||||
|
`\r\n--${BOUNDARY}`
|
||||||
|
+ `\r\nContent-Disposition: form-data; name="${key}"`
|
||||||
|
+ `\r\n\r\n${value}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function formDataFile(key, filename, contentType) {
|
||||||
|
const buf = Buffer.allocUnsafe(100000);
|
||||||
|
return Buffer.concat([
|
||||||
|
Buffer.from(`\r\n--${BOUNDARY}\r\n`),
|
||||||
|
Buffer.from(`Content-Disposition: form-data; name="${key}"`
|
||||||
|
+ `; filename="${filename}"\r\n`),
|
||||||
|
Buffer.from(`Content-Type: ${contentType}\r\n\r\n`),
|
||||||
|
randomFillSync(buf)
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
const reqChunks = [
|
||||||
|
Buffer.concat([
|
||||||
|
formDataFile('file', 'file.bin', 'application/octet-stream'),
|
||||||
|
formDataSection('foo', 'foo value'),
|
||||||
|
]),
|
||||||
|
formDataSection('bar', 'bar value'),
|
||||||
|
Buffer.from(`\r\n--${BOUNDARY}--\r\n`)
|
||||||
|
];
|
||||||
|
const bb = busboy({
|
||||||
|
headers: {
|
||||||
|
'content-type': `multipart/form-data; boundary=${BOUNDARY}`
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const expected = [
|
||||||
|
{ type: 'file',
|
||||||
|
name: 'file',
|
||||||
|
info: {
|
||||||
|
filename: 'file.bin',
|
||||||
|
encoding: '7bit',
|
||||||
|
mimeType: 'application/octet-stream',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ type: 'field',
|
||||||
|
name: 'foo',
|
||||||
|
val: 'foo value',
|
||||||
|
info: {
|
||||||
|
nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: '7bit',
|
||||||
|
mimeType: 'text/plain',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ type: 'field',
|
||||||
|
name: 'bar',
|
||||||
|
val: 'bar value',
|
||||||
|
info: {
|
||||||
|
nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: '7bit',
|
||||||
|
mimeType: 'text/plain',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const results = [];
|
||||||
|
|
||||||
|
bb.on('field', (name, val, info) => {
|
||||||
|
results.push({ type: 'field', name, val, info });
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('file', (name, stream, info) => {
|
||||||
|
results.push({ type: 'file', name, info });
|
||||||
|
// Simulate a pipe where the destination is pausing (perhaps due to waiting
|
||||||
|
// for file system write to finish)
|
||||||
|
setTimeout(() => {
|
||||||
|
stream.resume();
|
||||||
|
}, 10);
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('close', mustCall(() => {
|
||||||
|
assert.deepStrictEqual(
|
||||||
|
results,
|
||||||
|
expected,
|
||||||
|
'Results mismatch.\n'
|
||||||
|
+ `Parsed: ${inspect(results)}\n`
|
||||||
|
+ `Expected: ${inspect(expected)}`
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
|
||||||
|
for (const chunk of reqChunks)
|
||||||
|
bb.write(chunk);
|
||||||
|
bb.end();
|
||||||
1053
node_modules/busboy/test/test-types-multipart.js
generated
vendored
Normal file
1053
node_modules/busboy/test/test-types-multipart.js
generated
vendored
Normal file
File diff suppressed because it is too large
Load diff
488
node_modules/busboy/test/test-types-urlencoded.js
generated
vendored
Normal file
488
node_modules/busboy/test/test-types-urlencoded.js
generated
vendored
Normal file
|
|
@ -0,0 +1,488 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
const { transcode } = require('buffer');
|
||||||
|
const { inspect } = require('util');
|
||||||
|
|
||||||
|
const busboy = require('..');
|
||||||
|
|
||||||
|
const active = new Map();
|
||||||
|
|
||||||
|
const tests = [
|
||||||
|
{ source: ['foo'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Unassigned value'
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Assigned value'
|
||||||
|
},
|
||||||
|
{ source: ['foo&bar=baz'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['bar',
|
||||||
|
'baz',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Unassigned and assigned value'
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['baz',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Assigned and unassigned value'
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['baz',
|
||||||
|
'bla',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Two assigned values'
|
||||||
|
},
|
||||||
|
{ source: ['foo&bar'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['bar',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Two unassigned values'
|
||||||
|
},
|
||||||
|
{ source: ['foo&bar&'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['bar',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Two unassigned values and ampersand'
|
||||||
|
},
|
||||||
|
{ source: ['foo+1=bar+baz%2Bquux'],
|
||||||
|
expected: [
|
||||||
|
['foo 1',
|
||||||
|
'bar baz+quux',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Assigned key and value with (plus) space'
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar%20baz%21'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar baz!',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Assigned value with encoded bytes'
|
||||||
|
},
|
||||||
|
{ source: ['foo%20bar=baz%20bla%21'],
|
||||||
|
expected: [
|
||||||
|
['foo bar',
|
||||||
|
'baz bla!',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Assigned value with encoded bytes #2'
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar%20baz%21&num=1000'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar baz!',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['num',
|
||||||
|
'1000',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Two assigned values, one with encoded bytes'
|
||||||
|
},
|
||||||
|
{ source: [
|
||||||
|
Array.from(transcode(Buffer.from('foo'), 'utf8', 'utf16le')).map(
|
||||||
|
(n) => `%${n.toString(16).padStart(2, '0')}`
|
||||||
|
).join(''),
|
||||||
|
'=',
|
||||||
|
Array.from(transcode(Buffer.from('😀!'), 'utf8', 'utf16le')).map(
|
||||||
|
(n) => `%${n.toString(16).padStart(2, '0')}`
|
||||||
|
).join(''),
|
||||||
|
],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'😀!',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'UTF-16LE',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
charset: 'UTF-16LE',
|
||||||
|
what: 'Encoded value with multi-byte charset'
|
||||||
|
},
|
||||||
|
{ source: [
|
||||||
|
'foo=<',
|
||||||
|
Array.from(transcode(Buffer.from('©:^þ'), 'utf8', 'latin1')).map(
|
||||||
|
(n) => `%${n.toString(16).padStart(2, '0')}`
|
||||||
|
).join(''),
|
||||||
|
],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'<©:^þ',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'ISO-8859-1',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
charset: 'ISO-8859-1',
|
||||||
|
what: 'Encoded value with single-byte, ASCII-compatible, non-UTF8 charset'
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [],
|
||||||
|
what: 'Limits: zero fields',
|
||||||
|
limits: { fields: 0 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: one field',
|
||||||
|
limits: { fields: 1 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'bar',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['baz',
|
||||||
|
'bla',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: field part lengths match limits',
|
||||||
|
limits: { fieldNameSize: 3, fieldSize: 3 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['fo',
|
||||||
|
'bar',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['ba',
|
||||||
|
'bla',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: truncated field name',
|
||||||
|
limits: { fieldNameSize: 2 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['foo',
|
||||||
|
'ba',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['baz',
|
||||||
|
'bl',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: truncated field value',
|
||||||
|
limits: { fieldSize: 2 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['fo',
|
||||||
|
'ba',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['ba',
|
||||||
|
'bl',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: truncated field name and value',
|
||||||
|
limits: { fieldNameSize: 2, fieldSize: 2 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['fo',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['ba',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: truncated field name and zero value limit',
|
||||||
|
limits: { fieldNameSize: 2, fieldSize: 0 }
|
||||||
|
},
|
||||||
|
{ source: ['foo=bar&baz=bla'],
|
||||||
|
expected: [
|
||||||
|
['',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
['',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: true,
|
||||||
|
valueTruncated: true,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Limits: truncated zero field name and zero value limit',
|
||||||
|
limits: { fieldNameSize: 0, fieldSize: 0 }
|
||||||
|
},
|
||||||
|
{ source: ['&'],
|
||||||
|
expected: [],
|
||||||
|
what: 'Ampersand'
|
||||||
|
},
|
||||||
|
{ source: ['&&&&&'],
|
||||||
|
expected: [],
|
||||||
|
what: 'Many ampersands'
|
||||||
|
},
|
||||||
|
{ source: ['='],
|
||||||
|
expected: [
|
||||||
|
['',
|
||||||
|
'',
|
||||||
|
{ nameTruncated: false,
|
||||||
|
valueTruncated: false,
|
||||||
|
encoding: 'utf-8',
|
||||||
|
mimeType: 'text/plain' },
|
||||||
|
],
|
||||||
|
],
|
||||||
|
what: 'Assigned value, empty name and value'
|
||||||
|
},
|
||||||
|
{ source: [''],
|
||||||
|
expected: [],
|
||||||
|
what: 'Nothing'
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const test of tests) {
|
||||||
|
active.set(test, 1);
|
||||||
|
|
||||||
|
const { what } = test;
|
||||||
|
const charset = test.charset || 'utf-8';
|
||||||
|
const bb = busboy({
|
||||||
|
limits: test.limits,
|
||||||
|
headers: {
|
||||||
|
'content-type': `application/x-www-form-urlencoded; charset=${charset}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const results = [];
|
||||||
|
|
||||||
|
bb.on('field', (key, val, info) => {
|
||||||
|
results.push([key, val, info]);
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('file', () => {
|
||||||
|
throw new Error(`[${what}] Unexpected file`);
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('close', () => {
|
||||||
|
active.delete(test);
|
||||||
|
|
||||||
|
assert.deepStrictEqual(
|
||||||
|
results,
|
||||||
|
test.expected,
|
||||||
|
`[${what}] Results mismatch.\n`
|
||||||
|
+ `Parsed: ${inspect(results)}\n`
|
||||||
|
+ `Expected: ${inspect(test.expected)}`
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const src of test.source) {
|
||||||
|
const buf = (typeof src === 'string' ? Buffer.from(src, 'utf8') : src);
|
||||||
|
bb.write(buf);
|
||||||
|
}
|
||||||
|
bb.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Byte-by-byte versions
|
||||||
|
for (let test of tests) {
|
||||||
|
test = { ...test };
|
||||||
|
test.what += ' (byte-by-byte)';
|
||||||
|
active.set(test, 1);
|
||||||
|
|
||||||
|
const { what } = test;
|
||||||
|
const charset = test.charset || 'utf-8';
|
||||||
|
const bb = busboy({
|
||||||
|
limits: test.limits,
|
||||||
|
headers: {
|
||||||
|
'content-type': `application/x-www-form-urlencoded; charset="${charset}"`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const results = [];
|
||||||
|
|
||||||
|
bb.on('field', (key, val, info) => {
|
||||||
|
results.push([key, val, info]);
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('file', () => {
|
||||||
|
throw new Error(`[${what}] Unexpected file`);
|
||||||
|
});
|
||||||
|
|
||||||
|
bb.on('close', () => {
|
||||||
|
active.delete(test);
|
||||||
|
|
||||||
|
assert.deepStrictEqual(
|
||||||
|
results,
|
||||||
|
test.expected,
|
||||||
|
`[${what}] Results mismatch.\n`
|
||||||
|
+ `Parsed: ${inspect(results)}\n`
|
||||||
|
+ `Expected: ${inspect(test.expected)}`
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const src of test.source) {
|
||||||
|
const buf = (typeof src === 'string' ? Buffer.from(src, 'utf8') : src);
|
||||||
|
for (let i = 0; i < buf.length; ++i)
|
||||||
|
bb.write(buf.slice(i, i + 1));
|
||||||
|
}
|
||||||
|
bb.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
let exception = false;
|
||||||
|
process.once('uncaughtException', (ex) => {
|
||||||
|
exception = true;
|
||||||
|
throw ex;
|
||||||
|
});
|
||||||
|
process.on('exit', () => {
|
||||||
|
if (exception || active.size === 0)
|
||||||
|
return;
|
||||||
|
process.exitCode = 1;
|
||||||
|
console.error('==========================');
|
||||||
|
console.error(`${active.size} test(s) did not finish:`);
|
||||||
|
console.error('==========================');
|
||||||
|
console.error(Array.from(active.keys()).map((v) => v.what).join('\n'));
|
||||||
|
});
|
||||||
|
}
|
||||||
20
node_modules/busboy/test/test.js
generated
vendored
Normal file
20
node_modules/busboy/test/test.js
generated
vendored
Normal file
|
|
@ -0,0 +1,20 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const { spawnSync } = require('child_process');
|
||||||
|
const { readdirSync } = require('fs');
|
||||||
|
const { join } = require('path');
|
||||||
|
|
||||||
|
const files = readdirSync(__dirname).sort();
|
||||||
|
for (const filename of files) {
|
||||||
|
if (filename.startsWith('test-')) {
|
||||||
|
const path = join(__dirname, filename);
|
||||||
|
console.log(`> Running ${filename} ...`);
|
||||||
|
const result = spawnSync(`${process.argv0} ${path}`, {
|
||||||
|
shell: true,
|
||||||
|
stdio: 'inherit',
|
||||||
|
windowsHide: true
|
||||||
|
});
|
||||||
|
if (result.status !== 0)
|
||||||
|
process.exitCode = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
97
node_modules/bytes/History.md
generated
vendored
Normal file
97
node_modules/bytes/History.md
generated
vendored
Normal file
|
|
@ -0,0 +1,97 @@
|
||||||
|
3.1.2 / 2022-01-27
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Fix return value for un-parsable strings
|
||||||
|
|
||||||
|
3.1.1 / 2021-11-15
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Fix "thousandsSeparator" incorrecting formatting fractional part
|
||||||
|
|
||||||
|
3.1.0 / 2019-01-22
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Add petabyte (`pb`) support
|
||||||
|
|
||||||
|
3.0.0 / 2017-08-31
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Change "kB" to "KB" in format output
|
||||||
|
* Remove support for Node.js 0.6
|
||||||
|
* Remove support for ComponentJS
|
||||||
|
|
||||||
|
2.5.0 / 2017-03-24
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Add option "unit"
|
||||||
|
|
||||||
|
2.4.0 / 2016-06-01
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Add option "unitSeparator"
|
||||||
|
|
||||||
|
2.3.0 / 2016-02-15
|
||||||
|
==================
|
||||||
|
|
||||||
|
* Drop partial bytes on all parsed units
|
||||||
|
* Fix non-finite numbers to `.format` to return `null`
|
||||||
|
* Fix parsing byte string that looks like hex
|
||||||
|
* perf: hoist regular expressions
|
||||||
|
|
||||||
|
2.2.0 / 2015-11-13
|
||||||
|
==================
|
||||||
|
|
||||||
|
* add option "decimalPlaces"
|
||||||
|
* add option "fixedDecimals"
|
||||||
|
|
||||||
|
2.1.0 / 2015-05-21
|
||||||
|
==================
|
||||||
|
|
||||||
|
* add `.format` export
|
||||||
|
* add `.parse` export
|
||||||
|
|
||||||
|
2.0.2 / 2015-05-20
|
||||||
|
==================
|
||||||
|
|
||||||
|
* remove map recreation
|
||||||
|
* remove unnecessary object construction
|
||||||
|
|
||||||
|
2.0.1 / 2015-05-07
|
||||||
|
==================
|
||||||
|
|
||||||
|
* fix browserify require
|
||||||
|
* remove node.extend dependency
|
||||||
|
|
||||||
|
2.0.0 / 2015-04-12
|
||||||
|
==================
|
||||||
|
|
||||||
|
* add option "case"
|
||||||
|
* add option "thousandsSeparator"
|
||||||
|
* return "null" on invalid parse input
|
||||||
|
* support proper round-trip: bytes(bytes(num)) === num
|
||||||
|
* units no longer case sensitive when parsing
|
||||||
|
|
||||||
|
1.0.0 / 2014-05-05
|
||||||
|
==================
|
||||||
|
|
||||||
|
* add negative support. fixes #6
|
||||||
|
|
||||||
|
0.3.0 / 2014-03-19
|
||||||
|
==================
|
||||||
|
|
||||||
|
* added terabyte support
|
||||||
|
|
||||||
|
0.2.1 / 2013-04-01
|
||||||
|
==================
|
||||||
|
|
||||||
|
* add .component
|
||||||
|
|
||||||
|
0.2.0 / 2012-10-28
|
||||||
|
==================
|
||||||
|
|
||||||
|
* bytes(200).should.eql('200b')
|
||||||
|
|
||||||
|
0.1.0 / 2012-07-04
|
||||||
|
==================
|
||||||
|
|
||||||
|
* add bytes to string conversion [yields]
|
||||||
23
node_modules/bytes/LICENSE
generated
vendored
Normal file
23
node_modules/bytes/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
(The MIT License)
|
||||||
|
|
||||||
|
Copyright (c) 2012-2014 TJ Holowaychuk <tj@vision-media.ca>
|
||||||
|
Copyright (c) 2015 Jed Watson <jed.watson@me.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
'Software'), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
||||||
|
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
||||||
|
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
||||||
|
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
152
node_modules/bytes/Readme.md
generated
vendored
Normal file
152
node_modules/bytes/Readme.md
generated
vendored
Normal file
|
|
@ -0,0 +1,152 @@
|
||||||
|
# Bytes utility
|
||||||
|
|
||||||
|
[![NPM Version][npm-image]][npm-url]
|
||||||
|
[![NPM Downloads][downloads-image]][downloads-url]
|
||||||
|
[![Build Status][ci-image]][ci-url]
|
||||||
|
[![Test Coverage][coveralls-image]][coveralls-url]
|
||||||
|
|
||||||
|
Utility to parse a string bytes (ex: `1TB`) to bytes (`1099511627776`) and vice-versa.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
This is a [Node.js](https://nodejs.org/en/) module available through the
|
||||||
|
[npm registry](https://www.npmjs.com/). Installation is done using the
|
||||||
|
[`npm install` command](https://docs.npmjs.com/getting-started/installing-npm-packages-locally):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ npm install bytes
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
```js
|
||||||
|
var bytes = require('bytes');
|
||||||
|
```
|
||||||
|
|
||||||
|
#### bytes(number|string value, [options]): number|string|null
|
||||||
|
|
||||||
|
Default export function. Delegates to either `bytes.format` or `bytes.parse` based on the type of `value`.
|
||||||
|
|
||||||
|
**Arguments**
|
||||||
|
|
||||||
|
| Name | Type | Description |
|
||||||
|
|---------|----------|--------------------|
|
||||||
|
| value | `number`|`string` | Number value to format or string value to parse |
|
||||||
|
| options | `Object` | Conversion options for `format` |
|
||||||
|
|
||||||
|
**Returns**
|
||||||
|
|
||||||
|
| Name | Type | Description |
|
||||||
|
|---------|------------------|-------------------------------------------------|
|
||||||
|
| results | `string`|`number`|`null` | Return null upon error. Numeric value in bytes, or string value otherwise. |
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```js
|
||||||
|
bytes(1024);
|
||||||
|
// output: '1KB'
|
||||||
|
|
||||||
|
bytes('1KB');
|
||||||
|
// output: 1024
|
||||||
|
```
|
||||||
|
|
||||||
|
#### bytes.format(number value, [options]): string|null
|
||||||
|
|
||||||
|
Format the given value in bytes into a string. If the value is negative, it is kept as such. If it is a float, it is
|
||||||
|
rounded.
|
||||||
|
|
||||||
|
**Arguments**
|
||||||
|
|
||||||
|
| Name | Type | Description |
|
||||||
|
|---------|----------|--------------------|
|
||||||
|
| value | `number` | Value in bytes |
|
||||||
|
| options | `Object` | Conversion options |
|
||||||
|
|
||||||
|
**Options**
|
||||||
|
|
||||||
|
| Property | Type | Description |
|
||||||
|
|-------------------|--------|-----------------------------------------------------------------------------------------|
|
||||||
|
| decimalPlaces | `number`|`null` | Maximum number of decimal places to include in output. Default value to `2`. |
|
||||||
|
| fixedDecimals | `boolean`|`null` | Whether to always display the maximum number of decimal places. Default value to `false` |
|
||||||
|
| thousandsSeparator | `string`|`null` | Example of values: `' '`, `','` and `'.'`... Default value to `''`. |
|
||||||
|
| unit | `string`|`null` | The unit in which the result will be returned (B/KB/MB/GB/TB). Default value to `''` (which means auto detect). |
|
||||||
|
| unitSeparator | `string`|`null` | Separator to use between number and unit. Default value to `''`. |
|
||||||
|
|
||||||
|
**Returns**
|
||||||
|
|
||||||
|
| Name | Type | Description |
|
||||||
|
|---------|------------------|-------------------------------------------------|
|
||||||
|
| results | `string`|`null` | Return null upon error. String value otherwise. |
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```js
|
||||||
|
bytes.format(1024);
|
||||||
|
// output: '1KB'
|
||||||
|
|
||||||
|
bytes.format(1000);
|
||||||
|
// output: '1000B'
|
||||||
|
|
||||||
|
bytes.format(1000, {thousandsSeparator: ' '});
|
||||||
|
// output: '1 000B'
|
||||||
|
|
||||||
|
bytes.format(1024 * 1.7, {decimalPlaces: 0});
|
||||||
|
// output: '2KB'
|
||||||
|
|
||||||
|
bytes.format(1024, {unitSeparator: ' '});
|
||||||
|
// output: '1 KB'
|
||||||
|
```
|
||||||
|
|
||||||
|
#### bytes.parse(string|number value): number|null
|
||||||
|
|
||||||
|
Parse the string value into an integer in bytes. If no unit is given, or `value`
|
||||||
|
is a number, it is assumed the value is in bytes.
|
||||||
|
|
||||||
|
Supported units and abbreviations are as follows and are case-insensitive:
|
||||||
|
|
||||||
|
* `b` for bytes
|
||||||
|
* `kb` for kilobytes
|
||||||
|
* `mb` for megabytes
|
||||||
|
* `gb` for gigabytes
|
||||||
|
* `tb` for terabytes
|
||||||
|
* `pb` for petabytes
|
||||||
|
|
||||||
|
The units are in powers of two, not ten. This means 1kb = 1024b according to this parser.
|
||||||
|
|
||||||
|
**Arguments**
|
||||||
|
|
||||||
|
| Name | Type | Description |
|
||||||
|
|---------------|--------|--------------------|
|
||||||
|
| value | `string`|`number` | String to parse, or number in bytes. |
|
||||||
|
|
||||||
|
**Returns**
|
||||||
|
|
||||||
|
| Name | Type | Description |
|
||||||
|
|---------|-------------|-------------------------|
|
||||||
|
| results | `number`|`null` | Return null upon error. Value in bytes otherwise. |
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```js
|
||||||
|
bytes.parse('1KB');
|
||||||
|
// output: 1024
|
||||||
|
|
||||||
|
bytes.parse('1024');
|
||||||
|
// output: 1024
|
||||||
|
|
||||||
|
bytes.parse(1024);
|
||||||
|
// output: 1024
|
||||||
|
```
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
[MIT](LICENSE)
|
||||||
|
|
||||||
|
[ci-image]: https://badgen.net/github/checks/visionmedia/bytes.js/master?label=ci
|
||||||
|
[ci-url]: https://github.com/visionmedia/bytes.js/actions?query=workflow%3Aci
|
||||||
|
[coveralls-image]: https://badgen.net/coveralls/c/github/visionmedia/bytes.js/master
|
||||||
|
[coveralls-url]: https://coveralls.io/r/visionmedia/bytes.js?branch=master
|
||||||
|
[downloads-image]: https://badgen.net/npm/dm/bytes
|
||||||
|
[downloads-url]: https://npmjs.org/package/bytes
|
||||||
|
[npm-image]: https://badgen.net/npm/v/bytes
|
||||||
|
[npm-url]: https://npmjs.org/package/bytes
|
||||||
170
node_modules/bytes/index.js
generated
vendored
Normal file
170
node_modules/bytes/index.js
generated
vendored
Normal file
|
|
@ -0,0 +1,170 @@
|
||||||
|
/*!
|
||||||
|
* bytes
|
||||||
|
* Copyright(c) 2012-2014 TJ Holowaychuk
|
||||||
|
* Copyright(c) 2015 Jed Watson
|
||||||
|
* MIT Licensed
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module exports.
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
module.exports = bytes;
|
||||||
|
module.exports.format = format;
|
||||||
|
module.exports.parse = parse;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Module variables.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
|
||||||
|
var formatThousandsRegExp = /\B(?=(\d{3})+(?!\d))/g;
|
||||||
|
|
||||||
|
var formatDecimalsRegExp = /(?:\.0*|(\.[^0]+)0+)$/;
|
||||||
|
|
||||||
|
var map = {
|
||||||
|
b: 1,
|
||||||
|
kb: 1 << 10,
|
||||||
|
mb: 1 << 20,
|
||||||
|
gb: 1 << 30,
|
||||||
|
tb: Math.pow(1024, 4),
|
||||||
|
pb: Math.pow(1024, 5),
|
||||||
|
};
|
||||||
|
|
||||||
|
var parseRegExp = /^((-|\+)?(\d+(?:\.\d+)?)) *(kb|mb|gb|tb|pb)$/i;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert the given value in bytes into a string or parse to string to an integer in bytes.
|
||||||
|
*
|
||||||
|
* @param {string|number} value
|
||||||
|
* @param {{
|
||||||
|
* case: [string],
|
||||||
|
* decimalPlaces: [number]
|
||||||
|
* fixedDecimals: [boolean]
|
||||||
|
* thousandsSeparator: [string]
|
||||||
|
* unitSeparator: [string]
|
||||||
|
* }} [options] bytes options.
|
||||||
|
*
|
||||||
|
* @returns {string|number|null}
|
||||||
|
*/
|
||||||
|
|
||||||
|
function bytes(value, options) {
|
||||||
|
if (typeof value === 'string') {
|
||||||
|
return parse(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === 'number') {
|
||||||
|
return format(value, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Format the given value in bytes into a string.
|
||||||
|
*
|
||||||
|
* If the value is negative, it is kept as such. If it is a float,
|
||||||
|
* it is rounded.
|
||||||
|
*
|
||||||
|
* @param {number} value
|
||||||
|
* @param {object} [options]
|
||||||
|
* @param {number} [options.decimalPlaces=2]
|
||||||
|
* @param {number} [options.fixedDecimals=false]
|
||||||
|
* @param {string} [options.thousandsSeparator=]
|
||||||
|
* @param {string} [options.unit=]
|
||||||
|
* @param {string} [options.unitSeparator=]
|
||||||
|
*
|
||||||
|
* @returns {string|null}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
function format(value, options) {
|
||||||
|
if (!Number.isFinite(value)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var mag = Math.abs(value);
|
||||||
|
var thousandsSeparator = (options && options.thousandsSeparator) || '';
|
||||||
|
var unitSeparator = (options && options.unitSeparator) || '';
|
||||||
|
var decimalPlaces = (options && options.decimalPlaces !== undefined) ? options.decimalPlaces : 2;
|
||||||
|
var fixedDecimals = Boolean(options && options.fixedDecimals);
|
||||||
|
var unit = (options && options.unit) || '';
|
||||||
|
|
||||||
|
if (!unit || !map[unit.toLowerCase()]) {
|
||||||
|
if (mag >= map.pb) {
|
||||||
|
unit = 'PB';
|
||||||
|
} else if (mag >= map.tb) {
|
||||||
|
unit = 'TB';
|
||||||
|
} else if (mag >= map.gb) {
|
||||||
|
unit = 'GB';
|
||||||
|
} else if (mag >= map.mb) {
|
||||||
|
unit = 'MB';
|
||||||
|
} else if (mag >= map.kb) {
|
||||||
|
unit = 'KB';
|
||||||
|
} else {
|
||||||
|
unit = 'B';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var val = value / map[unit.toLowerCase()];
|
||||||
|
var str = val.toFixed(decimalPlaces);
|
||||||
|
|
||||||
|
if (!fixedDecimals) {
|
||||||
|
str = str.replace(formatDecimalsRegExp, '$1');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (thousandsSeparator) {
|
||||||
|
str = str.split('.').map(function (s, i) {
|
||||||
|
return i === 0
|
||||||
|
? s.replace(formatThousandsRegExp, thousandsSeparator)
|
||||||
|
: s
|
||||||
|
}).join('.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return str + unitSeparator + unit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse the string value into an integer in bytes.
|
||||||
|
*
|
||||||
|
* If no unit is given, it is assumed the value is in bytes.
|
||||||
|
*
|
||||||
|
* @param {number|string} val
|
||||||
|
*
|
||||||
|
* @returns {number|null}
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
|
||||||
|
function parse(val) {
|
||||||
|
if (typeof val === 'number' && !isNaN(val)) {
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof val !== 'string') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test if the string passed is valid
|
||||||
|
var results = parseRegExp.exec(val);
|
||||||
|
var floatValue;
|
||||||
|
var unit = 'b';
|
||||||
|
|
||||||
|
if (!results) {
|
||||||
|
// Nothing could be extracted from the given string
|
||||||
|
floatValue = parseInt(val, 10);
|
||||||
|
unit = 'b'
|
||||||
|
} else {
|
||||||
|
// Retrieve the value and the unit
|
||||||
|
floatValue = parseFloat(results[1]);
|
||||||
|
unit = results[4].toLowerCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isNaN(floatValue)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Math.floor(map[unit] * floatValue);
|
||||||
|
}
|
||||||
42
node_modules/bytes/package.json
generated
vendored
Normal file
42
node_modules/bytes/package.json
generated
vendored
Normal file
|
|
@ -0,0 +1,42 @@
|
||||||
|
{
|
||||||
|
"name": "bytes",
|
||||||
|
"description": "Utility to parse a string bytes to bytes and vice-versa",
|
||||||
|
"version": "3.1.2",
|
||||||
|
"author": "TJ Holowaychuk <tj@vision-media.ca> (http://tjholowaychuk.com)",
|
||||||
|
"contributors": [
|
||||||
|
"Jed Watson <jed.watson@me.com>",
|
||||||
|
"Théo FIDRY <theo.fidry@gmail.com>"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"keywords": [
|
||||||
|
"byte",
|
||||||
|
"bytes",
|
||||||
|
"utility",
|
||||||
|
"parse",
|
||||||
|
"parser",
|
||||||
|
"convert",
|
||||||
|
"converter"
|
||||||
|
],
|
||||||
|
"repository": "visionmedia/bytes.js",
|
||||||
|
"devDependencies": {
|
||||||
|
"eslint": "7.32.0",
|
||||||
|
"eslint-plugin-markdown": "2.2.1",
|
||||||
|
"mocha": "9.2.0",
|
||||||
|
"nyc": "15.1.0"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"History.md",
|
||||||
|
"LICENSE",
|
||||||
|
"Readme.md",
|
||||||
|
"index.js"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"lint": "eslint .",
|
||||||
|
"test": "mocha --check-leaks --reporter spec",
|
||||||
|
"test-ci": "nyc --reporter=lcov --reporter=text npm test",
|
||||||
|
"test-cov": "nyc --reporter=html --reporter=text npm test"
|
||||||
|
}
|
||||||
|
}
|
||||||
17
node_modules/call-bind-apply-helpers/.eslintrc
generated
vendored
Normal file
17
node_modules/call-bind-apply-helpers/.eslintrc
generated
vendored
Normal file
|
|
@ -0,0 +1,17 @@
|
||||||
|
{
|
||||||
|
"root": true,
|
||||||
|
|
||||||
|
"extends": "@ljharb",
|
||||||
|
|
||||||
|
"rules": {
|
||||||
|
"func-name-matching": 0,
|
||||||
|
"id-length": 0,
|
||||||
|
"new-cap": [2, {
|
||||||
|
"capIsNewExceptions": [
|
||||||
|
"GetIntrinsic",
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
"no-extra-parens": 0,
|
||||||
|
"no-magic-numbers": 0,
|
||||||
|
},
|
||||||
|
}
|
||||||
12
node_modules/call-bind-apply-helpers/.github/FUNDING.yml
generated
vendored
Normal file
12
node_modules/call-bind-apply-helpers/.github/FUNDING.yml
generated
vendored
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: [ljharb]
|
||||||
|
patreon: # Replace with a single Patreon username
|
||||||
|
open_collective: # Replace with a single Open Collective username
|
||||||
|
ko_fi: # Replace with a single Ko-fi username
|
||||||
|
tidelift: npm/call-bind-apply-helpers
|
||||||
|
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
||||||
|
liberapay: # Replace with a single Liberapay username
|
||||||
|
issuehunt: # Replace with a single IssueHunt username
|
||||||
|
otechie: # Replace with a single Otechie username
|
||||||
|
custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2']
|
||||||
9
node_modules/call-bind-apply-helpers/.nycrc
generated
vendored
Normal file
9
node_modules/call-bind-apply-helpers/.nycrc
generated
vendored
Normal file
|
|
@ -0,0 +1,9 @@
|
||||||
|
{
|
||||||
|
"all": true,
|
||||||
|
"check-coverage": false,
|
||||||
|
"reporter": ["text-summary", "text", "html", "json"],
|
||||||
|
"exclude": [
|
||||||
|
"coverage",
|
||||||
|
"test"
|
||||||
|
]
|
||||||
|
}
|
||||||
30
node_modules/call-bind-apply-helpers/CHANGELOG.md
generated
vendored
Normal file
30
node_modules/call-bind-apply-helpers/CHANGELOG.md
generated
vendored
Normal file
|
|
@ -0,0 +1,30 @@
|
||||||
|
# Changelog
|
||||||
|
|
||||||
|
All notable changes to this project will be documented in this file.
|
||||||
|
|
||||||
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/)
|
||||||
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [v1.0.2](https://github.com/ljharb/call-bind-apply-helpers/compare/v1.0.1...v1.0.2) - 2025-02-12
|
||||||
|
|
||||||
|
### Commits
|
||||||
|
|
||||||
|
- [types] improve inferred types [`e6f9586`](https://github.com/ljharb/call-bind-apply-helpers/commit/e6f95860a3c72879cb861a858cdfb8138fbedec1)
|
||||||
|
- [Dev Deps] update `@arethetypeswrong/cli`, `@ljharb/tsconfig`, `@types/tape`, `es-value-fixtures`, `for-each`, `has-strict-mode`, `object-inspect` [`e43d540`](https://github.com/ljharb/call-bind-apply-helpers/commit/e43d5409f97543bfbb11f345d47d8ce4e066d8c1)
|
||||||
|
|
||||||
|
## [v1.0.1](https://github.com/ljharb/call-bind-apply-helpers/compare/v1.0.0...v1.0.1) - 2024-12-08
|
||||||
|
|
||||||
|
### Commits
|
||||||
|
|
||||||
|
- [types] `reflectApply`: fix types [`4efc396`](https://github.com/ljharb/call-bind-apply-helpers/commit/4efc3965351a4f02cc55e836fa391d3d11ef2ef8)
|
||||||
|
- [Fix] `reflectApply`: oops, Reflect is not a function [`83cc739`](https://github.com/ljharb/call-bind-apply-helpers/commit/83cc7395de6b79b7730bdf092f1436f0b1263c75)
|
||||||
|
- [Dev Deps] update `@arethetypeswrong/cli` [`80bd5d3`](https://github.com/ljharb/call-bind-apply-helpers/commit/80bd5d3ae58b4f6b6995ce439dd5a1bcb178a940)
|
||||||
|
|
||||||
|
## v1.0.0 - 2024-12-05
|
||||||
|
|
||||||
|
### Commits
|
||||||
|
|
||||||
|
- Initial implementation, tests, readme [`7879629`](https://github.com/ljharb/call-bind-apply-helpers/commit/78796290f9b7430c9934d6f33d94ae9bc89fce04)
|
||||||
|
- Initial commit [`3f1dc16`](https://github.com/ljharb/call-bind-apply-helpers/commit/3f1dc164afc43285631b114a5f9dd9137b2b952f)
|
||||||
|
- npm init [`081df04`](https://github.com/ljharb/call-bind-apply-helpers/commit/081df048c312fcee400922026f6e97281200a603)
|
||||||
|
- Only apps should have lockfiles [`5b9ca0f`](https://github.com/ljharb/call-bind-apply-helpers/commit/5b9ca0fe8101ebfaf309c549caac4e0a017ed930)
|
||||||
21
node_modules/call-bind-apply-helpers/LICENSE
generated
vendored
Normal file
21
node_modules/call-bind-apply-helpers/LICENSE
generated
vendored
Normal file
|
|
@ -0,0 +1,21 @@
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2024 Jordan Harband
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
62
node_modules/call-bind-apply-helpers/README.md
generated
vendored
Normal file
62
node_modules/call-bind-apply-helpers/README.md
generated
vendored
Normal file
|
|
@ -0,0 +1,62 @@
|
||||||
|
# call-bind-apply-helpers <sup>[![Version Badge][npm-version-svg]][package-url]</sup>
|
||||||
|
|
||||||
|
[![github actions][actions-image]][actions-url]
|
||||||
|
[![coverage][codecov-image]][codecov-url]
|
||||||
|
[![dependency status][deps-svg]][deps-url]
|
||||||
|
[![dev dependency status][dev-deps-svg]][dev-deps-url]
|
||||||
|
[![License][license-image]][license-url]
|
||||||
|
[![Downloads][downloads-image]][downloads-url]
|
||||||
|
|
||||||
|
[![npm badge][npm-badge-png]][package-url]
|
||||||
|
|
||||||
|
Helper functions around Function call/apply/bind, for use in `call-bind`.
|
||||||
|
|
||||||
|
The only packages that should likely ever use this package directly are `call-bind` and `get-intrinsic`.
|
||||||
|
Please use `call-bind` unless you have a very good reason not to.
|
||||||
|
|
||||||
|
## Getting started
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm install --save call-bind-apply-helpers
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage/Examples
|
||||||
|
|
||||||
|
```js
|
||||||
|
const assert = require('assert');
|
||||||
|
const callBindBasic = require('call-bind-apply-helpers');
|
||||||
|
|
||||||
|
function f(a, b) {
|
||||||
|
assert.equal(this, 1);
|
||||||
|
assert.equal(a, 2);
|
||||||
|
assert.equal(b, 3);
|
||||||
|
assert.equal(arguments.length, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
const fBound = callBindBasic([f, 1]);
|
||||||
|
|
||||||
|
delete Function.prototype.call;
|
||||||
|
delete Function.prototype.bind;
|
||||||
|
|
||||||
|
fBound(2, 3);
|
||||||
|
```
|
||||||
|
|
||||||
|
## Tests
|
||||||
|
|
||||||
|
Clone the repo, `npm install`, and run `npm test`
|
||||||
|
|
||||||
|
[package-url]: https://npmjs.org/package/call-bind-apply-helpers
|
||||||
|
[npm-version-svg]: https://versionbadg.es/ljharb/call-bind-apply-helpers.svg
|
||||||
|
[deps-svg]: https://david-dm.org/ljharb/call-bind-apply-helpers.svg
|
||||||
|
[deps-url]: https://david-dm.org/ljharb/call-bind-apply-helpers
|
||||||
|
[dev-deps-svg]: https://david-dm.org/ljharb/call-bind-apply-helpers/dev-status.svg
|
||||||
|
[dev-deps-url]: https://david-dm.org/ljharb/call-bind-apply-helpers#info=devDependencies
|
||||||
|
[npm-badge-png]: https://nodei.co/npm/call-bind-apply-helpers.png?downloads=true&stars=true
|
||||||
|
[license-image]: https://img.shields.io/npm/l/call-bind-apply-helpers.svg
|
||||||
|
[license-url]: LICENSE
|
||||||
|
[downloads-image]: https://img.shields.io/npm/dm/call-bind-apply-helpers.svg
|
||||||
|
[downloads-url]: https://npm-stat.com/charts.html?package=call-bind-apply-helpers
|
||||||
|
[codecov-image]: https://codecov.io/gh/ljharb/call-bind-apply-helpers/branch/main/graphs/badge.svg
|
||||||
|
[codecov-url]: https://app.codecov.io/gh/ljharb/call-bind-apply-helpers/
|
||||||
|
[actions-image]: https://img.shields.io/endpoint?url=https://github-actions-badge-u3jn4tfpocch.runkit.sh/ljharb/call-bind-apply-helpers
|
||||||
|
[actions-url]: https://github.com/ljharb/call-bind-apply-helpers/actions
|
||||||
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue