Antonio De Lucreziis, studente di Matematica e macchinista del PHC
Il PHC è un gruppo di studenti di Matematica con interessi per, open source, Linux, self-hosting e soprattutto smanettare sia con hardware e software (veniteci pure a trovare!)
Go, linguaggio di programmazione compilato, statically typed e con una libreria standard con tutte le cose essenziali.
Un tool per NodeJS per creare velocemente web app in HTML, JS, CSS utilizzando tutto l'ecosistema di NodeJS e NPM senza configurare quasi nulla
Javascript
Typescript
Python
NodeJS
Golang
https://example-website.xyz/
Contatore: 73
Incrementa
Decrementa
Utente
Server
Utente
Server
http://example-website.xyz
Utente
Server
http://example-website.xyz
Utente
Server
http://123.123.123.123:80
Il DNS si occupa di risolvere il dominio e convertirlo in un IPv4 (o IPv6)
Client
Server
Connessione TLS
Scambio di chiavi
Scambio di dati in modo sicuro (HTTP)
?
Server (Macchina)
:80
example.com (192.0.2.42)
Server (Macchina)
:80
Server (Programma)
example.com (192.0.2.42)
$ mkdir gdg-counter-website
$ cd gdg-counter-website
# inizializziamo il file "go.mod"
$ go mod init gdg-talk-counter
# o se avessimo un repo git anche così...
$ go mod init github.com/aziis98/gdg-talk-counter
# Crea il file "package.json"
$ npm init
# Installa ViteJS come dipendenza di development
$ npm install -D vite
O anche con un altro package manager alternativo a npm
(ad esempio io userò pnpm
)
{
"name": "frontend",
"version": "1.0.0",
"scripts": {
// avvia il server di ViteJS in modalità di development
"dev": "vite",
// crea la cartella "dist/" con tutti gli asset e bundle
"build": "vite build"
},
"author": "aziis98",
"license": "MIT",
"devDependencies": {
"vite": "^3.2.3"
}
}
package.json
<!DOCTYPE html>
<html lang="en">
<head>
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>GDG Talk Counter</title>
</head>
<body>
<h1>GDG Counter Website</h1>
<div class="app">
<div id="counter-value">???</div>
<button id="btn-increment">Incrementa</button>
<button id="btn-decrement">Decrementa</button>
</div>
<script type="module" src="/src/main.js"></script>
</body>
</html>
index.html
const counterElement = document.querySelector('#counter-value')
const incrementButton = document.querySelector('#btn-increment')
const decrementButton = document.querySelector('#btn-decrement')
function updateCounter(value) {
counterElement.textContent = `Counter: ${value}`
}
src/main.js
Client
Server
PATCH esempio.org/a/b/c
Client
Server
POST esempio.org/a/b/c
Client
Server
GET esempio.org/a/b/c
Client
Server
DELETE esempio.org/a/b/c
Client
Server
PUT esempio.org/a/b/c
Client
Server
PATCH esempio.org/a/b/c
Client
Server
POST esempio.org/a/b/c
Client
Server
GET esempio.org/a/b/c
Client
Server
DELETE esempio.org/a/b/c
Client
Server
PUT esempio.org/a/b/c
Client
Server
GET /api/value
valore aggiornato del contatore
Client
Server
POST /api/increment
valore aggiornato del contatore
Client
Server
POST /api/decrement
valore aggiornato del contatore
/* ... */
incrementButton.addEventListener('click', () => {
fetch('/api/increment', { method: 'POST' })
.then(res => res.json())
.then(data => updateCounter(data))
})
decrementButton.addEventListener('click', () => {
fetch('/api/decrement', { method: 'POST' })
.then(res => res.json())
.then(data => updateCounter(data))
})
fetch('/api/value')
.then(res => res.json())
.then(data => updateCounter(data))
src/main.js
/* ... */
incrementButton.addEventListener('click', async () => {
const res = await fetch('/api/increment', { method: 'POST' })
const data = await res.json()
updateCounter(data)
})
decrementButton.addEventListener('click', async () => {
const res = await fetch('/api/decrement', { method: 'POST' })
const data = await res.json()
updateCounter(data)
})
const res = await fetch('/api/value')
const data = await res.json()
updateCounter(data)
src/main.js
package main
import (
"log"
"net/http"
)
func main() {
mux := http.NewServeMux()
setupRoutes(mux) // visto meglio in seguito
server := http.Server{
Addr: ":4000",
Handler: mux,
}
log.Fatal(server.ListenAndServe())
}
main.go
/* ... */
func setupRoutes(mux *http.ServeMux) {
counter := 0
mux.HandleFunc("/api/status", func(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodGet {
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
return
}
w.Header().Set("Content-Type", "application/json")
err := json.NewEncoder(w).Encode("ok")
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
})
mux.Handle("/", http.FileServer((http.Dir("./dist/"))))
}
main.go
import { defineConfig } from 'vite'
export default defineConfig({
server: {
port: 3000,
proxy: {
'/api': 'http://localhost:4000/',
},
},
})
vite.config.js
Development
ViteJS Dev Server
Go Server
:3000 /api/...
:3000 /...
:4000 /api/...
{...}
{...}
:3000
Production
Go Server
:4000 /...
:4000 /api/...
dist/...
{...}
:4000
/* ... */
func setupRoutes(mux *http.ServeMux) {
counter := 0
/* ... */
mux.HandleFunc("/api/value", func(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodGet {
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
return
}
w.Header().Set("Content-Type", "application/json")
err := json.NewEncoder(w).Encode(counter)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
})
/* ... */
}
main.go
counter := 0
/* ... */
mux.HandleFunc("/api/increment", func(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
return
}
counter++
w.Header().Set("Content-Type", "application/json")
err := json.NewEncoder(w).Encode(counter)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
})
main.go
counter := 0
/* ... */
mux.HandleFunc("/api/decrement", func(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost {
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
return
}
counter--
w.Header().Set("Content-Type", "application/json")
err := json.NewEncoder(w).Encode(counter)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
})
main.go