Valval is the fastest web framework in V language.
This means you can develop a website quickly and run it even faster!
import valval
fn hello(req valval.Request) valval.Response {
return valval.response_ok('hello world')
}
fn main() {
mut app := valval.new_app(true)
app.register('/', hello)
valval.runserver(app, 8012)
}
$ git clone https://github.com/vlang/v
$ cd v
$ make
Install OpenSSL
macOS:
$ brew install openssl
Debian/Ubuntu:
$ sudo apt install libssl-dev openssl ca-certificates
Windows (Win10 Verified): Source can be downloaded from:
You can find a Graphic installer if that's more to you're liking.
Or use V in docker, it includes OpenSSL
docker run -it -p 8012:8012 --name vlang taojy123/vlang bash
$ git clone https://github.com/taojy123/valval
$ ln -s $(pwd)/valval ~/.vmodules/valval
Watchman123456 has registered the module with vpm. Simply use the following if you have v on your PATH variable:
$ v install watchman123456.valval
Note: If you use vpm; you'll have to change the import to:
import watchman123456.valval
As well as the usage to watchman123456.valval
A minimal Valval application looks something like this:
// demo.v
module main
import valval
fn hello(req valval.Request) valval.Response {
return valval.response_ok('hello world')
}
fn main() {
mut app := valval.new_app(true)
app.register('/', hello)
valval.runserver(app, 8012)
}
Run server
$ v run demo.v
Then you can visit http://127.0.0.1:8012/
to see the website
$ curl http://127.0.0.1:8012/
hello world
You can decide whether to use debug mode when calling valval.new_app()
mut app := valval.new_app(true) // debug mode
mut app := valval.new_app(false) // production mode
debug mode will print out more infomation while app running
You can decide the service port number when calling the valval.runserver()
valval.runserver(app, 8012) // listening 8012 port
valval.runserver(app, 80) // listening 80 port
The valval server will bind 0.0.0.0
address, so you can visit the website by 127.0.0.1:Port
or ServerIp:Port
Use the App.register()
function to band a handler function to request path
The handler function should have a parameter of type Request
and return a Response
mut app := valval.new_app(true)
app.register('/', hello) // http://127.0.0.1
app.register('/users', function1) // http://127.0.0.1/users
app.register('/user/info', function2) // http://127.0.0.1/user/info
app.register('POST:/book', function3) // http://127.0.0.1/book by POST
app.register('DELETE:/book', function4) // http://127.0.0.1/book by DELETE
app.register('/book', function5) // http://127.0.0.1/book by other methods
app.register('*', function6) // all remain
valval.runserver(app, 80)
Currently, only the following data can be parsed:
- query parameters by GET request; by
valval.Request.query[xxx]
x-www-form-urlencoded
parameters by POST / PUT / PATCH request; byvalval.Request.form[xxx]
fn hello(req valval.Request) valval.Response {
mut name = request.query['name']
if name == '' {
name = 'world'
}
return valval.response_ok('hello $name')
}
fn post_hello(req valval.Request) valval.Response {
mut name = request.form['name']
if name == '' {
name = 'world'
}
return valval.response_ok('hello $name')
}
app.register('GET:/hello', hello)
app.register('POST:/hello', post_hello)
valval.Request.get()
provides a quick way to get data whether it is from query
or form
.
fn hello(req valval.Request) valval.Response {
name = request.get('name', 'world') // default: 'world'
return valval.response_ok('hello $name')
}
app.register('/hello', hello)
More types of request data will be supported in the future:
- parameters in url
multipart/form-data
by POST requestapplication/json
by POST request- uploaded files
Use valval.App.serve_static
to serve local files
mut app := valval.new_app(true)
app.serve_static('/static/', './relative/path/to/static/')
// visit http://127.0.0.1/static/xxx.js ...
app.serve_static('/static2/', '/absolute/path/to/static2/')
// visit http://127.0.0.1/static2/yyy.css ...
valval.runserver(app, 80)
Valval used a whole new idea to implement the template function; inspired by Vue's system.
Has the following advantages:
- You don't need to spend time learning how to use templates, if you have used
Vue
before. - If you haven't used
Vue
, you also can learn it fast, because it's so easy. - It can integrate some commonly used UI frameworks, such as:
element
,mint
,vant
,antd
,bootstrap
... - I don't need to spend time developing built-in templates π.
An example for template:
server.v
:
import (
valval
json
)
struct User {
name string
age int
sex bool
}
fn users(req valval.Request) valval.Response {
// create a view by template file (`test6.html` can be a relative or absolute path)
// use `element` (https://github.com/ElemeFE/element) as ui framework
mut view := valval.new_view(req, 'users.html', 'element') or {
return valval.response_bad(err)
}
users := [
User{'Lucy', 13, false},
User{'Lily', 13, false},
User{'Jim', 12, true},
]
msg := 'This is a page of three user'
// use view.set to bind data for rendering template
// the second parameter must be a json string
view.set('users', json.encode(users))
view.set('msg', json.encode(msg))
return valval.response_view(view)
}
users.html
:
<html>
<head>
<title>Users Page</title>
</head>
<body>
<!-- Content in body can use template syntax -->
<h3>{{msg}}</h3>
<p v-for="u in users">
<span>{{u.name}}</span> ,
<span>{{u.age}}</span> ,
<el-tag v-if="u.sex">Male</el-tag>
<el-tag v-else>Female</el-tag>
</p>
</body>
</html>
Use valval.response_redirect()
to generate a redirect response
fn test1(req valval.Request) valval.Response {
name = req.get('name', '')
if name == '' {
return valval.response_redirect('/not_found')
}
return valval.response_ok('hello $name')
}
In addition to the responses mentioned above (response_ok
, response_view
, response_redirect
)
Valval also provides other response types, as follows:
struct User {
name string
age int
sex bool
}
fn text(req valval.Request) valval.Response {
return valval.response_text('this is plain text response')
}
fn json(req valval.Request) valval.Response {
user = User{'Tom', 12, true}
return valval.response_json(user)
// -> {"name": "Tom", "age": 12, "sex": true}
}
fn json_str(req valval.Request) valval.Response {
user = User{'Tom', 12, true}
user_str = json.encode(user)
return valval.response_json_str(user_str)
// -> {"name": "Tom", "age": 12, "sex": true}
}
fn file(req valval.Request) valval.Response {
return valval.response_file('path/to/local/file')
}
fn bad(req valval.Request) valval.Response {
return valval.response_bad('Parameter error!')
// response with statu code 400
}
- You can visit https://github.com/taojy123/valval/tree/master/example to see the complete example.
- And the official website of valval (https://valval.cool) is also written with the valval framework: https://github.com/taojy123/valval_website
- valval.new_app(debug bool) App
- valval.runserver(app App, port int)
- valval.new_view(req Request, template string, ui string) ?View
- valval.response_ok(content string) Response
- valval.response_text(content string) Response
- valval.response_json(obj T) Response
- valval.response_json_str(data string) Response
- valval.response_file(path string) Response
- valval.response_redirect(url string) Response
- valval.response_bad(msg string) Response
- valval.response_view(view View) Response
valval.Request {
β pub:
β app App
β method string
β path string
β query map[string]string
β form map[string]string
β body string
β headers map[string]string
}
- fn (req Request) get(key string, default_value string) string
- fn (req Request) is_api() bool
- fn (req Request) is_page() bool
valval.Response {
β pub mut:
β status int = 200
β body string = ''
β content_type string = 'text/html; charset=utf-8'
β headers map[string]string
}
- fn (res mut Response) set_header(key string, value string)
valval.View {
β req Request
β template string
β ui string = 'element'
β mut:
β context map[string]string
β pub:
β content string // html after template compiled
}
- fn (view mut View) set(key string, data string)
valval.App {
β pub:
β name string = 'ValvalApp'
β debug bool = true
β run_ts int = 0
β mut:
β router map[string]Handler
β static_map map[string]string
}
- fn (app mut App) register(path string, func fn(Request) Response)
- fn (app mut App) serve_static(static_prefix string, static_root string)
valval.Server {
β pub:
β address string = '0.0.0.0'
β port int = 8012
β mut:
β app App
}
- fn (server Server) run()