v/tutorials/Building a simple web blog .../README.md

415 lines
10 KiB
Markdown
Raw Normal View History

## Building a 150 KB web blog in V & SQLite
2019-12-14 05:39:51 +01:00
2019-12-14 03:31:09 +01:00
Hello,
2020-01-18 17:03:47 +01:00
In this guide, we'll build a simple web blog in V.
2019-12-14 03:31:09 +01:00
The benefits of using V for web:
- A safe, fast, language with the development agility of Python or Ruby and
the performance of C.
2019-12-14 03:31:09 +01:00
- Zero dependencies: everything you need for web development comes with the language
in a 1 MB package.
2019-12-14 03:31:09 +01:00
- Very small resulting binaries: the blog we'll create in this tutorial is about 150 KB.
- Easy deployments: a single binary file that even includes the precompiled templates.
2019-12-14 05:36:18 +01:00
- Runs on the cheapest hardware with minimum footprint: for most apps a $3 instance
is enough.
2019-12-14 06:24:14 +01:00
- Fast development without any boilerplate.
2019-12-14 03:31:09 +01:00
_Please note that V and Vweb are at a very early stage and are changing rapidly._
2019-12-14 03:31:09 +01:00
The code is available <a href='https://github.com/vlang/v/tree/master/tutorials/code/blog'>here</a>.
2019-12-14 05:36:18 +01:00
2019-12-14 03:31:09 +01:00
### Installing V
```
wget https://github.com/vlang/v/releases/latest/download/v_linux.zip
2020-12-01 16:09:33 +01:00
unzip v_linux.zip
2019-12-14 03:31:09 +01:00
cd v
sudo ./v symlink
```
Now V should be globally available on your system.
2020-12-01 16:09:33 +01:00
> On macOS use `v_macos.zip`, on Windows - `v_windows.zip`.
> If you use a BSD system, Solaris, Android, or simply want to install V
> from source, follow the simple instructions here:
> https://github.com/vlang/v#installing-v-from-source
2019-12-14 03:31:09 +01:00
### Install SQLite development dependency
If you don't have it already installed, look at the
2021-03-22 03:42:07 +01:00
[`sqlite` README](../../vlib/sqlite/README.md) for instructions.
2019-12-14 03:31:09 +01:00
### Creating a new Vweb project
2019-12-14 05:36:18 +01:00
V projects can be created anywhere and don't need to have a certain structure:
2019-12-14 03:31:09 +01:00
```bash
mkdir blog
v init
2019-12-14 03:31:09 +01:00
```
2020-01-18 17:03:47 +01:00
First, let's create a simple hello world website:
2019-12-14 03:31:09 +01:00
```v oksyntax
2019-12-14 03:31:09 +01:00
// blog.v
module main
2020-05-20 14:21:30 +02:00
import vweb
2019-12-14 03:31:09 +01:00
struct App {
2020-12-31 17:31:17 +01:00
vweb.Context
2019-12-14 03:31:09 +01:00
}
fn main() {
app := App{}
vweb.run(app, 8081)
2019-12-14 03:31:09 +01:00
}
['/index']
2020-07-07 22:01:18 +02:00
pub fn (mut app App) index() vweb.Result {
return app.text('Hello world from vweb!')
2019-12-14 03:31:09 +01:00
}
```
Run it with
```bash
v run blog.v
```
2019-12-14 16:59:02 +01:00
```
Running a Vweb app on http://localhost:8081 ...
2019-12-14 16:59:02 +01:00
```
Vweb helpfully provided a link, open http://localhost:8081/ in your browser:
2019-12-14 03:31:09 +01:00
2021-03-04 15:03:02 +01:00
<img width=662 src="https://github.com/vlang/v/blob/master/tutorials/building_a_simple_web_blog_with_vweb/img/hello.png?raw=true">
2019-12-14 03:31:09 +01:00
The `App` struct is an entry point of our web application. If you have experience
with an MVC web framework, you can think of it as a controller. (Vweb is
2020-12-31 17:31:17 +01:00
not an MVC framework however.) It embeds the vweb Context object, that's why we get access
to methods like `.text()`.
2019-12-14 03:31:09 +01:00
As you can see, there are no routing rules. The `index()` action handles the `/` request by default.
2020-01-18 17:03:47 +01:00
Vweb often uses convention over configuration and adding a new action requires
2019-12-14 03:31:09 +01:00
no routing rules either:
```v oksyntax
// blog.v
import vweb
import time
fn (mut app App) time() vweb.Result {
return app.text(time.now().format())
2019-12-14 03:31:09 +01:00
}
```
2021-03-04 15:03:02 +01:00
<img width=662 src="https://github.com/vlang/v/blob/master/tutorials/building_a_simple_web_blog_with_vweb/img/time.png?raw=true">
2019-12-14 03:31:09 +01:00
> TIP: run the following command to live-reload the server: `v watch run blog.v`
2019-12-14 03:31:09 +01:00
2020-01-18 17:03:47 +01:00
The `.text(string)` method returns a plain text document with the provided
2019-12-14 03:31:09 +01:00
text, which isn't frequently used in websites.
### HTML View
Let's return an HTML view instead. Create `index.html` in the same directory:
```html
<html>
<head>
<title>V Blog</title>
</head>
<body>
<b>@message</b>
<br />
<img src="https://vlang.io/img/v-logo.png" width="100" />
</body>
2019-12-14 03:31:09 +01:00
</html>
```
and update our `index()` action so that it returns the HTML view we just created:
```v ignore
// blog.v
2020-07-07 22:01:18 +02:00
pub fn (mut app App) index() vweb.Result {
2019-12-14 03:31:09 +01:00
message := 'Hello, world from Vweb!'
2020-07-04 18:56:18 +02:00
return $vweb.html()
2019-12-14 03:31:09 +01:00
}
```
2021-03-04 15:03:02 +01:00
<img width=662 src="https://github.com/vlang/v/blob/master/tutorials/building_a_simple_web_blog_with_vweb/img/hello_html.png?raw=true">
2019-12-14 03:31:09 +01:00
Good, now we have an actual HTML page.
The V template language is similar to C#'s Razor: `@message` prints the value
of `message`.
You may notice something unusual: the `message` variable created in the `index()`
action is automatically available in the view.
It's another feature of Vweb to reduce the boilerplate in your web apps.
No need to create view models just to pass data, or use an unsafe and untyped
alternative, like C#'s `ViewBag["message"]`.
2020-01-18 17:03:47 +01:00
Making all action variables available in the view may seem crazy,
2019-12-14 03:31:09 +01:00
but V is a language with pure functions by default, and you won't be able
to modify any data from a view. `<b>@foo.bar()</b>` will only work if the `bar()` method
doesn't modify `foo`.
The HTML template is compiled to V during the compilation of the website,
that's done by the `$vweb.html()` line.
2019-12-14 03:31:09 +01:00
(`$` always means compile time actions in V.) offering the following benefits:
- Great performance, since the templates don't need to be compiled
on every request, like in almost every major web framework.
2019-12-14 03:31:09 +01:00
- Easier deployment, since all your HTML templates are compiled
into a single binary file together with the web application itself.
2019-12-14 03:31:09 +01:00
- All errors in the templates are guaranteed to be caught during compilation.
### Fetching data with V ORM
Now let's display some articles!
We'll be using V's builtin ORM and a SQLite database.
2020-07-04 18:56:18 +02:00
(V ORM will also support MySQL, Postgre, and SQL Server soon.)
2019-12-14 03:31:09 +01:00
2020-07-04 18:56:18 +02:00
Add a SQLite handle to `App`:
2019-12-14 03:31:09 +01:00
```v oksyntax
// blog.v
2020-07-04 18:56:18 +02:00
import sqlite
import vweb
2020-07-04 18:56:18 +02:00
2019-12-14 03:31:09 +01:00
struct App {
vweb.Context
pub mut:
db sqlite.DB
2019-12-14 03:31:09 +01:00
}
```
In `fn main()` we'll connect to a database.
Code in the `main()` function is run only once during app's startup, so we are going
to have one DB connection for all requests. This improves the performance of the web application,
since a DB connection doesn't have to be set up for each request.
```v oksyntax
// blog.v
fn main() {
mut app := App{
db: sqlite.connect(':memory:') or { panic(err) }
}
sql app.db {
create table Article
}
first_article := Article{
title: 'Hello, world!'
text: 'V is great.'
}
second_article := Article{
title: 'Second post.'
text: 'Hm... what should I write about?'
}
sql app.db {
insert first_article into Article
insert second_article into Article
}
vweb.run(app, 8080)
2019-12-14 03:31:09 +01:00
}
```
Create a new file `article.v`:
```v oksyntax
2020-07-04 18:56:18 +02:00
// article.v
2019-12-14 03:31:09 +01:00
module main
struct Article {
id int [primary; sql: serial]
2019-12-14 03:31:09 +01:00
title string
text string
}
2019-12-14 17:58:55 +01:00
pub fn (app &App) find_all_articles() []Article {
return sql app.db {
select from Article
}
2019-12-14 03:31:09 +01:00
}
```
Notice that the `Article` structure conforms to the same structure and naming as
the database table in the creation SQL statement. Also we need to add ORM decorators
to our primary key to let it know that it is the primary key and it should auto-increment
2019-12-14 03:31:09 +01:00
Let's fetch the articles in the `index()` action:
```v ignore
// blog.v
pub fn (app &App) index() vweb.Result {
2019-12-14 03:31:09 +01:00
articles := app.find_all_articles()
return $vweb.html()
2019-12-14 03:31:09 +01:00
}
```
Finally, let's update our view:
```html
<body>
@for article in articles
<div>
<b>@article.title</b> <br />
@article.text
</div>
@end
2019-12-14 03:31:09 +01:00
</body>
```
```bash
v run .
```
2021-03-04 15:03:02 +01:00
<img width=662 src="https://github.com/vlang/v/blob/master/tutorials/building_a_simple_web_blog_with_vweb/img/articles1.png?raw=true">
2019-12-14 03:31:09 +01:00
That was very simple, wasn't it?
The built-in V ORM uses a syntax very similar to SQL. The queries are built with V.
For example, if we only wanted to find articles with ids between 100 and 200, we'd do:
```v oksyntax
// article.v
return sql app.db {
select from Article where id >= 100 && id <= 200
}
2019-12-14 03:31:09 +01:00
```
Retrieving a single article is very simple:
```v oksyntax
// article.v
2019-12-14 03:31:09 +01:00
pub fn (app &App) retrieve_article() ?Article {
return sql app.db {
select from Article limit 1
}
2019-12-14 03:31:09 +01:00
}
```
V ORM uses V's optionals for single values, which is very useful, since
bad queries will always be handled by the developer:
```v ignore
// article.v
2019-12-14 03:31:09 +01:00
article := app.retrieve_article(10) or {
2020-12-31 17:31:17 +01:00
app.text('Article not found')
2019-12-14 03:31:09 +01:00
return
}
```
2019-12-14 22:55:05 +01:00
### Adding new articles
Create `new.html`:
```html
<html>
<head>
<title>V Blog</title>
</head>
<body>
<form action="/new_article" method="post">
<input type="text" placeholder="Title" name="title" /> <br />
<textarea placeholder="Text" name="text"></textarea>
<input type="submit" />
</form>
</body>
2019-12-14 22:55:05 +01:00
</html>
```
```v ignore
// article.v
import vweb
[post]
2021-10-15 01:24:22 +02:00
pub fn (mut app App) new_article(title string, text string) vweb.Result {
if title == '' || text == '' {
return app.text('Empty text/title')
2019-12-14 22:55:05 +01:00
}
article := Article{
title: title
text: text
}
2020-07-04 18:56:18 +02:00
println(article)
sql app.db {
insert article into Article
}
return app.redirect('/')
2019-12-14 22:55:05 +01:00
}
```
The decorator on our function tells vweb that it is an HTTP POST type operation.
2019-12-14 22:55:05 +01:00
This time Vweb parses the HTTP form and assigns correct values with correct types to
function arguments, which saves a lot of typing (e.g. `title := app.form['title']` is
not necessary).
2019-12-14 22:55:05 +01:00
We need to update `index.html` to add a link to the "new article" page:
2019-12-14 23:00:04 +01:00
```html
<a href="/new">New article</a>
2019-12-14 23:00:04 +01:00
```
2019-12-14 22:55:05 +01:00
Next we need to add the HTML endpoint to our code like we did with `index.html`:
```v ignore
['/new']
pub fn (mut app App) new() vweb.Result {
return $vweb.html()
}
```
Re-running this code will now allow us to add new posts to our blog endpoint
2019-12-14 22:55:05 +01:00
### JSON endpoints
2020-01-18 17:03:47 +01:00
This tutorial used the traditional server-side rendering. If you prefer
2019-12-14 22:55:05 +01:00
to render everything on the client or need an API, creating JSON endpoints
in V is very simple:
```v oksyntax
// article.v
import vweb
import json
['/articles'; get]
pub fn (mut app App) articles() vweb.Result {
2019-12-14 22:55:05 +01:00
articles := app.find_all_articles()
return app.json(json.encode(articles))
2019-12-14 22:55:05 +01:00
}
```
2021-03-04 15:03:02 +01:00
<img width=662 src="https://github.com/vlang/v/blob/master/tutorials/building_a_simple_web_blog_with_vweb/img/articles_json.png?raw=true">
2019-12-14 22:55:05 +01:00
### Persistent data
If one wants to persist data they need to use a file instead of memory SQLite Database.
Replace the db setup code with this instead:
2021-08-03 15:13:34 +02:00
```
db: sqlite.connect('blog.db') or { panic(err) }
```
As we can see it attempts to open a file in the current directory named `blog.db`.
If the database file doesn't exist it will create it. The second command will
create the table `Article` if none exists already. Now every time the
app is run you will see the articles created from the previous executions
2019-12-14 22:55:05 +01:00
2019-12-16 23:21:10 +01:00
To be continued...
2019-12-14 03:31:09 +01:00
2019-12-14 06:24:14 +01:00
For an example of a more sophisticated web app written in V, check out Vorum: https://github.com/vlang/vorum