mirror of
https://github.com/vlang/v.git
synced 2025-08-03 17:57:59 -04:00
327 lines
9.9 KiB
V
327 lines
9.9 KiB
V
module veb
|
|
|
|
import json
|
|
import net
|
|
import net.http
|
|
import os
|
|
|
|
enum ContextReturnType {
|
|
normal
|
|
file
|
|
}
|
|
|
|
pub enum RedirectType {
|
|
found = int(http.Status.found)
|
|
moved_permanently = int(http.Status.moved_permanently)
|
|
see_other = int(http.Status.see_other)
|
|
temporary_redirect = int(http.Status.temporary_redirect)
|
|
permanent_redirect = int(http.Status.permanent_redirect)
|
|
}
|
|
|
|
// The Context struct represents the Context which holds the HTTP request and response.
|
|
// It has fields for the query, form, files and methods for handling the request and response
|
|
@[heap]
|
|
pub struct Context {
|
|
mut:
|
|
// veb will try to infer the content type base on file extension,
|
|
// and if `content_type` is not empty the `Content-Type` header will always be
|
|
// set to this value
|
|
content_type string
|
|
// done is set to true when a response can be sent over `conn`
|
|
done bool
|
|
// if true the response should not be sent and the connection should be closed
|
|
// manually.
|
|
takeover bool
|
|
// how the http response should be handled by veb's backend
|
|
return_type ContextReturnType = .normal
|
|
return_file string
|
|
// If the `Connection: close` header is present the connection should always be closed
|
|
client_wants_to_close bool
|
|
pub:
|
|
// TODO: move this to `handle_request`
|
|
// time.ticks() from start of veb connection handle.
|
|
// You can use it to determine how much time is spent on your request.
|
|
page_gen_start i64
|
|
pub mut:
|
|
req http.Request
|
|
custom_mime_types map[string]string
|
|
// TCP connection to client. Only for advanced usage!
|
|
conn &net.TcpConn = unsafe { nil }
|
|
// Map containing query params for the route.
|
|
// http://localhost:3000/index?q=vpm&order_by=desc => { 'q': 'vpm', 'order_by': 'desc' }
|
|
query map[string]string
|
|
// Multipart-form fields.
|
|
form map[string]string
|
|
// Files from multipart-form.
|
|
files map[string][]http.FileData
|
|
res http.Response
|
|
// use form_error to pass errors from the context to your frontend
|
|
form_error string
|
|
livereload_poll_interval_ms int = 250
|
|
}
|
|
|
|
// returns the request header data from the key
|
|
pub fn (ctx &Context) get_header(key http.CommonHeader) !string {
|
|
return ctx.req.header.get(key)!
|
|
}
|
|
|
|
// returns the request header data from the key
|
|
pub fn (ctx &Context) get_custom_header(key string) !string {
|
|
return ctx.req.header.get_custom(key)!
|
|
}
|
|
|
|
// set a header on the response object
|
|
pub fn (mut ctx Context) set_header(key http.CommonHeader, value string) {
|
|
ctx.res.header.set(key, value)
|
|
}
|
|
|
|
// set a custom header on the response object
|
|
pub fn (mut ctx Context) set_custom_header(key string, value string) ! {
|
|
ctx.res.header.set_custom(key, value)!
|
|
}
|
|
|
|
// send_response_to_client finalizes the response headers and sets Content-Type to `mimetype`
|
|
// and the response body to `response`
|
|
pub fn (mut ctx Context) send_response_to_client(mimetype string, response string) Result {
|
|
if ctx.done && !ctx.takeover {
|
|
eprintln('[veb] a response cannot be sent twice over one connection')
|
|
return Result{}
|
|
}
|
|
// ctx.done is only set in this function, so in order to sent a response over the connection
|
|
// this value has to be set to true. Assuming the user doesn't use `ctx.conn` directly.
|
|
ctx.done = true
|
|
ctx.res.body = response
|
|
$if veb_livereload ? {
|
|
if mimetype == 'text/html' {
|
|
ctx.res.body = response.replace('</html>', '<script src="/veb_livereload/${veb_livereload_server_start}/script.js"></script>\n</html>')
|
|
}
|
|
}
|
|
|
|
// set Content-Type and Content-Length headers
|
|
mut custom_mimetype := if ctx.content_type.len == 0 { mimetype } else { ctx.content_type }
|
|
if custom_mimetype != '' {
|
|
ctx.res.header.set(.content_type, custom_mimetype)
|
|
}
|
|
if ctx.res.body != '' {
|
|
ctx.res.header.set(.content_length, ctx.res.body.len.str())
|
|
}
|
|
// send veb's closing headers
|
|
ctx.res.header.set(.server, 'veb')
|
|
if !ctx.takeover && ctx.client_wants_to_close {
|
|
// Only sent the `Connection: close` header when the client wants to close
|
|
// the connection. This typically happens when the client only supports HTTP 1.0
|
|
ctx.res.header.set(.connection, 'close')
|
|
}
|
|
// set the http version
|
|
ctx.res.set_version(.v1_1)
|
|
if ctx.res.status_code == 0 {
|
|
ctx.res.set_status(.ok)
|
|
}
|
|
|
|
if ctx.takeover {
|
|
fast_send_resp(mut ctx.conn, ctx.res) or {}
|
|
}
|
|
// result is send in `veb.v`, `handle_route`
|
|
return Result{}
|
|
}
|
|
|
|
// Response with payload and content-type `text/html`
|
|
pub fn (mut ctx Context) html(s string) Result {
|
|
return ctx.send_response_to_client('text/html', s)
|
|
}
|
|
|
|
// Response with `s` as payload and content-type `text/plain`
|
|
pub fn (mut ctx Context) text(s string) Result {
|
|
return ctx.send_response_to_client('text/plain', s)
|
|
}
|
|
|
|
// Response with json_s as payload and content-type `application/json`
|
|
pub fn (mut ctx Context) json[T](j T) Result {
|
|
json_s := json.encode(j)
|
|
return ctx.send_response_to_client('application/json', json_s)
|
|
}
|
|
|
|
// Response with a pretty-printed JSON result
|
|
pub fn (mut ctx Context) json_pretty[T](j T) Result {
|
|
json_s := json.encode_pretty(j)
|
|
return ctx.send_response_to_client('application/json', json_s)
|
|
}
|
|
|
|
// Response HTTP_OK with file as payload
|
|
pub fn (mut ctx Context) file(file_path string) Result {
|
|
if !os.exists(file_path) {
|
|
eprintln('[veb] file "${file_path}" does not exist')
|
|
return ctx.not_found()
|
|
}
|
|
|
|
ext := os.file_ext(file_path)
|
|
|
|
mut content_type := ctx.content_type
|
|
if content_type.len == 0 {
|
|
if ct := ctx.custom_mime_types[ext] {
|
|
content_type = ct
|
|
} else {
|
|
content_type = mime_types[ext]
|
|
}
|
|
}
|
|
|
|
if content_type.len == 0 {
|
|
eprintln('[veb] no MIME type found for extension "${ext}"')
|
|
return ctx.server_error('')
|
|
}
|
|
|
|
return ctx.send_file(content_type, file_path)
|
|
}
|
|
|
|
fn (mut ctx Context) send_file(content_type string, file_path string) Result {
|
|
mut file := os.open(file_path) or {
|
|
eprint('[veb] error while trying to open file: ${err.msg()}')
|
|
ctx.res.set_status(.not_found)
|
|
return ctx.text('resource does not exist')
|
|
}
|
|
|
|
// seek from file end to get the file size
|
|
file.seek(0, .end) or {
|
|
eprintln('[veb] error while trying to read file: ${err.msg()}')
|
|
return ctx.server_error('could not read resource')
|
|
}
|
|
file_size := file.tell() or {
|
|
eprintln('[veb] error while trying to read file: ${err.msg()}')
|
|
return ctx.server_error('could not read resource')
|
|
}
|
|
file.close()
|
|
|
|
if ctx.takeover {
|
|
// it's a small file so we can send the response directly
|
|
data := os.read_file(file_path) or {
|
|
eprintln('[veb] error while trying to read file: ${err.msg()}')
|
|
return ctx.server_error('could not read resource')
|
|
}
|
|
return ctx.send_response_to_client(content_type, data)
|
|
} else {
|
|
ctx.return_type = .file
|
|
ctx.return_file = file_path
|
|
|
|
// set response headers
|
|
ctx.send_response_to_client(content_type, '')
|
|
ctx.res.header.set(.content_length, file_size.str())
|
|
return Result{}
|
|
}
|
|
}
|
|
|
|
// Response HTTP_OK with s as payload
|
|
pub fn (mut ctx Context) ok(s string) Result {
|
|
mut mime := if ctx.content_type.len == 0 { 'text/plain' } else { ctx.content_type }
|
|
return ctx.send_response_to_client(mime, s)
|
|
}
|
|
|
|
// send an error 400 with a message
|
|
pub fn (mut ctx Context) request_error(msg string) Result {
|
|
ctx.res.set_status(.bad_request)
|
|
return ctx.send_response_to_client('text/plain', msg)
|
|
}
|
|
|
|
// send an error 500 with a message
|
|
pub fn (mut ctx Context) server_error(msg string) Result {
|
|
ctx.res.set_status(.internal_server_error)
|
|
return ctx.send_response_to_client('text/plain', msg)
|
|
}
|
|
|
|
// send an error with a custom status
|
|
pub fn (mut ctx Context) server_error_with_status(s http.Status) Result {
|
|
ctx.res.set_status(s)
|
|
return ctx.send_response_to_client('text/plain', 'Server error')
|
|
}
|
|
|
|
// send a 204 No Content response without body and content-type
|
|
pub fn (mut ctx Context) no_content() Result {
|
|
ctx.res.set_status(.no_content)
|
|
return ctx.send_response_to_client('', '')
|
|
}
|
|
|
|
@[params]
|
|
pub struct RedirectParams {
|
|
pub:
|
|
typ RedirectType
|
|
}
|
|
|
|
// Redirect to an url
|
|
pub fn (mut ctx Context) redirect(url string, params RedirectParams) Result {
|
|
status := http.Status(params.typ)
|
|
ctx.res.set_status(status)
|
|
|
|
ctx.res.header.add(.location, url)
|
|
return ctx.send_response_to_client('text/plain', status.str())
|
|
}
|
|
|
|
// before_request is always the first function that is executed and acts as middleware
|
|
pub fn (mut ctx Context) before_request() Result {
|
|
return Result{}
|
|
}
|
|
|
|
// returns a HTTP 404 response
|
|
pub fn (mut ctx Context) not_found() Result {
|
|
ctx.res.set_status(.not_found)
|
|
return ctx.send_response_to_client('text/plain', '404 Not Found')
|
|
}
|
|
|
|
// Gets a cookie by a key
|
|
pub fn (ctx &Context) get_cookie(key string) ?string {
|
|
if cookie := ctx.req.cookie(key) {
|
|
return cookie.value
|
|
} else {
|
|
return none
|
|
}
|
|
}
|
|
|
|
// Sets a cookie
|
|
pub fn (mut ctx Context) set_cookie(cookie http.Cookie) {
|
|
cookie_raw := cookie.str()
|
|
if cookie_raw == '' {
|
|
eprintln('[veb] error setting cookie: name of cookie is invalid.\n${cookie}')
|
|
return
|
|
}
|
|
ctx.res.header.add(.set_cookie, cookie_raw)
|
|
}
|
|
|
|
// set_content_type sets the Content-Type header to `mime`
|
|
pub fn (mut ctx Context) set_content_type(mime string) {
|
|
ctx.content_type = mime
|
|
}
|
|
|
|
// takeover_conn prevents veb from automatically sending a response and closing
|
|
// the connection. You are responsible for closing the connection.
|
|
// In takeover mode if you call a Context method the response will be directly
|
|
// send over the connection and you can send multiple responses.
|
|
// This function is useful when you want to keep the connection alive and/or
|
|
// send multiple responses. Like with the SSE.
|
|
pub fn (mut ctx Context) takeover_conn() {
|
|
ctx.takeover = true
|
|
}
|
|
|
|
// user_agent returns the user-agent header for the current client
|
|
pub fn (ctx &Context) user_agent() string {
|
|
return ctx.req.header.get(.user_agent) or { '' }
|
|
}
|
|
|
|
// Returns the ip address from the current user
|
|
pub fn (ctx &Context) ip() string {
|
|
mut ip := ctx.req.header.get_custom('CF-Connecting-IP') or { '' }
|
|
if ip == '' {
|
|
ip = ctx.req.header.get(.x_forwarded_for) or { '' }
|
|
}
|
|
if ip == '' {
|
|
ip = ctx.req.header.get_custom('X-Forwarded-For') or { '' }
|
|
}
|
|
if ip == '' {
|
|
ip = ctx.req.header.get_custom('X-Real-Ip') or { '' }
|
|
}
|
|
if ip.contains(',') {
|
|
ip = ip.all_before(',')
|
|
}
|
|
if ip == '' {
|
|
ip = ctx.conn.peer_ip() or { '' }
|
|
}
|
|
return ip
|
|
}
|