Organize Your Thoughts

Build Your Own Website

Created At: 2/1/2025, 3:53:51 AM | Last Edited: 2/7/2025, 12:17:48 AM

Why Build Anything at All? thinking


        If you genuinely care about anything and want to avoid sounding like a mad scientist when you talk about it, I highly recommend you start forming your thoughts and efforts into digestible pieces of content. Engineers are notoriously horrible with communication, but we care and want to build something that truly works. To make something work, then, means to make something that works for people. But, if no one can understand what we are doing or why, we haven’t actually made anything that works. We’ve made something else, which is fine; there are many things we can do for ourselves or our own benefit, and the world doesn’t have to understand them completely. Instead, making something that works starts with correctly identifying real people’s issues. We have to spend time talking to them about their difficulties and perspectives and listen when they tell us what they need or what we aren’t providing. If we do all this and still fail, it’s okay; you were just trying to help.


If you want to reach the people who need you and your skills the most start sharing and they will come. Many companies have seen the benefits of opening up their private works and allowing people to participate in the systems they rely on, like google_logo Google’s Chromium chromium_logo or red_hat_logo Red Hat’s Podman podman_logo. Similarly, I hope sharing what I do reaches an audience of individual personal computer users who could benefit, while acknowledging that many real people can’t even make it here. Even if no one reads these paragraphs, writing them organized my thoughts about website technology so at least I can understand what I am doing and why. If you don’t follow the rest of this guide to make a website to host your content, I highly recommend building some other habit of explaining what you do, even if you’re literally talking to a wall or a mirror, so if someone ever does ask you for help; you’ll be ready. Now that you and I both know why we are here, let’s get into it:

Requirements

  1. A computer computer that can run deno_logo Deno. Deno is a software written in rust_logo Rust, so there are some unofficial builds too. If you aren’t running Mac, Linux or Windows I recommend looking around.

  2. An internet connection 🛜. In America, internet connections are paid services provided by organizations like xfinity_logo Xfinity or Verizon. To connect you to a computer computer located across the world earth_africa, these organizations work with larger networks that manage global connections, like undersea cables and satellites.

  3. (optional) If you want others to be able to access your website, you will need a github_logo GitHub account.


Exploring Website Technology mag_right

2 Kinds of Software: Servers & Clients


        All websites and website technologies, of every language and scale, boil down stew to 3 main ingredients. These ingredients are just different kinds of page_facing_uptext we store in files:

  1. JavaScript nologo: Code that the browsers can run. Chrome runs JavaScript files using Google’s open-source V8 JavaScript Engine. To execute a .js file from your command line, you’ll need a runtime--software that reads and runs code. The most popular JavaScript runtime, Node.js, and what we will use today, Deno, are also built on the V8 engine. After installing, to run the code written in server.js, for example, you would have to do $ deno server.js, similar to how Python code is run.

  2. HTML html_logo: This language describes to browsers what to draw in the window canvas. It contains the content and the ordering of paragraphs of text, images, links, boxes, and all the visual elements in the browser window. The JavaScript code can also read and change the window’s HTML: this is how most meaningful visual content changes and code interactions happen.

  3. CSS css_logo: This is a set of properties you can use to modify the display of HTML elements. To match my HTML examples above, you could set text font, images opacity, links color, and boxes size. The same HTML can look significantly different depending on the .css style file sent with it. If you didn’t adequately receive the .css file when visiting a site, all the text could be black on the left, and the background would be white, depending on the browser’s CSS defaults. You may have briefly witnessed this as a slow site loads or if your connection was interrupted before getting the CSS.


The boiling down gets complicated as the three ingredients start to mix. Browsers are large softwares; web technologies, and as you’ll see, are up to so much of this boiling automatically under the hood. Let’s start with a minimal example. I’ll assume you’re using a browser, so let’s type example.com into the address bar:

Press F12 to open Browser Tools -> `Network` tab
Press F12 to open Browser Tools -> Network tab

After I hit ↵ Enter, my browser requests the / HTML file to display in its main window and favicon.ico to fill in the icon in the tab title bar. In the image above, my browser’s favicon request 404s (file not found), but the / response is a document of type HTML. example.com is a minimal example, so it doesn’t have a favicon; without it, the main page will still load fine since / did not 404 and instead was "found".

A New Dish spaghetti


        Next, let’s examine how intense relationships twisted_rightwards_arrows between website technologies and the three main ingredients can get. Our minimal example.com has sent 1 HTML file:

Open a new tab and paste this into your address bar: `view-source:https://example.com/`
Open a new tab and paste this into your address bar: view-source:https://example.com/

On line 4, we have an HTML <title> tag which modern browsers read and then render as the tab name:

`example.com` Tab Name with no favicon
example.com Tab Name with no favicon

On line 9, there is an HTML <style> tag of type text/css, meaning the next block should be read as an inline .css file. On line 14, the inline CSS sets which font the browser should use to display text in the <body> tag:

`example.com` Styled Content
example.com Styled Content

Using the browser Developer Tools, I can delete some of what they sent to show you what a world with no CSS could look like:

`example.com` No CSS Alternate Universe
example.com No CSS Alternate Universe

We are witnessing our first most basic twist twisted_rightwards_arrows on the ingredients: The page_facing_up text stored in the file is now of 2 languages that work according to 2 different standards stored in different ways on 2 computers. I don’t suppose you’ll find example.com’s index.html in your Downloads folder, and example.com’s server system could be a mouse trained to jump on a keyboard to compose HTML and CSS and pack it up as an HTTP response. Most websites don’t employ mice and instead use software to listen for requests and craft & deliver responses. In fact, example.com tells us its server software is ECAcc (dcd/7D5D) in its response headers, a set of metadata server softwares can send along with responses. ECAcc (dcd/7D5D) is a CDN made by Verizon. The best resources for learning how to use the three main ingredients and their standards are either W3Schools or, my preference, Mozilla. Mozilla also covers new fire web technologies like WASM and WebGL


In the following sections, we will see how to:

  1. Download deno_logo Deno and fresh_logo Fresh and use them to create a website and make it accessible over the internet (skip to section).
  2. Use Deno and Fresh as the software to generate the HTML files and send them along with JS code to run in the browser as a response to build a cohesive website (skip to section).

Wait, It’s All Computers & File Systems? earth_africaman_astronaut gunwoman_astronaut

        example.com is like a name for a computer that my browser can use to find and connect to over the internet. The software running on that computer, serving the file, is considered; The Server. Our greedy, needy browser software requesting so many files would be The Client. Server & Client are sometimes used to refer to the machine, causing confusion with server computers, which are computers optimized to run server software. As we move into a computing landscape where most computers are running server and client software, I suggest, when needed, we use Host when referring to computers running server software and Guest for computers running client software.


Most servers send /index.html by default when a file is not specified: /. You can see this is true by going to example.com/index.html or google.com/index.html, and if you want to go on a tangent try google.com/robots.txt. The idea, traditionally, is that browsers are like remote file system browsers, but the files we are sharing have advanced. If you have python_logoPython installed, you can easily see your own computer’s fs in your browser by trying:

$ python3 -m http.server # 8001 # <- uncomment if port in use 📋

By visiting the link provided by Python in your browser, you’ll be able to navigate the folder in which you ran the command. This command starts a program on your computer that listens for HTTP requests on port 8000. This program would be considered a server, which isn’t surprising since we ran Python’s built-in http.server module using the -m option. This Python program serves the files of the folder it was run in, and its HTML webpage response looks like a file browser UI. A port is like an extension to a computer name where that computer is listening for data that follows a specific protocol. HTTP and HTTPS are different protocols for transferring website files; they define how the HTML and other files should be asked for and packed up. Most client software delivers HTTP requests to port 80 by default if no ports are specified in the request. Multiple softwares can be listening on one computer, so thus we have Ports so that 1 machine can listen for many protocols on different ports. Secure websites redirect HTTP requests to port 443, and secure browsers stick to HTTPS whenever possible, effectively upgrading the security before receiving website content.


Other than 80 and 443 for website content, there are many default ports that other non-website servers listen on, indicating that there are many kinds of server softwares and protocols for them. For example, there are default ports for 22 for SSH, which serves a remote shell, and 53 for DNS, the system that turns example.com into a real computer address;. Hence, it serves IP addresses. Client software knows how to properly request and use this information, so any other computers running browsers on the same network as your computer running the Python server can connect to your computer using the port you opened above. To prevent this, you must either use a firewall to block requests or just stop serving and close the http.server by sending Ctrl+C. Check out this section of the vpn/dns post that covers setting up firewalls.

`example.com` HTTP (first 2 requests) vs. HTTPS (second 2 requests)
example.com HTTP (first 2 requests) vs. HTTPS (second 2 requests)

You can see this by visiting the insecure example.com:80/ or example.com:443/ and seeing that sending an HTTPS request to port 80 would fail since that port is expecting HTTP traffic: https://example.com:80. Since that link cant be visited in my browser, Firefox by Mozilla, my browser will never put it in its browser history, so the browser will never make the underline purple, which has been set in the CSS as the visited link color. Not all browsers boil down curry the same; on my phone, the last visited link is the hover color until I tap something else, then the link loses focus and displays the visited color purple. The Mozilla reference tries its best to keep track of these browser differences.

Website: Publishing

Enter the Twisted World twisted_rightwards_arrows


        Server software and the hosts they run on are very diverse. Using JavaScript-based server software provides a smooth experience because you can easily craft HTML, CSS, and JS in JavaScript, which the language was built to do. However, there are some definitive drawbacks to writing code in JavaScript as it was originally intended to only ever run in the browser. I prefer using TypeScript typescript_logo, which extends JavaScript to make it more like a traditional programming language and addresses these drawbacks. Deno is a growing and promising JavaScript, TypeScript, and WebAssembly runtime, meaning we can use it to run code we write in .ts files and have it boil down into JS code for clients. Fresh is a set of server software and tools written in TypeScript that we will run using Deno. burst.deno.dev is the name of a computer on the internet that is using Deno to run TypeScript code that listens for website requests and dishes back responses. You can view the public GitHub repo, where the code for this website is stored. Whenever it is updated, GitHub runs a script that packs up all my files and sends them to Deno Deploy. Deno Deploy starts running the code on a computer named burst.deno.dev, and you can connect to it the same way you can connect to deno.dev.


The documentation for Fresh is easy to navigate and short. I recommend checking it out as they explain each step in detail and offer alternative configurations. Here are the minimal steps to get a website up that others can visit, just like this one! ( without the blog or cubesob )

Deno = goat & Fresh = boat

  1. First, we have to install Deno. Follow this link for OS-specific instructions. To check if it installed correctly, open a terminal and try:
$ deno -v 📋
  1. We will use Deno to create a new Fresh Project. A Fresh Project consists of a bunch of TypeScript, most of which is code that listens for and responds to requests provided by Fresh. The rest is the code that crafts each page of the website. Fresh gives you some sample pages, but you’ll eventually want to replace these with your own code. This command will download all the needed Fresh code and put it in a folder you’ll have to name:
$ deno run -A -r https://fresh.deno.dev 📋
  1. Check out your minimal website by following the link in the command line. You’ll receive your website from your own computer (twisted_rightwards_arrows host = guest)
$ cd whatever-you-named-it 📋
$ deno task start 📋
  1. If you want others to be able to visit your website easily, you’ll first need a GitHub account. Then you’ll need to upload your code to Deno Deploy: (follow the link Deno gives you in the command line to see your site on the internet!)
$ deno install -gArf jsr:@deno/deployctl 📋
$ deployctl deploy 📋

        Step 4 is also technically optional since there are many free hosting options available on the internet. This means many companies will give you access to a computer that has a URL that can be found on the internet. This is nice because you usually have to pay to put your URL in a public registry that maps names like example.com to actual computers. For example, you can ask the world to connect to your personal computer face_with_peeking_eye for all the files smiling_imppage_facing_up. There are many ways to host files securely, and Deno Deploy has its limitations as to what it can host, but it is a good introduction to cloud hosting and immediately offers a URL everyone can use easily. If you don’t want to send everyone the default Fresh lemon index.html, let’s check out how I used Fresh to send you this blog post:

Website: Building building_construction

Islands beach_umbrella


        Fresh focuses on a feature called Islands, which makes it easy to specify what code should be run on the server versus the Client. I want to write blog posts like the one you are reading now in Markdown. Markdown is a simple language that simplifies text formatting, like wrapping words with * for italics. This simplicity makes it the preferred language for text posts across the internet; GitHub READMES, ChatGPT Conversations, and Discord Messages all use Markdown. However, I also want to send you non-text content that Markdown cannot describe, like the cube below. For this to work smoothly, I’ll have to convert the Markdown into HTML partially over here and have you fill out the rest over there on an beach_umbrella Island:

Over here on the server, I’ll focus on getting the static content page_facing_up (HTML + JS text) ready and delivered while your browser handles the dynamic content red_square (running JS). Limiting the server to simple text-processing tasks dramatically affects how fast each page loads since I only have to compute the text to send you before responding, including the code [ page_facing_up ] for the cube [ red_square ]. I won’t have to run any code to generate the refresh-rate based cube before responding to your browser while navigating here. Your browser will immediately display the static text, and after that, start running the code to display the above SpinningCube as soon as it can.

Dynamic Routing


        Next, I want to set up a blog system where I can easily make new posts. For now, I have just made a single file in posts called Build_Your_Own_Website.md:

Fresh Project: Directory Tree
Fresh Project: Directory Tree

Using dynamic routing, /blog/[postname].tsx will serve [postname].md as HTML when you visit the corresponding /blog/[postname], and if I add another Markdown file in the posts folder for a new link to show up on the homepage & a new route, e.g. /blog/post2 will be generated. Going to a post I haven’t made yet would fail since its corresponding .md file wouldn’t be found by my server resulting in a 404.

Combination - Islands in Blog Posts


Code Responsible For Building Blog Post Pages: <code>/blog/[postname].tsx</code>
Code Responsible For Building Blog Post Pages: /blog/[postname].tsx

        Above, BlogRendererSS is a Markdown-to-HTML renderer that will run on the server when you request a blog post; turning [postname].md into HTML. DynamicMarkdownItem is an Island that replaces {{ SpinningCube }}’s HTML with the corresponding actual Island’s HTML. This replacement code will run in your browser. The static HTML I sent will come alive and move, having been replaced by the actual HTML and JavaScript code that displays the cube. This twist completes our goal of having a simple but dynamic blogging system that effectively separates Server and Client work. If you want to see exactly how everything is setup postname.tsx,BlogRendererSS.tsx, DynamicMarkdownItem.tsx, and the rest of the files are all on Github.

<strong>SpinningCube</strong> Island in Markdown
SpinningCube Island in Markdown

Using Islands helps build a smooth experience by separating computing resources for specific tasks. For the code blocks above, you get to run the JavaScript code in your browser that copies the text from the website I sent you into your clipboard. I have nothing to do with it. I sent you the text and code that I think you should have. Decidingthinking what to handle locally and what to send can vary depending on the goal; for example, for security, if I bought an API key key to access google_logo Google Maps to show a Street View in a blog post; I could make sure that API connection happens on the serverclosed_lock_with_key and not on an Island on the Clientunlock otherwise anyone could steal the connection details and destroy my wallet with the wonders of elastic pricing. The homepage will load faster if you received the cube here, or this page will load faster if you got the cube there since it is your cube now! You are now in charge of the Cube Island I’ve sent. Try moving the cube across screens with different refresh rates. Your browser will tell the cube to match your monitor’s refresh-rate without needing any help from the server. I am not sending you video frames or animation updates, just sharing the code of it so you use your own GPU to produce your own frames and animations!!! space_invader