commit b5c43a96638be1c62840a9536385670e2533972f Author: oddbyte Date: Sun Jan 18 13:22:41 2026 -0500 Initial commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..3761dbd --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +node_modules/ +private/ +*/private/ +package-lock.json diff --git a/blog/articles/ageverification.html b/blog/articles/ageverification.html new file mode 100644 index 0000000..a80511a --- /dev/null +++ b/blog/articles/ageverification.html @@ -0,0 +1,20 @@ +

These Laws Will Make Your Government ID Public

+

+ Imagine if every website had a bouncer that demanded your ID, took a photo of it, and stored it in an online database that could be hacked at any moment. Unfortunately, this is now the reality for half of the United States. Half of the states in the US now require you to hand over your government ID or scan your face to access sites containing anything deemed "sexual content" (this sometimes includes things that are not explicitly sexual, such as health information for LGBTQ+ people and similar topics, usually labeled as content that is "harmful to minors"). A handful of states also require you to hand over such identification when signing up for social media websites. While these laws are aimed at protecting children, they end up endangering them by stripping them of their digital anonymity and creating treasure troves of their personal information for hackers to access.
+

+ Your government ID is a "keys to the kingdom" asset for hackers to have access to. With access to your government ID they can contact companies and request they change your password, sign up for online accounts as you, and even ruin your credit score. By requiring everyone online to upload their valid government identification to a multitude of for-profit companies, these laws are putting this critical information at high risk of being hacked, breached, or otherwise exposed.
+

+ Age verification may sound simple and easy to enforce: if the person is viewing a pornography website, ask for proof that they are indeed overage before allowing them to continue. However, these laws have dangerous consequences unforeseen by legislators, such as deanonymizing the internet, allowing hackers to access sensitive government identification for millions of people, and even worse, putting the very children the laws are supposed to protect at risk via data breaches.
+

+

+ On top of the security risks, these laws have seen some reputable VPN companies receiving a 1400% spike in signups, showing that the laws are likely not even effective on any child that has the technical know-how to install a VPN app off of the app store (which is something that many children are already doing anyway in order to avoid school internet blocks).
+

+

+ Some proponents of age verification laws say that it could solve the dead internet theory, however even they admit these laws are problematic, stating "I want to make something very clear: Online age and ID verification has a number of problems. Very, very real problems that every single person is right to be concerned about. What verification data will be collected and stored (and how)? What additional security concerns are created because of ID verification? Will the burden of that verification be too much for some sites to handle? How will those verification systems be abused by corporations and governments? And those are just off the top of my head." (A quote from Brian Lunduke of the Lunduke Journal). An issue with this argument is that adding ID verification will not stop bots. The bots will just pretend to sign up from a jurisdiction without age verification, or just leverage the troves of stolen IDs publicly available online for anyone to access.
+

+

+ Another issue with age verification is how some states include content that they deem "harmful to minors" as something that needs verification. This includes health information for LGBTQ+ people. This is really dangerous because it risks the outing of LGBTQ+ individuals to a community or family where such people may not be welcomed. This could end up in the queer person being kicked out of the community, and in extreme cases even physically harmed.
+

+

+ In conclusion, you should keep yourself safe. If you find yourself in a jurisdiction where age verification is mandated, you should use a VPN or other privacy solution (such as the Tor network) to avoid sending off your personal information. You should also contact your local representatives, and ask them to not allow these laws to ever go through and advocate for laws that actually protect the individuals they target. If you want to have a bigger impact, volunteer for the Electronic Frontier Foundation.
+

\ No newline at end of file diff --git a/blog/testingtopic/verykewlblog.html b/blog/testingtopic/verykewlblog.html new file mode 100644 index 0000000..c4c0089 --- /dev/null +++ b/blog/testingtopic/verykewlblog.html @@ -0,0 +1 @@ +

hallo!!!

\ No newline at end of file diff --git a/blog/testingtopic2/otherkewlblog.html b/blog/testingtopic2/otherkewlblog.html new file mode 100644 index 0000000..157cc52 --- /dev/null +++ b/blog/testingtopic2/otherkewlblog.html @@ -0,0 +1 @@ +

hihi!!!

\ No newline at end of file diff --git a/index.js b/index.js new file mode 100755 index 0000000..e23636b --- /dev/null +++ b/index.js @@ -0,0 +1,651 @@ +// Import all the things +const axios = require('axios'); +const express = require('express'); +const fs = require('node:fs'); +const path = require('node:path'); +const readline = require('node:readline/promises'); +const http = require('node:http'); + +// yes, it is `new new require('ytmusic-api')`, i didnt fuck up +// this is literally black magic i have no idea how or why it works +const ytmusic = new new require('ytmusic-api'); // i love me a new new + +// Create interface for console +const rl = readline.createInterface({input:process.stdin, output:process.stdout}); + +// Global constants + +/** + * Should always be `/prod/portfolio`, dynamically caching it in case i decide to move it to a different path sometime in the future + */ +const baseDir = process.cwd(); + +/** + * The ID for my favorite moosic playlist :D + */ +const playlistId = 'PLnlTMS4cxfx3-et_L8APzpEgy_eCf18-U'; + +/** + * Port that nginx proxies to public + */ +const port = 48915; + +// Helper functions I've made to do things and stuff :P + +/** + * Loops a thing every hour + * @param {Function} something + * @param {Boolean} skipFirst + * @returns jack shit +*/ +// I know, I know, I am just the most bestest and awesomest coder everrrrr +const loopHourly = (something, skipFirst = false) => { + // If skipFirst is false, we do the thing immediately + if (!skipFirst) setTimeout(() => { + let cancel = false; + something(); + if (!cancel) loopHourly(something); + return; + }, 3600000 - new Date().getTime() % 3600000); // 3600000 ms = 1000 ms (one second) * 60 seconds (one minute) * 60 minutes (one hour) + + // If skipFirst is true, wait a hour then execute + else setTimeout(() => { + loopHourly(something) + }, 3600000 - new Date().getTime() % 3600000); +} + +function sendSource(req, res, filePath) { + try { + // This will throw an error, which I catch, if the file isnt found + // Use lstatSync to have a sync read while also not going through symlinks (to prevent arbritrary reads via symlinks) + // Further restricted by my use of the --permission commandline flag + // I also added a manual check to block any file reads outside the base dir (/prod/portfolio as of the time of writing this) + const stat = fs.lstatSync(filePath); + + if (!stat.isFile()) { + // If it's not a file, error + res.status(403).send('
Not a file
'); + } else { + // If it's a file, send the content + const fileContent = fs.readFileSync(filePath, 'utf8'); + res.render('sourceviewer', { faviconb64, filePath, fileContent }); + } + } catch (err) { + // Legit why are you doing this + // If it's whitelisted, it's listed under the index page + // If it's not whitelisted, you cant access it anyway :P + res.status(404).send('File not found'); + } +} + +// -------------------------------------------------------------------------------------- +// - Below is black magic. I have no idea how it works. - +// - I have either stolen it from somewhere or completely forgotten how it works. - +// - If you know how it works please explain it to me PLEASE I BEG OF YOU HALP :AAAAAA: - +// -------------------------------------------------------------------------------------- + +/** + * Pulls all the pictures for the songs and stores them in a base64 key:value array where the key is the song id and the value is the thumbnail + * @param {*} song + */ +const populateThumbnails = (async song => { + let id = song.videoId; + let thumbnail; + + const response = await axios.get(song.thumbnails[song.thumbnails.length - 1].url, { responseType: 'arraybuffer' }); + + // We now check if Youtube moosic has yelled back a thing we want to hear + if (response.status == 200) { + thumbnail = Buffer.from(response.data, "utf-8").toString('base64'); + thumbnails.push({ id, thumbnail }); + } else { + console.error("Failed to pull thumbnail: " + response); + } +}); + +// Updates the list of songs by yelling at youtube moosic to gimme the playlist +async function updateSongs() { + + // Yell at youtube moosic to get a list of songs + playlistSongs = await ytmusic.getPlaylistVideos(playlistId); + + if (typeof playlistSongs == 'object') { + // For each song that youtube moosic has yelled back at us, give them individually to populateThumbnails() + await Promise.all(playlistSongs.map(async song => await populateThumbnails(song))); + } + else { + // like legit this shouldnt even be possible + console.log("somehow i have managed to get this code into an area i thought impossible"); + console.log("*various anxiousness noises*"); + console.log("help"); + throw TypeError('playlistSongs is not a object!'); // this should never ever happen, if it does that is very very bad + } +} + +/** + * @param {Array} pathPattern This thingy is the indexPaths below this function + * @returns {Array} an array called `results` + */ +function getDirectoryContents(pathPattern) { + const results = []; + + // Handle wildcards (including multiple wildcards like blog/*/*) + if (pathPattern.includes('*')) { + const segments = pathPattern.split('/').filter(s => s); + + /** + * Recursively resolves the pattern segments against the file system. + * @param {Array} remainingSegments - The parts of the path left to match (e.g., ['*']). + * @param {String} currentAbsPath - The absolute path on disk we are currently checking. + * @returns {Array} - Array of matched file/directory objects. + */ + function traverse(remainingSegments, currentAbsPath) { + // Base case: We have matched all segments. Return the node at this location. + if (remainingSegments.length === 0) { + return [createNode(currentAbsPath)]; + } + + const segment = remainingSegments[0]; + const rest = remainingSegments.slice(1); + + if (segment === '*') { + // Wildcard: Read current directory and recurse for all children + try { + const files = fs.readdirSync(currentAbsPath, { withFileTypes: true }); + const matches = []; + + files.forEach(file => { + const fullPath = path.join(currentAbsPath, file.name); + // Recurse with the remaining segments + const childMatches = traverse(rest, fullPath); + matches.push(...childMatches); + }); + + return matches; + } catch (err) { + console.error(`Error reading directory ${currentAbsPath}:`, err); + return []; + } + } else { + // Exact segment: Join path and recurse + const fullPath = path.join(currentAbsPath, segment); + return traverse(rest, fullPath); + } + } + + /** + * Creates the result object for a specific path. + * If it's a directory, it recursively fetches children. + */ + function createNode(fullPath) { + try { + const stat = fs.statSync(fullPath); + const relativePath = path.relative(baseDir, fullPath); + + if (stat.isDirectory()) { + // Recursively get children using the main function + // We append '/*' to get the immediate contents of this directory + const children = getDirectoryContents(`${relativePath}/*`); + return { + name: path.basename(fullPath), + path: relativePath, + type: 'directory', + children: children + }; + } else { + return { + name: path.basename(fullPath), + path: relativePath, + type: 'file' + }; + } + } catch (err) { + console.error(`Error reading path ${fullPath}:`, err); + return null; + } + } + + // Start traversal from the base directory + const nodes = traverse(segments, baseDir); + // Filter out any null results from errors + nodes.forEach(node => { + if (node) results.push(node); + }); + + } else { + // Handle exact paths + + // Thanks past me, for the extremely helpful comment above + // I now understand exactly what this does /s + + const fullPath = path.join(baseDir, pathPattern); + try { + const stat = fs.statSync(fullPath); + const relativePath = path.relative(baseDir, fullPath); + + if (stat.isDirectory()) { + const files = fs.readdirSync(fullPath, { withFileTypes: true }); + files.forEach(file => { + const childPath = path.join(fullPath, file.name); + const childRelativePath = path.relative(baseDir, childPath); + + if (file.isDirectory()) { + const children = getDirectoryContents(`${childRelativePath}/*`); + results.push({ + name: file.name, + path: childRelativePath, + type: 'directory', + children: children + }); + } else { + results.push({ + name: file.name, + path: childRelativePath, + type: 'file' + }); + } + }); + } else { + results.push({ + name: path.basename(fullPath), + path: relativePath, + type: 'file' + }); + } + } catch (err) { + console.error(`Error reading path ${fullPath}:`, err); + } + } + + return results; +} + +/** + * An array of the whitelisted directories and their contents + */ +let whitelistedFiles = []; + +/** + * Whitelisted paths for the source code (to prevent arbritrary reads and stuffs), + * Also referred to as `pathPattern` in whatever the hell `getDirectoryContents()` is + */ +const indexPaths = [ + 'blog/*', + 'blog/*/*', + 'public/*', + 'public/piskelfile/*', + 'views/*', + 'views/partials/*', + 'index.js' +]; + +/** + * directory tree object (built in the func buildDirectoryTree below) + */ +let directoryTree; + +/** + * sorted result + */ +let sortedResult; + +// Build directory tree +const buildDirectoryTree = (items) => { + const tree = {}; + items.forEach(item => { + const parts = item.path.split(path.sep).filter(part => part !== ''); + let currentLevel = tree; + for (let i = 0; i < parts.length; i++) { + const part = parts[i]; + if (!currentLevel[part]) { + currentLevel[part] = { + name: part, + path: parts.slice(0, i + 1).join(path.sep), + type: i === parts.length - 1 ? item.type : 'directory', + children: {} + }; + } + currentLevel = currentLevel[part].children; + } + if (item.type === 'file') { + currentLevel[parts[parts.length - 1]] = { + ...currentLevel[parts[parts.length - 1]], + type: 'file' + }; + } + }); + return tree; +}; + +// Convert the tree to an array for easier rendering +const treeToArray = (node) => { + const result = []; + Object.values(node).forEach(item => { + if (item.type === 'directory') { + const directoryItem = { + name: item.name, + path: item.path, + type: 'directory', + children: treeToArray(item.children) + }; + result.push(directoryItem); + } else { + result.push({ + name: item.name, + path: item.path, + type: 'file' + }); + } + }); + return result.sort((a, b) => { + if (a.type === 'directory' && b.type === 'file') return -1; + if (a.type === 'file' && b.type === 'directory') return 1; + return a.name.localeCompare(b.name); + }); +}; + +async function updateFileIndex() { + // Clear whitelisted files to prevent memory leak + whitelistedFiles = []; + + indexPaths.forEach(path => { + let contents = getDirectoryContents(path); + whitelistedFiles.push(...contents); + }); + + directoryTree = buildDirectoryTree(whitelistedFiles); + sortedResult = treeToArray(directoryTree); +} + +// ---------------------------------------------------------- +// - End black magic - +// - I am pretty sure I know what is going on below here :p - +// ---------------------------------------------------------- + +var blog; // Setup the blog variable + +// Blog stuffs + +async function generateBlogIndex() { + var blogfolder = fs.readdirSync(path.join(baseDir, 'blog/')); + var tmpobj = {}; + blogfolder.forEach(thing => { + if (fs.statSync(path.join(baseDir, `blog/${thing}`)).isDirectory()) { + var files = fs.readdirSync(path.join(baseDir, `blog/${thing}`)); + tmpobj[thing] = {}; + files.forEach(file => { + tmpobj[thing][file] = true; + }); + } + }); + blog = tmpobj; +} + +async function send404(req, res) { + // TODO: actual 404 page + res.status(404).send("

404 Not found

"); +} + +// Begin the server-ing things +const app = express(); +var server; // Set at app.listen (bottom), used to kill server on ctrl-c + +// YT Music stuff +/** + * An object that stores a Map of songs (also objects) + */ +let playlistSongs = new Map; +/** + * Key:Value array of songs. The song ID is the key, and a base64 encoded image is the value + */ +let thumbnails = []; + +// Other middlewares +app.set('view engine', 'ejs'); +app.use(express.static('public')); +app.use(express.static('private')); +app.use(express.urlencoded({ extended: true })); +app.use((req, res, next) => { + res.setHeader("Content-Security-Policy","default-src 'self' 'unsafe-inline'; script-src-elem 'self' 'unsafe-inline'; img-src 'self' data:; frame-src 'self';"); + next(); +}); + +// Cache the favicon cause I'm probably not gonna change that for a while +const favicon = fs.readFileSync('public/favicon.ico'); +// Also cache a base64 version of it to pass to my pages +const faviconb64 = favicon.toString('base64'); + +// --------------------- +// - Routes begin here - +// --------------------- + +// Sends the cached favicon.ico to prevent an unneeded read on a static file :P +app.get('/favicon.ico', (req, res) => { + res.status(200).send(favicon); +}); + +// Home page +app.get('/', (req, res) => { + res.render('home', { faviconb64 }); +}); + +// Fun Fact page +app.get('/funfact', (req, res) => { + res.render('funfact', { }); +}); + +// The page with all my art :> +app.get('/art', (req, res) => { + res.render('art', { faviconb64 }); +}); + +// Source code page +app.get('/source', (req, res) => { + res.render('source', { faviconb64, paths: sortedResult }); +}); + +// Route to serve indexed files +app.get(/^\/source\/.*/, (req, res) => { + const requestedPath = req.path.split('/source/')[1]; + + // Check if the requested file is whitelisted + const isIndexed = indexPaths.some(pattern => { + if (pattern.toString().includes("*")) { + return whitelistedFiles.some(pattern => { return requestedPath.startsWith(pattern.path) && !(requestedPath.split(pattern.path).some(pattern => {return pattern.includes("/");}));}); + } + return requestedPath === pattern; + }); + + if (!isIndexed) { + return res.status(403).send('
File not whitelisted
'); + } + + const filePath = path.join(baseDir, requestedPath); + + // Completely prevent ../ attacks (hopefully) + // Well I mean it wont prevent bind mounts but if you can bind mount I'm fucked anyways cause you're root + // Symlinks might (?) bypass this?? + // Hardlinks defo bypass this afaik. + // because i pass the --permission flag and only pass --allow-fs-read=/prod/portfolio/* + // you can't write via NJS at all + // nor read outside /prod/portfolio + // you'd need to break outta nodejs and start a native process + // and my filesystem permissions wont letchu write stuff anyway + if (!filePath.startsWith(baseDir)) { + return res.status(403).send('
File not whitelisted
'); + } + + sendSource(req, res, filePath); +}); + +// Moosic page +app.get('/music', (req, res) => { + res.render('moosic', { faviconb64, playlistId, playlistSongs, thumbnails }); +}); + +// Blog +app.get('/blog', (req, res) => { + if (req.query && req.query.sample == "1") { + res.render('blog', { faviconb64, type: "sample", page: null, blog }); + } + else res.render('blog', { faviconb64, type: "main", page: null, blog }); +}); + +// Individual blog pages / topic pages +app.get(/^\/blog\/.*/, (req, res) => { + let requestedTopic = req.path.split('/blog/')[1].split('/')[0]; + let requestedPage = req.path.split('/blog/')[1].split('/')[1]; + let topic = blog[requestedTopic]; + let page; + if (typeof topic != "object") { + send404(req, res); + return; + } + if (topic[requestedPage] == true) { + page = fs.readFileSync(path.join(baseDir, 'blog', requestedTopic, requestedPage)); + res.render('blog', { faviconb64, type: "individual", page, blog }) + } else send404(req, res); +}); + +app.get('/editor', (req, res) => { + res.render('editor', { faviconb64 }) +}); + +// Error handling (must be 2nd to last last app.use to make sure it catches all the errorings) +app.use(function(err, req, res, next) { + if(!err) { + next(); + return; + } + console.error('-------------------------------ERROR---------------------------------------\n'); + console.error(err); // Log da errors + console.error('-----------------------------USER-INFO-------------------------------------\n'); + console.error("Body:\n"); + console.error(req.body); + console.error("\n"); + console.error("Cookies:\n"); + console.error(req.cookies); + console.error("\n"); + console.error("Host:\n"); + console.error(req.host); + console.error("\n"); + console.error("Path:\n"); + console.error(req.path); + console.error("\n"); + console.error('----------------------------------------------------------------------------\n'); + res.status(400).send("
An internal server error has occurred. Please bother me if you see this.
Contact me at @oddbyte.11 on Signal, or contact@oddbyte.dev if you are a caveman and are stuck in the last decade.
"); // Give the user something that they can read +}); + +// 404 +app.use(function(req, res) { + send404(req, res); +}); + +function printHelp() { + rl.write("\n\n-------------------------------HELP-----------------------------------------\n"); + rl.write("> help [?] - prints this message\n"); + rl.write("> updateMusic [um] - runs updateSongs()\n"); + rl.write("> updateFileIndex [ufi] - runs updateFileIndex()\n"); + rl.write("> eval [exec] - runs raw JS code, returns output (if any)\n"); + rl.write("> quit [exit] - stops the server, and quits\n"); + rl.write("-------------------------------HELP-----------------------------------------\n\n") +} + +function quit() { + server.close(); + rl.write('\n\nI do declare - end broadcast\n\n'); // Yoinked Lunduke Journal's signoff :> + rl.close(); + process.exit(0); +} + +async function customConsole() { + while (1) { + // Pull commands + const cmd_raw = await rl.question("> "); + const cmd_raw_args = cmd_raw.split(" "); + const cmd_args = cmd_raw_args.slice(1); + const cmd = cmd_raw_args[0]; + + switch(cmd) { + case "?": + case "help": + printHelp(); + break; + case "um": + case "updateMusic": + updateSongs(); + break; + case "ufi": + case "updateFileIndex": + updateFileIndex(); + break; + case "exec": + case "eval": + // TODO: Fix this not actually using the correct context for whatever reason it just evals in an empty context + // ...... i dont know what or how or why but magically it is working now so i am just not gonna touch it and hope + // whatever black magic made it work keeps working .-. + try { + rl.write(eval(cmd_args.toString()) + "\n"); + } catch (err) { + console.error(err); + } + break; + case "quit": + case "exit": + quit(); + return; + case "": + break; + default: + rl.write(`Command not found: ${cmd}`) + printHelp(); + break; + } + } +} + +// Wrap in a async function to wait for youtube music response before starting http server +// (to prevent a race condition where people can view the moosic page be4 it is ready) +async function main() { + + // Init the moosics stuff (black magic) + await ytmusic.initialize(); + + // Populate playlistSongs and thumbnails + await updateSongs(); + + // Populate the blog index + await generateBlogIndex(); + + // Populate file index + await updateFileIndex(); + + server = http.createServer(app); + + await (async () => { + server.listen(port, () => { + console.log(`Listening to ${port}`); + + // Start hourly loop to update playlist + loopHourly(async () => await updateSongs()); + + // Start hourly loop to update file index + loopHourly(async () => await updateFileIndex()); + }); + })(); + + // Start console + customConsole(); +} + +// Handle a few signals + +rl.on('SIGINT', () => { + quit(); +}); + +rl.on('SIGTERM', () => { + quit(); +}); + +main(); diff --git a/package.json b/package.json new file mode 100755 index 0000000..8b6adce --- /dev/null +++ b/package.json @@ -0,0 +1,15 @@ +{ + "name": "portfolio", + "version": "1.2.6", + "main": "index.js", + "author": "oddbyte", + "type": "commonjs", + "license": "fuck you it's mine", + "dependencies": { + "axios": "^1.10.0", + "ejs": "^3.1.10", + "express": "^5.1.0", + "sqlite3": "^5.1.7", + "ytmusic-api": "^5.3.0" + } +} diff --git a/public/-.-.png b/public/-.-.png new file mode 100644 index 0000000..e2c0075 Binary files /dev/null and b/public/-.-.png differ diff --git a/public/00.png b/public/00.png new file mode 100644 index 0000000..1f4d0b0 Binary files /dev/null and b/public/00.png differ diff --git a/public/><.png b/public/><.png new file mode 100644 index 0000000..7cbe347 Binary files /dev/null and b/public/><.png differ diff --git a/public/DIE.png b/public/DIE.png new file mode 100644 index 0000000..9480e56 Binary files /dev/null and b/public/DIE.png differ diff --git a/public/Dotdotdot.png b/public/Dotdotdot.png new file mode 100644 index 0000000..27d12bb Binary files /dev/null and b/public/Dotdotdot.png differ diff --git a/public/^^.png b/public/^^.png new file mode 100644 index 0000000..0d5469c Binary files /dev/null and b/public/^^.png differ diff --git a/public/activated.png b/public/activated.png new file mode 100644 index 0000000..86da50f Binary files /dev/null and b/public/activated.png differ diff --git a/public/angy.png b/public/angy.png new file mode 100644 index 0000000..71ff562 Binary files /dev/null and b/public/angy.png differ diff --git a/public/base_full.png b/public/base_full.png new file mode 100644 index 0000000..beda8df Binary files /dev/null and b/public/base_full.png differ diff --git a/public/blank.png b/public/blank.png new file mode 100644 index 0000000..f8fc06d Binary files /dev/null and b/public/blank.png differ diff --git a/public/blank_full.png b/public/blank_full.png new file mode 100644 index 0000000..f448ad2 Binary files /dev/null and b/public/blank_full.png differ diff --git a/public/dotDotdot.png b/public/dotDotdot.png new file mode 100644 index 0000000..9a0db43 Binary files /dev/null and b/public/dotDotdot.png differ diff --git a/public/dotdotDot.png b/public/dotdotDot.png new file mode 100644 index 0000000..9fa2a9e Binary files /dev/null and b/public/dotdotDot.png differ diff --git a/public/dotdotdot.png b/public/dotdotdot.png new file mode 100644 index 0000000..52913f7 Binary files /dev/null and b/public/dotdotdot.png differ diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000..fef5237 Binary files /dev/null and b/public/favicon.ico differ diff --git a/public/hihi.png b/public/hihi.png new file mode 100644 index 0000000..ed93e62 Binary files /dev/null and b/public/hihi.png differ diff --git a/public/loading.gif b/public/loading.gif new file mode 100644 index 0000000..32b12d4 Binary files /dev/null and b/public/loading.gif differ diff --git a/public/piskelfile/-.-.piskel b/public/piskelfile/-.-.piskel new file mode 100644 index 0000000..9e1a5b7 --- /dev/null +++ b/public/piskelfile/-.-.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"-.-","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/00.piskel b/public/piskelfile/00.piskel new file mode 100644 index 0000000..dff7dbf --- /dev/null +++ b/public/piskelfile/00.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"blank","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/><.piskel b/public/piskelfile/><.piskel new file mode 100644 index 0000000..7203512 --- /dev/null +++ b/public/piskelfile/><.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"blank","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/DIE.piskel b/public/piskelfile/DIE.piskel new file mode 100644 index 0000000..1512e81 --- /dev/null +++ b/public/piskelfile/DIE.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"DIE","description":"","fps":0,"height":64,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}","{\"name\":\"Layer 1 (imported)\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/Dotdotdot.piskel b/public/piskelfile/Dotdotdot.piskel new file mode 100644 index 0000000..f838598 --- /dev/null +++ b/public/piskelfile/Dotdotdot.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"Dotdotdot","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/^^.piskel b/public/piskelfile/^^.piskel new file mode 100644 index 0000000..7a7c54c --- /dev/null +++ b/public/piskelfile/^^.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"blank","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/activated.piskel b/public/piskelfile/activated.piskel new file mode 100644 index 0000000..ed8c9a2 --- /dev/null +++ b/public/piskelfile/activated.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"activated","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/angy.piskel b/public/piskelfile/angy.piskel new file mode 100644 index 0000000..4eb3aa3 --- /dev/null +++ b/public/piskelfile/angy.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"angy","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/base.piskel b/public/piskelfile/base.piskel new file mode 100644 index 0000000..950c970 --- /dev/null +++ b/public/piskelfile/base.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"base","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/base_full.piskel b/public/piskelfile/base_full.piskel new file mode 100644 index 0000000..3ab6225 --- /dev/null +++ b/public/piskelfile/base_full.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"base_full","description":"","fps":0,"height":64,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}","{\"name\":\"Layer 1 (imported)\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/blank.piskel b/public/piskelfile/blank.piskel new file mode 100644 index 0000000..e0af562 --- /dev/null +++ b/public/piskelfile/blank.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"blank","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/blank_full.piskel b/public/piskelfile/blank_full.piskel new file mode 100644 index 0000000..cadcfb2 --- /dev/null +++ b/public/piskelfile/blank_full.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"blank_full","description":"","fps":0,"height":64,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}","{\"name\":\"Layer 1 (imported)\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/dotDotdot.piskel b/public/piskelfile/dotDotdot.piskel new file mode 100644 index 0000000..51e2c51 --- /dev/null +++ b/public/piskelfile/dotDotdot.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"dotDotdot","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/dotdotDot.piskel b/public/piskelfile/dotdotDot.piskel new file mode 100644 index 0000000..44003ac --- /dev/null +++ b/public/piskelfile/dotdotDot.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"dotdotDot","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/dotdotdot.piskel b/public/piskelfile/dotdotdot.piskel new file mode 100644 index 0000000..70b9b52 --- /dev/null +++ b/public/piskelfile/dotdotdot.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"dotdotdot","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/hihi.piskel b/public/piskelfile/hihi.piskel new file mode 100644 index 0000000..f3c09f9 --- /dev/null +++ b/public/piskelfile/hihi.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"hihi","description":"","fps":0,"height":64,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}","{\"name\":\"Layer 1 (imported)\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/question.piskel b/public/piskelfile/question.piskel new file mode 100644 index 0000000..76ca960 --- /dev/null +++ b/public/piskelfile/question.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"?","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/sad.piskel b/public/piskelfile/sad.piskel new file mode 100644 index 0000000..dab1398 --- /dev/null +++ b/public/piskelfile/sad.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"scared","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/piskelfile/scared.piskel b/public/piskelfile/scared.piskel new file mode 100644 index 0000000..bf970ea --- /dev/null +++ b/public/piskelfile/scared.piskel @@ -0,0 +1 @@ +{"modelVersion":2,"piskel":{"name":"scared","description":"","fps":0,"height":32,"width":32,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":1,\"chunks\":[{\"layout\":[[0]],\"base64PNG\":\"\"}]}"],"hiddenFrames":[""]}} \ No newline at end of file diff --git a/public/question.png b/public/question.png new file mode 100644 index 0000000..c9554ac Binary files /dev/null and b/public/question.png differ diff --git a/public/robots.txt b/public/robots.txt new file mode 100644 index 0000000..2929e0d --- /dev/null +++ b/public/robots.txt @@ -0,0 +1,58 @@ +# As a condition of accessing this website, you agree to abide by the following +# content signals: + +# (a) If a content-signal = yes, you may collect content for the corresponding +# use. +# (b) If a content-signal = no, you may not collect content for the +# corresponding use. +# (c) If the website operator does not include a content signal for a +# corresponding use, the website operator neither grants nor restricts +# permission via content signal with respect to the corresponding use. + +# The content signals and their meanings are: + +# search: building a search index and providing search results (e.g., returning +# hyperlinks and short excerpts from your website's contents). Search does not +# include providing AI-generated search summaries. +# ai-input: inputting content into one or more AI models (e.g., retrieval +# augmented generation, grounding, or other real-time taking of content for +# generative AI search answers). +# ai-train: training or fine-tuning AI models. + +# ANY RESTRICTIONS EXPRESSED VIA CONTENT SIGNALS ARE EXPRESS RESERVATIONS OF +# RIGHTS UNDER ARTICLE 4 OF THE EUROPEAN UNION DIRECTIVE 2019/790 ON COPYRIGHT +# AND RELATED RIGHTS IN THE DIGITAL SINGLE MARKET. + +User-Agent: * +Content-signal: search=yes,ai-train=no,ai-input=no +Allow: / + +User-agent: Amazonbot +Disallow: / + +User-agent: Applebot-Extended +Disallow: / + +User-agent: Bytespider +Disallow: / + +User-agent: CCBot +Disallow: / + +User-agent: ClaudeBot +Disallow: / + +User-agent: Employer +Disallow: / + +User-agent: Google-Extended +Disallow: / + +User-agent: GPTBot +Disallow: / + +User-Agent: Kagibot +Allow: / + +User-agent: meta-externalagent +Disallow: / diff --git a/public/sad.png b/public/sad.png new file mode 100644 index 0000000..b71e671 Binary files /dev/null and b/public/sad.png differ diff --git a/public/scared.png b/public/scared.png new file mode 100644 index 0000000..10f06a8 Binary files /dev/null and b/public/scared.png differ diff --git a/start.sh b/start.sh new file mode 100755 index 0000000..96acf7a --- /dev/null +++ b/start.sh @@ -0,0 +1,10 @@ +#!/bin/bash +echo '==== Cleaning up dead screens ====' +screen -wipe portfolio +echo '==== Killing old server ====' +sudo -iu prod fuser $(sudo -iu prod which node) -k +echo '==== Restarting Nginx ====' +rc-service nginx restart +echo '===== Starting server ====' +screen -dmS portfolio bash -c "sudo -iu prod bash -c 'cd /prod/portfolio && node --permission --allow-fs-read=/prod/portfolio/* .'" +echo 'boosh' diff --git a/views/art.ejs b/views/art.ejs new file mode 100644 index 0000000..06fc93f --- /dev/null +++ b/views/art.ejs @@ -0,0 +1,143 @@ + + + + Art + + + + + + + + +<%- include('partials/navbar') %> +
+
+
+
+

Art

+
+

+ Hihi :D
+ Here are all my various versions I have drawn of my profile picture ^^
+ I drew them using Piskel. +

+
+
+

Face

+
+

Blank

+
+ Piskelfile +
+
+

Base

+ +
+ Piskelfile +
+
+

Activated

+
+ Piskelfile +
+
+

><

+
+ Piskelfile +
+
+

^^

+
+ Piskelfile +
+
+

0.0

+
+ Piskelfile +
+
+

?

+
+ Piskelfile +
+
+

Angy

+
+ Piskelfile +
+
+

Sad

+
+ Piskelfile +
+
+

Scared

+
+ Piskelfile +
+
+

...

+
+ Piskelfile +
+
+

...

+
+ Piskelfile +
+
+

...

+
+ Piskelfile +
+
+

...

+
+ Piskelfile +
+
+

Loading

+
+

Made with Ezgif

+
+
+

-.-

+
+ Piskelfile +
+
+
+
+

Full body

+
+

Blank

+
+ Piskelfile +
+
+

Base

+
+ Piskelfile +
+
+

Hihi

+
+ Piskelfile +
+
+

DIE

+
+ Piskelfile +
+
+
+
+
+
+
+
+ + \ No newline at end of file diff --git a/views/blog.ejs b/views/blog.ejs new file mode 100644 index 0000000..7966b5e --- /dev/null +++ b/views/blog.ejs @@ -0,0 +1,69 @@ + + +<% if (type != "sample") { %> + The oddbyte blog + + + + <% } %> + + + + +<% if (type != "sample") { include('partials/navbar') } %> +
+
+
+
+ <% if (type == "main") { %> +
+

My blog

+

TODO: Main blog page here

+ <% } else if (type == "individual") { %> +
<%- page %>
+ <% } else if (type == "sample") { %> +
+ <% } %> +
+
+
<% if (type != "sample") { %> +
+
<% } else { %> + + <% } %> +
+ + diff --git a/views/editor.ejs b/views/editor.ejs new file mode 100644 index 0000000..e5e624d --- /dev/null +++ b/views/editor.ejs @@ -0,0 +1,257 @@ + + + + Oddbyte + + + + + + + + +<%- include('partials/navbar') %> +
+
+
+
+ Switch to preview +
+ +

Markdown Editor

+ +
+ + +
+
+
+
+
+
+ + + + + \ No newline at end of file diff --git a/views/home.ejs b/views/home.ejs new file mode 100755 index 0000000..bf078de --- /dev/null +++ b/views/home.ejs @@ -0,0 +1,106 @@ + + + + Oddbyte + + + + + + + + +<%- include('partials/navbar') %> +
+
+
+
+

Oddbyte

+
+ my profile picture +

+ Hihi :D
+ I am a nerd that happens to like hacking stuff.
+ I mainly nerd about privacy, android, web development, cybersecurity, and sysadmin.
+ My goal is to become a CISO. +

+
+

Stats

+
+
+ Species: + Hooman (I think) +
+
+ Height: + 5'9" +
+
+ Operating System: + Anxiousness OS +
+
+ Processor: + Fried Potato +
+
+ RAM: + Not Enough +
+
+ Bugs: + Anxiety^(3), Nerd^(2), ADHD^() +
+
+ Current thoughts: + Why does CSS suck so much? +
+
+ Gender: + Dood +
+
+ Age: + 2.5×ln(e)+2(e^(-i×π)×7.25)×cos(7π) +
+
+
+

Contact me

+

You can contact me by messaging me on:
+ - Signal (preferred), @oddbyte.01,
+ - Mastodon, @oddbyte@mastodon.social,
+ - Github, @oddbyte, or my
+ - Email, contact@oddbyte.dev
+

+
+

Other hooman

+

+ Check out this random internet person I happen to be aware of: voxel.top
+ He is also a cybersecurity / privacy nerd
+ His website sucks ass needs improvement
+ (yeah, lets go with "needs improvement") +

+<%/*

+ Check out my buddy's site, catocat.uk
+ She may have stolen things taken inspiration from my website. +

+*/%> +
+
+
+
+
+
+ + diff --git a/views/moosic.ejs b/views/moosic.ejs new file mode 100755 index 0000000..73c22d3 --- /dev/null +++ b/views/moosic.ejs @@ -0,0 +1,138 @@ + + + Oddbyte's music + + + + + + + + +<%- include('partials/navbar') %><% function toTitleCase(str) { + return str.replace( + /\w\S*/g, + text => text.charAt(0).toUpperCase() + text.substring(1).toLowerCase() + ); +} %> +
+
+
+
+

My favorite music

+

Open my favorite music playlist in youtube music

+ +
+
+
+
+
Made by Oddbyte.
+
+ diff --git a/views/partials/funfact.ejs b/views/partials/funfact.ejs new file mode 100755 index 0000000..d09cf95 --- /dev/null +++ b/views/partials/funfact.ejs @@ -0,0 +1,14 @@ +

<% const facts = [ + 'The first computer bug was an actual moth', + 'Android was originally an operating system for cameras', + '"password" is still one of the most commonly used passwords', + 'People call me funny sometimes I guess', + 'Haiii <3', + 'e^(i*pi) = -1', + 'According to the leading fan theory, 1 + 1 might actually not be 11', + 'xkcd.com is a thing, and it is funny most of the time', + 'Apparently I am a hooman', + 'This message changes every time you refresh the page', + 'I wish my friends would talk to me more often', + 'I am a functional introvert' + ]; %>Fun Fact: <%= facts[Math.floor(Math.random() * facts.length)] %>

diff --git a/views/partials/navbar.ejs b/views/partials/navbar.ejs new file mode 100755 index 0000000..185b58b --- /dev/null +++ b/views/partials/navbar.ejs @@ -0,0 +1,8 @@ + diff --git a/views/partials/style.ejs b/views/partials/style.ejs new file mode 100755 index 0000000..8a5399b --- /dev/null +++ b/views/partials/style.ejs @@ -0,0 +1,188 @@ + /* Begin default stylesheet */ + body { + font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; + background-color: #1E1E1E; + color: #e0e0e0; + padding: 20px; + text-align: center; + display: flex; + justify-content: center; + align-items: center; + flex-direction: column; + } + + *, *:before, *:after { + box-sizing: border-box; + } + + #container_top { + align-self: center; + background: rgb(33,43,54); + border-radius: 12px; + box-shadow: 1.5px 2px 1.25px #bb00ff33; + display: block; + min-width: 95%; + max-width: 1200px; + margin-top: 15px; + } + + #container_main { + display: flex; + flex-direction: column; + justify-content: center; + align-items: flex-start; + min-width: 100%; + max-width: 1200px; + } + #container_thing { + flex: 1; + min-width: 100%; + max-width: 1200px; + } + #h-break { + margin-top: 1em; + background-color: #66666666; + border-radius: 5px; + width: 3px; + align-self: stretch; + flex-shrink: 0; + } + #v-break { + margin: 1em 1em 1em 0em; + background-color: #6666661f; + border-radius: 5px; + height: 3px; + max-height: 3px; + position: relative; + width: 100%; + align-self: center; + } + a { + color: #78A9FF; + transition: color 0.5s, text-decoration-color 0.5s; + text-decoration: underline; + text-decoration-thickness: .15rem; + text-decoration-skip-ink: none; + text-decoration-color: #66666666; + text-underline-offset: .05rem; + } + a:hover { + color: #b3a2ff; + text-decoration-color: #78aaff66; + } + p { font-size: 1.35em; } + h1 { text-align: center; color: #4a90e2; font-size: 3em; } + h2 { font-size: 2em; margin: 0; color: #4a90e2; } + h3 { font-size: 1.75em; color: #6ba8e3; } + h4 { font-size: 1.5em; color: #8bc1e4; } + h5 { font-size: 1.25em; color: #a3cce6; } + h6 { font-size: 1.1em; color: #bbd7e7; font-weight: bold; } + code, kbd, pre, samp { + font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, + Liberation Mono, Courier New, monospace; + font-size: 1em; + background-color: rgb(43, 43, 43); + color: rgb(255, 174, 87); + border-radius: 3px; + } + footer { + text-align: center; + font-size: 1.125em; + color: #942ebd; + margin-top: 20px; + margin-bottom: 10px; + position: relative; + left: 0; + bottom: 0; + width: 100%; + clear: both; + } + *:focus { + outline-style: solid; + outline-color: transparent; + box-shadow: 0 0 0 2.5px rgba(147, 112, 219, 0.3); + } + .mobileonly { display: none; } + .selectDisable { + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -o-user-select: none; + user-select: none; + } + #button { + margin-top: 0; + font-size: 0.85rem; + color: rgba(255, 255, 255, 0.95); + text-decoration: none; + padding: 0.35rem 0.7rem; + border-radius: 12px; + background-image: linear-gradient(to bottom, rgba(114, 137, 218, 0.2), rgba(103, 58, 183, 0.15)); + font-weight: 500; + letter-spacing: 0.02em; + border: 1px solid rgba(147, 112, 219, 0.3); + display: inline-block; + box-sizing: border-box; + } + #navbar_contain { + min-width: 0; + width: 100%; + box-sizing: border-box; + max-width: 1200px; + margin: 0 auto; + } + #navbar { + font-family: inherit; + margin-top: 5px; + color: rgba(255, 255, 255, 0.95); + text-decoration: none; + padding: 0.4rem 0.8rem; + border-radius: 12px; + background-image: linear-gradient(to bottom, rgb(33 41 50), rgb(34 48 59)); + font-weight: 500; + font-size: 1.35em; + border: 1px solid rgba(147, 112, 219, 0.3); + display: block; + width: 100%; + text-align: center; + box-sizing: border-box; + } + #button:hover { + background-image: linear-gradient(to bottom, rgba(114, 137, 218, 0.3), rgba(103, 58, 183, 0.25)); + color: white; + box-shadow: 0 2px 8px rgba(103, 58, 183, 0.2); + } + @media (max-width: 1201px) { + body { + padding: 10px; align-items: stretch; + } + #container_top, #container_main, #container_thing { + min-width: 100%; + max-width: 100%; + margin: .5em 0; + } + #container_main { + flex-direction: column; align-items: stretch; + } + #h-break { width: 100%; height: 3px; margin: 1em 0; } + p { font-size: 1.1em; } + h1 { font-size: 2.2em; } + h2 { font-size: 1.6em; } + h3 { font-size: 1.4em; } + h4 { font-size: 1.2em; } + h5 { font-size: 1.1em; } + h6 { font-size: 1em; } + #navbar_contain { + padding: 0 0.5rem; + } + #navbar { + font-size: 0.85rem; + padding: 0.35rem 0.7rem; + } + #button { + font-size: 0.75rem; + padding: 0.3rem 0.6rem; + } + .mobileonly { display: block; } + .desktoponly { display: none; } + } diff --git a/views/source.ejs b/views/source.ejs new file mode 100755 index 0000000..35a8d13 --- /dev/null +++ b/views/source.ejs @@ -0,0 +1,79 @@ + + + + + + Directory Tree + + + + +<%- include('partials/navbar') %> +
+
+
+
+

Directory Tree

+

(here is where all the raw files are)

+ +
    + <% function renderTree(items, indentLevel = 0) { %> + <% items.forEach(item => { %> +
  • + <% if (item.type === 'directory') { %> + <%= item.name %> + <% if (item.children && item.children.length > 0) { %> +
      + <%= renderTree(item.children, indentLevel + 1) %> +
    + <% } %> + <% } else { %> + <%= item.name %> + <% } %> +
  • + <% }); %> + <% } %> + + <%= renderTree(paths) %> +
+ +

Rendered Pages

+
+
+

these are the thingies that you see when you use my site like a normal person (in a browser, hopefully)

+

+ / <-- Main Page
+ /music <-- My Youtube Music playlist
+ /source <-- You are here :P
+ /art <-- My art ^.^
+ /blog <-- My crappy blog
+

+
+
+

Files

+
+
+

+ /favicon.ico <-- icon
+ /robots.txt <-- tells some robots to go away +

+
+
+
+
+
+ + diff --git a/views/sourceviewer.ejs b/views/sourceviewer.ejs new file mode 100644 index 0000000..dfc4ce2 --- /dev/null +++ b/views/sourceviewer.ejs @@ -0,0 +1,25 @@ + + + + Oddbyte + + + + + + + + +

Source code for <%-filePath %>

+

Back

+
+
<%= fileContent %>
+
+

Back

+ + diff --git a/views/template.ejs b/views/template.ejs new file mode 100644 index 0000000..ef181e0 --- /dev/null +++ b/views/template.ejs @@ -0,0 +1,29 @@ + + + + Oddbyte + + + + + + + + +<%- include('partials/navbar') %> +
+
+
+
+ +
+
+
+
+ +
+ + \ No newline at end of file