forked from vieter-v/vieter
				
			Merge pull request 'Rework of web framework' (#266) from Chewing_Bever/vieter:web-rework into dev
Reviewed-on: vieter-v/vieter#266database-fixes
						commit
						3e0a2584fa
					
				| 
						 | 
					@ -16,6 +16,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
 | 
				
			||||||
  providing a Git repository
 | 
					  providing a Git repository
 | 
				
			||||||
* CLI commands for searching the AUR & directly adding packages
 | 
					* CLI commands for searching the AUR & directly adding packages
 | 
				
			||||||
* HTTP routes for removing packages, arch-repos & repos
 | 
					* HTTP routes for removing packages, arch-repos & repos
 | 
				
			||||||
 | 
					* All endpoints serving files now support HTTP byte range requests
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Changed
 | 
					### Changed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -25,6 +26,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
 | 
				
			||||||
* Branch name for 'git' targets is now optional; if not provided, the
 | 
					* Branch name for 'git' targets is now optional; if not provided, the
 | 
				
			||||||
  repository will be cloned with the default branch
 | 
					  repository will be cloned with the default branch
 | 
				
			||||||
* Build containers now explicitely set the PATH variable
 | 
					* Build containers now explicitely set the PATH variable
 | 
				
			||||||
 | 
					* Refactor of web framework
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Removed
 | 
					### Removed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										7
									
								
								Makefile
								
								
								
								
							
							
						
						
									
										7
									
								
								Makefile
								
								
								
								
							| 
						 | 
					@ -83,13 +83,6 @@ fmt:
 | 
				
			||||||
test:
 | 
					test:
 | 
				
			||||||
	$(V) test $(SRC_DIR)
 | 
						$(V) test $(SRC_DIR)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Build & patch the V compiler
 | 
					 | 
				
			||||||
.PHONY: v
 | 
					 | 
				
			||||||
v: v/v
 | 
					 | 
				
			||||||
v/v:
 | 
					 | 
				
			||||||
	git clone --single-branch https://git.rustybever.be/vieter-v/v v
 | 
					 | 
				
			||||||
	make -C v
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
.PHONY: clean
 | 
					.PHONY: clean
 | 
				
			||||||
clean:
 | 
					clean:
 | 
				
			||||||
	rm -rf 'data' 'vieter' 'dvieter' 'pvieter' 'vieter.c' 'pkg' 'src/vieter' *.pkg.tar.zst 'suvieter' 'afvieter' '$(SRC_DIR)/_docs' 'docs/public'
 | 
						rm -rf 'data' 'vieter' 'dvieter' 'pvieter' 'vieter.c' 'pkg' 'src/vieter' *.pkg.tar.zst 'suvieter' 'afvieter' '$(SRC_DIR)/_docs' 'docs/public'
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										13
									
								
								README.md
								
								
								
								
							
							
						
						
									
										13
									
								
								README.md
								
								
								
								
							| 
						 | 
					@ -37,7 +37,6 @@ that.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Besides a V installer, Vieter also requires the following libraries to work:
 | 
					Besides a V installer, Vieter also requires the following libraries to work:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
* gc
 | 
					 | 
				
			||||||
* libarchive
 | 
					* libarchive
 | 
				
			||||||
* openssl
 | 
					* openssl
 | 
				
			||||||
* sqlite3
 | 
					* sqlite3
 | 
				
			||||||
| 
						 | 
					@ -48,15 +47,9 @@ update`.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Compiler
 | 
					### Compiler
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Vieter compiles with the standard Vlang compiler. However, I do maintain a
 | 
					I used to maintain a mirror that tracked the latest master, but nowadays, I
 | 
				
			||||||
[mirror](https://git.rustybever.be/vieter-v/v). This is to ensure my CI does
 | 
					maintain a Docker image containing the specific compiler version that Vieter
 | 
				
			||||||
not break without reason, as I control when & how frequently the mirror is
 | 
					builds with. Currently, this is V 0.3.
 | 
				
			||||||
updated to reflect the official repository.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
If you encounter issues using the latest V compiler, try using my mirror
 | 
					 | 
				
			||||||
instead. `make v` will clone the repository & build the mirror. Afterwards,
 | 
					 | 
				
			||||||
prepending any make command with `V_PATH=v/v` tells make to use the locally
 | 
					 | 
				
			||||||
compiled mirror instead.
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
## Contributing
 | 
					## Contributing
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,7 +2,7 @@ module client
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import net.http { Method }
 | 
					import net.http { Method }
 | 
				
			||||||
import net.urllib
 | 
					import net.urllib
 | 
				
			||||||
import response { Response }
 | 
					import web.response { Response }
 | 
				
			||||||
import json
 | 
					import json
 | 
				
			||||||
 | 
					
 | 
				
			||||||
pub struct Client {
 | 
					pub struct Client {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,7 +2,7 @@ module client
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import models { BuildLog, BuildLogFilter }
 | 
					import models { BuildLog, BuildLogFilter }
 | 
				
			||||||
import net.http { Method }
 | 
					import net.http { Method }
 | 
				
			||||||
import response { Response }
 | 
					import web.response { Response }
 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// get_build_logs returns all build logs.
 | 
					// get_build_logs returns all build logs.
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,7 +2,7 @@ module client
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import models { Target, TargetFilter }
 | 
					import models { Target, TargetFilter }
 | 
				
			||||||
import net.http { Method }
 | 
					import net.http { Method }
 | 
				
			||||||
import response { Response }
 | 
					import web.response { Response }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// get_targets returns a list of targets, given a filter object.
 | 
					// get_targets returns a list of targets, given a filter object.
 | 
				
			||||||
pub fn (c &Client) get_targets(filter TargetFilter) ?[]Target {
 | 
					pub fn (c &Client) get_targets(filter TargetFilter) ?[]Target {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,7 +3,7 @@ module server
 | 
				
			||||||
import web
 | 
					import web
 | 
				
			||||||
import net.http
 | 
					import net.http
 | 
				
			||||||
import net.urllib
 | 
					import net.urllib
 | 
				
			||||||
import response { new_data_response, new_response }
 | 
					import web.response { new_data_response, new_response }
 | 
				
			||||||
import db
 | 
					import db
 | 
				
			||||||
import time
 | 
					import time
 | 
				
			||||||
import os
 | 
					import os
 | 
				
			||||||
| 
						 | 
					@ -12,12 +12,8 @@ import models { BuildLog, BuildLogFilter }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_get_logs returns all build logs in the database. A 'target' query param can
 | 
					// v1_get_logs returns all build logs in the database. A 'target' query param can
 | 
				
			||||||
// optionally be added to limit the list of build logs to that repository.
 | 
					// optionally be added to limit the list of build logs to that repository.
 | 
				
			||||||
['/api/v1/logs'; get]
 | 
					['/api/v1/logs'; auth; get]
 | 
				
			||||||
fn (mut app App) v1_get_logs() web.Result {
 | 
					fn (mut app App) v1_get_logs() web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	filter := models.from_params<BuildLogFilter>(app.query) or {
 | 
						filter := models.from_params<BuildLogFilter>(app.query) or {
 | 
				
			||||||
		return app.json(http.Status.bad_request, new_response('Invalid query parameters.'))
 | 
							return app.json(http.Status.bad_request, new_response('Invalid query parameters.'))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -27,24 +23,16 @@ fn (mut app App) v1_get_logs() web.Result {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_get_single_log returns the build log with the given id.
 | 
					// v1_get_single_log returns the build log with the given id.
 | 
				
			||||||
['/api/v1/logs/:id'; get]
 | 
					['/api/v1/logs/:id'; auth; get]
 | 
				
			||||||
fn (mut app App) v1_get_single_log(id int) web.Result {
 | 
					fn (mut app App) v1_get_single_log(id int) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	log := app.db.get_build_log(id) or { return app.not_found() }
 | 
						log := app.db.get_build_log(id) or { return app.not_found() }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return app.json(http.Status.ok, new_data_response(log))
 | 
						return app.json(http.Status.ok, new_data_response(log))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_get_log_content returns the actual build log file for the given id.
 | 
					// v1_get_log_content returns the actual build log file for the given id.
 | 
				
			||||||
['/api/v1/logs/:id/content'; get]
 | 
					['/api/v1/logs/:id/content'; auth; get]
 | 
				
			||||||
fn (mut app App) v1_get_log_content(id int) web.Result {
 | 
					fn (mut app App) v1_get_log_content(id int) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	log := app.db.get_build_log(id) or { return app.not_found() }
 | 
						log := app.db.get_build_log(id) or { return app.not_found() }
 | 
				
			||||||
	file_name := log.start_time.custom_format('YYYY-MM-DD_HH-mm-ss')
 | 
						file_name := log.start_time.custom_format('YYYY-MM-DD_HH-mm-ss')
 | 
				
			||||||
	full_path := os.join_path(app.conf.data_dir, logs_dir_name, log.target_id.str(), log.arch,
 | 
						full_path := os.join_path(app.conf.data_dir, logs_dir_name, log.target_id.str(), log.arch,
 | 
				
			||||||
| 
						 | 
					@ -63,12 +51,8 @@ fn parse_query_time(query string) ?time.Time {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_post_log adds a new log to the database.
 | 
					// v1_post_log adds a new log to the database.
 | 
				
			||||||
['/api/v1/logs'; post]
 | 
					['/api/v1/logs'; auth; post]
 | 
				
			||||||
fn (mut app App) v1_post_log() web.Result {
 | 
					fn (mut app App) v1_post_log() web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// Parse query params
 | 
						// Parse query params
 | 
				
			||||||
	start_time_int := app.query['startTime'].int()
 | 
						start_time_int := app.query['startTime'].int()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,17 +2,13 @@ module server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import web
 | 
					import web
 | 
				
			||||||
import net.http
 | 
					import net.http
 | 
				
			||||||
import response { new_data_response, new_response }
 | 
					import web.response { new_data_response, new_response }
 | 
				
			||||||
import db
 | 
					import db
 | 
				
			||||||
import models { Target, TargetArch, TargetFilter }
 | 
					import models { Target, TargetArch, TargetFilter }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_get_targets returns the current list of targets.
 | 
					// v1_get_targets returns the current list of targets.
 | 
				
			||||||
['/api/v1/targets'; get]
 | 
					['/api/v1/targets'; auth; get]
 | 
				
			||||||
fn (mut app App) v1_get_targets() web.Result {
 | 
					fn (mut app App) v1_get_targets() web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	filter := models.from_params<TargetFilter>(app.query) or {
 | 
						filter := models.from_params<TargetFilter>(app.query) or {
 | 
				
			||||||
		return app.json(http.Status.bad_request, new_response('Invalid query parameters.'))
 | 
							return app.json(http.Status.bad_request, new_response('Invalid query parameters.'))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -22,24 +18,16 @@ fn (mut app App) v1_get_targets() web.Result {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_get_single_target returns the information for a single target.
 | 
					// v1_get_single_target returns the information for a single target.
 | 
				
			||||||
['/api/v1/targets/:id'; get]
 | 
					['/api/v1/targets/:id'; auth; get]
 | 
				
			||||||
fn (mut app App) v1_get_single_target(id int) web.Result {
 | 
					fn (mut app App) v1_get_single_target(id int) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	repo := app.db.get_target(id) or { return app.not_found() }
 | 
						repo := app.db.get_target(id) or { return app.not_found() }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return app.json(http.Status.ok, new_data_response(repo))
 | 
						return app.json(http.Status.ok, new_data_response(repo))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_post_target creates a new target from the provided query string.
 | 
					// v1_post_target creates a new target from the provided query string.
 | 
				
			||||||
['/api/v1/targets'; post]
 | 
					['/api/v1/targets'; auth; post]
 | 
				
			||||||
fn (mut app App) v1_post_target() web.Result {
 | 
					fn (mut app App) v1_post_target() web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	mut params := app.query.clone()
 | 
						mut params := app.query.clone()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// If a repo is created without specifying the arch, we assume it's meant
 | 
						// If a repo is created without specifying the arch, we assume it's meant
 | 
				
			||||||
| 
						 | 
					@ -63,24 +51,16 @@ fn (mut app App) v1_post_target() web.Result {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_delete_target removes a given target from the server's list.
 | 
					// v1_delete_target removes a given target from the server's list.
 | 
				
			||||||
['/api/v1/targets/:id'; delete]
 | 
					['/api/v1/targets/:id'; auth; delete]
 | 
				
			||||||
fn (mut app App) v1_delete_target(id int) web.Result {
 | 
					fn (mut app App) v1_delete_target(id int) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	app.db.delete_target(id)
 | 
						app.db.delete_target(id)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return app.json(http.Status.ok, new_response('Repo removed successfully.'))
 | 
						return app.json(http.Status.ok, new_response('Repo removed successfully.'))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// v1_patch_target updates a target's data with the given query params.
 | 
					// v1_patch_target updates a target's data with the given query params.
 | 
				
			||||||
['/api/v1/targets/:id'; patch]
 | 
					['/api/v1/targets/:id'; auth; patch]
 | 
				
			||||||
fn (mut app App) v1_patch_target(id int) web.Result {
 | 
					fn (mut app App) v1_patch_target(id int) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	app.db.update_target(id, app.query)
 | 
						app.db.update_target(id, app.query)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if 'arch' in app.query {
 | 
						if 'arch' in app.query {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,12 +0,0 @@
 | 
				
			||||||
module server
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import net.http
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// is_authorized checks whether the provided API key is correct.
 | 
					 | 
				
			||||||
fn (mut app App) is_authorized() bool {
 | 
					 | 
				
			||||||
	x_header := app.req.header.get_custom('X-Api-Key', http.HeaderQueryConfig{ exact: true }) or {
 | 
					 | 
				
			||||||
		return false
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return x_header.trim_space() == app.conf.api_key
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
| 
						 | 
					@ -7,13 +7,13 @@ import time
 | 
				
			||||||
import rand
 | 
					import rand
 | 
				
			||||||
import util
 | 
					import util
 | 
				
			||||||
import net.http
 | 
					import net.http
 | 
				
			||||||
import response { new_response }
 | 
					import web.response { new_response }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// healthcheck just returns a string, but can be used to quickly check if the
 | 
					// healthcheck just returns a string, but can be used to quickly check if the
 | 
				
			||||||
// server is still responsive.
 | 
					// server is still responsive.
 | 
				
			||||||
['/health'; get]
 | 
					['/health'; get]
 | 
				
			||||||
pub fn (mut app App) healthcheck() web.Result {
 | 
					pub fn (mut app App) healthcheck() web.Result {
 | 
				
			||||||
	return app.json(http.Status.ok, new_response('Healthy.'))
 | 
						return app.json(.ok, new_response('Healthy.'))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// get_repo_file handles all Pacman-related routes. It returns both the
 | 
					// get_repo_file handles all Pacman-related routes. It returns both the
 | 
				
			||||||
| 
						 | 
					@ -45,25 +45,12 @@ fn (mut app App) get_repo_file(repo string, arch string, filename string) web.Re
 | 
				
			||||||
		full_path = os.join_path(app.repo.repos_dir, repo, arch, filename, 'desc')
 | 
							full_path = os.join_path(app.repo.repos_dir, repo, arch, filename, 'desc')
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Scuffed way to respond to HEAD requests
 | 
					 | 
				
			||||||
	if app.req.method == http.Method.head {
 | 
					 | 
				
			||||||
		if os.exists(full_path) {
 | 
					 | 
				
			||||||
			return app.status(http.Status.ok)
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return app.not_found()
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return app.file(full_path)
 | 
						return app.file(full_path)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// put_package handles publishing a package to a repository.
 | 
					// put_package handles publishing a package to a repository.
 | 
				
			||||||
['/:repo/publish'; post]
 | 
					['/:repo/publish'; auth; post]
 | 
				
			||||||
fn (mut app App) put_package(repo string) web.Result {
 | 
					fn (mut app App) put_package(repo string) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	mut pkg_path := ''
 | 
						mut pkg_path := ''
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if length := app.req.header.get(.content_length) {
 | 
						if length := app.req.header.get(.content_length) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,15 +2,11 @@ module server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import web
 | 
					import web
 | 
				
			||||||
import net.http
 | 
					import net.http
 | 
				
			||||||
import response { new_response }
 | 
					import web.response { new_response }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// delete_package tries to remove the given package.
 | 
					// delete_package tries to remove the given package.
 | 
				
			||||||
['/:repo/:arch/:pkg'; delete]
 | 
					['/:repo/:arch/:pkg'; auth; delete]
 | 
				
			||||||
fn (mut app App) delete_package(repo string, arch string, pkg string) web.Result {
 | 
					fn (mut app App) delete_package(repo string, arch string, pkg string) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	res := app.repo.remove_pkg_from_arch_repo(repo, arch, pkg, true) or {
 | 
						res := app.repo.remove_pkg_from_arch_repo(repo, arch, pkg, true) or {
 | 
				
			||||||
		app.lerror('Error while deleting package: $err.msg()')
 | 
							app.lerror('Error while deleting package: $err.msg()')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -29,12 +25,8 @@ fn (mut app App) delete_package(repo string, arch string, pkg string) web.Result
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// delete_arch_repo tries to remove the given arch-repo.
 | 
					// delete_arch_repo tries to remove the given arch-repo.
 | 
				
			||||||
['/:repo/:arch'; delete]
 | 
					['/:repo/:arch'; auth; delete]
 | 
				
			||||||
fn (mut app App) delete_arch_repo(repo string, arch string) web.Result {
 | 
					fn (mut app App) delete_arch_repo(repo string, arch string) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	res := app.repo.remove_arch_repo(repo, arch) or {
 | 
						res := app.repo.remove_arch_repo(repo, arch) or {
 | 
				
			||||||
		app.lerror('Error while deleting arch-repo: $err.msg()')
 | 
							app.lerror('Error while deleting arch-repo: $err.msg()')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -53,12 +45,8 @@ fn (mut app App) delete_arch_repo(repo string, arch string) web.Result {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// delete_repo tries to remove the given repo.
 | 
					// delete_repo tries to remove the given repo.
 | 
				
			||||||
['/:repo'; delete]
 | 
					['/:repo'; auth; delete]
 | 
				
			||||||
fn (mut app App) delete_repo(repo string) web.Result {
 | 
					fn (mut app App) delete_repo(repo string) web.Result {
 | 
				
			||||||
	if !app.is_authorized() {
 | 
					 | 
				
			||||||
		return app.json(http.Status.unauthorized, new_response('Unauthorized.'))
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	res := app.repo.remove_repo(repo) or {
 | 
						res := app.repo.remove_repo(repo) or {
 | 
				
			||||||
		app.lerror('Error while deleting repo: $err.msg()')
 | 
							app.lerror('Error while deleting repo: $err.msg()')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -73,6 +73,7 @@ pub fn server(conf Config) ? {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	web.run(&App{
 | 
						web.run(&App{
 | 
				
			||||||
		logger: logger
 | 
							logger: logger
 | 
				
			||||||
 | 
							api_key: conf.api_key
 | 
				
			||||||
		conf: conf
 | 
							conf: conf
 | 
				
			||||||
		repo: repo
 | 
							repo: repo
 | 
				
			||||||
		db: db
 | 
							db: db
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,133 @@
 | 
				
			||||||
 | 
					module web
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import net.http
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// A dummy structure that returns from routes to indicate that you actually sent something to a user
 | 
				
			||||||
 | 
					[noinit]
 | 
				
			||||||
 | 
					pub struct Result {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pub const (
 | 
				
			||||||
 | 
						methods_with_form = [http.Method.post, .put, .patch]
 | 
				
			||||||
 | 
						headers_close     = http.new_custom_header_from_map({
 | 
				
			||||||
 | 
							'Server':                           'Vieter'
 | 
				
			||||||
 | 
							http.CommonHeader.connection.str(): 'close'
 | 
				
			||||||
 | 
						}) or { panic('should never fail') }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						http_302          = http.new_response(
 | 
				
			||||||
 | 
							status: .found
 | 
				
			||||||
 | 
							body: '302 Found'
 | 
				
			||||||
 | 
							header: headers_close
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						http_400          = http.new_response(
 | 
				
			||||||
 | 
							status: .bad_request
 | 
				
			||||||
 | 
							body: '400 Bad Request'
 | 
				
			||||||
 | 
							header: http.new_header(
 | 
				
			||||||
 | 
								key: .content_type
 | 
				
			||||||
 | 
								value: 'text/plain'
 | 
				
			||||||
 | 
							).join(headers_close)
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						http_401          = http.new_response(
 | 
				
			||||||
 | 
							status: .unauthorized
 | 
				
			||||||
 | 
							body: '401 Unauthorized'
 | 
				
			||||||
 | 
							header: http.new_header(
 | 
				
			||||||
 | 
								key: .content_type
 | 
				
			||||||
 | 
								value: 'text/plain'
 | 
				
			||||||
 | 
							).join(headers_close)
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						http_404          = http.new_response(
 | 
				
			||||||
 | 
							status: .not_found
 | 
				
			||||||
 | 
							body: '404 Not Found'
 | 
				
			||||||
 | 
							header: http.new_header(
 | 
				
			||||||
 | 
								key: .content_type
 | 
				
			||||||
 | 
								value: 'text/plain'
 | 
				
			||||||
 | 
							).join(headers_close)
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						http_500          = http.new_response(
 | 
				
			||||||
 | 
							status: .internal_server_error
 | 
				
			||||||
 | 
							body: '500 Internal Server Error'
 | 
				
			||||||
 | 
							header: http.new_header(
 | 
				
			||||||
 | 
								key: .content_type
 | 
				
			||||||
 | 
								value: 'text/plain'
 | 
				
			||||||
 | 
							).join(headers_close)
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						mime_types        = {
 | 
				
			||||||
 | 
							'.aac':    'audio/aac'
 | 
				
			||||||
 | 
							'.abw':    'application/x-abiword'
 | 
				
			||||||
 | 
							'.arc':    'application/x-freearc'
 | 
				
			||||||
 | 
							'.avi':    'video/x-msvideo'
 | 
				
			||||||
 | 
							'.azw':    'application/vnd.amazon.ebook'
 | 
				
			||||||
 | 
							'.bin':    'application/octet-stream'
 | 
				
			||||||
 | 
							'.bmp':    'image/bmp'
 | 
				
			||||||
 | 
							'.bz':     'application/x-bzip'
 | 
				
			||||||
 | 
							'.bz2':    'application/x-bzip2'
 | 
				
			||||||
 | 
							'.cda':    'application/x-cdf'
 | 
				
			||||||
 | 
							'.csh':    'application/x-csh'
 | 
				
			||||||
 | 
							'.css':    'text/css'
 | 
				
			||||||
 | 
							'.csv':    'text/csv'
 | 
				
			||||||
 | 
							'.doc':    'application/msword'
 | 
				
			||||||
 | 
							'.docx':   'application/vnd.openxmlformats-officedocument.wordprocessingml.document'
 | 
				
			||||||
 | 
							'.eot':    'application/vnd.ms-fontobject'
 | 
				
			||||||
 | 
							'.epub':   'application/epub+zip'
 | 
				
			||||||
 | 
							'.gz':     'application/gzip'
 | 
				
			||||||
 | 
							'.gif':    'image/gif'
 | 
				
			||||||
 | 
							'.htm':    'text/html'
 | 
				
			||||||
 | 
							'.html':   'text/html'
 | 
				
			||||||
 | 
							'.ico':    'image/vnd.microsoft.icon'
 | 
				
			||||||
 | 
							'.ics':    'text/calendar'
 | 
				
			||||||
 | 
							'.jar':    'application/java-archive'
 | 
				
			||||||
 | 
							'.jpeg':   'image/jpeg'
 | 
				
			||||||
 | 
							'.jpg':    'image/jpeg'
 | 
				
			||||||
 | 
							'.js':     'text/javascript'
 | 
				
			||||||
 | 
							'.json':   'application/json'
 | 
				
			||||||
 | 
							'.jsonld': 'application/ld+json'
 | 
				
			||||||
 | 
							'.mid':    'audio/midi audio/x-midi'
 | 
				
			||||||
 | 
							'.midi':   'audio/midi audio/x-midi'
 | 
				
			||||||
 | 
							'.mjs':    'text/javascript'
 | 
				
			||||||
 | 
							'.mp3':    'audio/mpeg'
 | 
				
			||||||
 | 
							'.mp4':    'video/mp4'
 | 
				
			||||||
 | 
							'.mpeg':   'video/mpeg'
 | 
				
			||||||
 | 
							'.mpkg':   'application/vnd.apple.installer+xml'
 | 
				
			||||||
 | 
							'.odp':    'application/vnd.oasis.opendocument.presentation'
 | 
				
			||||||
 | 
							'.ods':    'application/vnd.oasis.opendocument.spreadsheet'
 | 
				
			||||||
 | 
							'.odt':    'application/vnd.oasis.opendocument.text'
 | 
				
			||||||
 | 
							'.oga':    'audio/ogg'
 | 
				
			||||||
 | 
							'.ogv':    'video/ogg'
 | 
				
			||||||
 | 
							'.ogx':    'application/ogg'
 | 
				
			||||||
 | 
							'.opus':   'audio/opus'
 | 
				
			||||||
 | 
							'.otf':    'font/otf'
 | 
				
			||||||
 | 
							'.png':    'image/png'
 | 
				
			||||||
 | 
							'.pdf':    'application/pdf'
 | 
				
			||||||
 | 
							'.php':    'application/x-httpd-php'
 | 
				
			||||||
 | 
							'.ppt':    'application/vnd.ms-powerpoint'
 | 
				
			||||||
 | 
							'.pptx':   'application/vnd.openxmlformats-officedocument.presentationml.presentation'
 | 
				
			||||||
 | 
							'.rar':    'application/vnd.rar'
 | 
				
			||||||
 | 
							'.rtf':    'application/rtf'
 | 
				
			||||||
 | 
							'.sh':     'application/x-sh'
 | 
				
			||||||
 | 
							'.svg':    'image/svg+xml'
 | 
				
			||||||
 | 
							'.swf':    'application/x-shockwave-flash'
 | 
				
			||||||
 | 
							'.tar':    'application/x-tar'
 | 
				
			||||||
 | 
							'.tif':    'image/tiff'
 | 
				
			||||||
 | 
							'.tiff':   'image/tiff'
 | 
				
			||||||
 | 
							'.ts':     'video/mp2t'
 | 
				
			||||||
 | 
							'.ttf':    'font/ttf'
 | 
				
			||||||
 | 
							'.txt':    'text/plain'
 | 
				
			||||||
 | 
							'.vsd':    'application/vnd.visio'
 | 
				
			||||||
 | 
							'.wav':    'audio/wav'
 | 
				
			||||||
 | 
							'.weba':   'audio/webm'
 | 
				
			||||||
 | 
							'.webm':   'video/webm'
 | 
				
			||||||
 | 
							'.webp':   'image/webp'
 | 
				
			||||||
 | 
							'.woff':   'font/woff'
 | 
				
			||||||
 | 
							'.woff2':  'font/woff2'
 | 
				
			||||||
 | 
							'.xhtml':  'application/xhtml+xml'
 | 
				
			||||||
 | 
							'.xls':    'application/vnd.ms-excel'
 | 
				
			||||||
 | 
							'.xlsx':   'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
 | 
				
			||||||
 | 
							'.xml':    'application/xml'
 | 
				
			||||||
 | 
							'.xul':    'application/vnd.mozilla.xul+xml'
 | 
				
			||||||
 | 
							'.zip':    'application/zip'
 | 
				
			||||||
 | 
							'.3gp':    'video/3gpp'
 | 
				
			||||||
 | 
							'.3g2':    'video/3gpp2'
 | 
				
			||||||
 | 
							'.7z':     'application/x-7z-compressed'
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						max_http_post_size = 1024 * 1024
 | 
				
			||||||
 | 
						default_port       = 8080
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,10 @@ module web
 | 
				
			||||||
import net.urllib
 | 
					import net.urllib
 | 
				
			||||||
import net.http
 | 
					import net.http
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Method attributes that should be ignored when parsing, as they're used
 | 
				
			||||||
 | 
					// elsewhere.
 | 
				
			||||||
 | 
					const attrs_to_ignore = ['auth']
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Parsing function attributes for methods and path.
 | 
					// Parsing function attributes for methods and path.
 | 
				
			||||||
fn parse_attrs(name string, attrs []string) ?([]http.Method, string) {
 | 
					fn parse_attrs(name string, attrs []string) ?([]http.Method, string) {
 | 
				
			||||||
	if attrs.len == 0 {
 | 
						if attrs.len == 0 {
 | 
				
			||||||
| 
						 | 
					@ -32,7 +36,7 @@ fn parse_attrs(name string, attrs []string) ?([]http.Method, string) {
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		i++
 | 
							i++
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	if x.len > 0 {
 | 
						if x.len > 0 && x.any(!web.attrs_to_ignore.contains(it)) {
 | 
				
			||||||
		return IError(http.UnexpectedExtraAttributeError{
 | 
							return IError(http.UnexpectedExtraAttributeError{
 | 
				
			||||||
			attributes: x
 | 
								attributes: x
 | 
				
			||||||
		})
 | 
							})
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										440
									
								
								src/web/web.v
								
								
								
								
							
							
						
						
									
										440
									
								
								src/web/web.v
								
								
								
								
							| 
						 | 
					@ -12,146 +12,25 @@ import time
 | 
				
			||||||
import json
 | 
					import json
 | 
				
			||||||
import log
 | 
					import log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// A dummy structure that returns from routes to indicate that you actually sent something to a user
 | 
					 | 
				
			||||||
[noinit]
 | 
					 | 
				
			||||||
pub struct Result {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
pub const (
 | 
					 | 
				
			||||||
	methods_with_form = [http.Method.post, .put, .patch]
 | 
					 | 
				
			||||||
	headers_close     = http.new_custom_header_from_map({
 | 
					 | 
				
			||||||
		'Server':                           'VWeb'
 | 
					 | 
				
			||||||
		http.CommonHeader.connection.str(): 'close'
 | 
					 | 
				
			||||||
	}) or { panic('should never fail') }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	http_302          = http.new_response(
 | 
					 | 
				
			||||||
		status: .found
 | 
					 | 
				
			||||||
		body: '302 Found'
 | 
					 | 
				
			||||||
		header: headers_close
 | 
					 | 
				
			||||||
	)
 | 
					 | 
				
			||||||
	http_400          = http.new_response(
 | 
					 | 
				
			||||||
		status: .bad_request
 | 
					 | 
				
			||||||
		body: '400 Bad Request'
 | 
					 | 
				
			||||||
		header: http.new_header(
 | 
					 | 
				
			||||||
			key: .content_type
 | 
					 | 
				
			||||||
			value: 'text/plain'
 | 
					 | 
				
			||||||
		).join(headers_close)
 | 
					 | 
				
			||||||
	)
 | 
					 | 
				
			||||||
	http_404          = http.new_response(
 | 
					 | 
				
			||||||
		status: .not_found
 | 
					 | 
				
			||||||
		body: '404 Not Found'
 | 
					 | 
				
			||||||
		header: http.new_header(
 | 
					 | 
				
			||||||
			key: .content_type
 | 
					 | 
				
			||||||
			value: 'text/plain'
 | 
					 | 
				
			||||||
		).join(headers_close)
 | 
					 | 
				
			||||||
	)
 | 
					 | 
				
			||||||
	http_500          = http.new_response(
 | 
					 | 
				
			||||||
		status: .internal_server_error
 | 
					 | 
				
			||||||
		body: '500 Internal Server Error'
 | 
					 | 
				
			||||||
		header: http.new_header(
 | 
					 | 
				
			||||||
			key: .content_type
 | 
					 | 
				
			||||||
			value: 'text/plain'
 | 
					 | 
				
			||||||
		).join(headers_close)
 | 
					 | 
				
			||||||
	)
 | 
					 | 
				
			||||||
	mime_types        = {
 | 
					 | 
				
			||||||
		'.aac':    'audio/aac'
 | 
					 | 
				
			||||||
		'.abw':    'application/x-abiword'
 | 
					 | 
				
			||||||
		'.arc':    'application/x-freearc'
 | 
					 | 
				
			||||||
		'.avi':    'video/x-msvideo'
 | 
					 | 
				
			||||||
		'.azw':    'application/vnd.amazon.ebook'
 | 
					 | 
				
			||||||
		'.bin':    'application/octet-stream'
 | 
					 | 
				
			||||||
		'.bmp':    'image/bmp'
 | 
					 | 
				
			||||||
		'.bz':     'application/x-bzip'
 | 
					 | 
				
			||||||
		'.bz2':    'application/x-bzip2'
 | 
					 | 
				
			||||||
		'.cda':    'application/x-cdf'
 | 
					 | 
				
			||||||
		'.csh':    'application/x-csh'
 | 
					 | 
				
			||||||
		'.css':    'text/css'
 | 
					 | 
				
			||||||
		'.csv':    'text/csv'
 | 
					 | 
				
			||||||
		'.doc':    'application/msword'
 | 
					 | 
				
			||||||
		'.docx':   'application/vnd.openxmlformats-officedocument.wordprocessingml.document'
 | 
					 | 
				
			||||||
		'.eot':    'application/vnd.ms-fontobject'
 | 
					 | 
				
			||||||
		'.epub':   'application/epub+zip'
 | 
					 | 
				
			||||||
		'.gz':     'application/gzip'
 | 
					 | 
				
			||||||
		'.gif':    'image/gif'
 | 
					 | 
				
			||||||
		'.htm':    'text/html'
 | 
					 | 
				
			||||||
		'.html':   'text/html'
 | 
					 | 
				
			||||||
		'.ico':    'image/vnd.microsoft.icon'
 | 
					 | 
				
			||||||
		'.ics':    'text/calendar'
 | 
					 | 
				
			||||||
		'.jar':    'application/java-archive'
 | 
					 | 
				
			||||||
		'.jpeg':   'image/jpeg'
 | 
					 | 
				
			||||||
		'.jpg':    'image/jpeg'
 | 
					 | 
				
			||||||
		'.js':     'text/javascript'
 | 
					 | 
				
			||||||
		'.json':   'application/json'
 | 
					 | 
				
			||||||
		'.jsonld': 'application/ld+json'
 | 
					 | 
				
			||||||
		'.mid':    'audio/midi audio/x-midi'
 | 
					 | 
				
			||||||
		'.midi':   'audio/midi audio/x-midi'
 | 
					 | 
				
			||||||
		'.mjs':    'text/javascript'
 | 
					 | 
				
			||||||
		'.mp3':    'audio/mpeg'
 | 
					 | 
				
			||||||
		'.mp4':    'video/mp4'
 | 
					 | 
				
			||||||
		'.mpeg':   'video/mpeg'
 | 
					 | 
				
			||||||
		'.mpkg':   'application/vnd.apple.installer+xml'
 | 
					 | 
				
			||||||
		'.odp':    'application/vnd.oasis.opendocument.presentation'
 | 
					 | 
				
			||||||
		'.ods':    'application/vnd.oasis.opendocument.spreadsheet'
 | 
					 | 
				
			||||||
		'.odt':    'application/vnd.oasis.opendocument.text'
 | 
					 | 
				
			||||||
		'.oga':    'audio/ogg'
 | 
					 | 
				
			||||||
		'.ogv':    'video/ogg'
 | 
					 | 
				
			||||||
		'.ogx':    'application/ogg'
 | 
					 | 
				
			||||||
		'.opus':   'audio/opus'
 | 
					 | 
				
			||||||
		'.otf':    'font/otf'
 | 
					 | 
				
			||||||
		'.png':    'image/png'
 | 
					 | 
				
			||||||
		'.pdf':    'application/pdf'
 | 
					 | 
				
			||||||
		'.php':    'application/x-httpd-php'
 | 
					 | 
				
			||||||
		'.ppt':    'application/vnd.ms-powerpoint'
 | 
					 | 
				
			||||||
		'.pptx':   'application/vnd.openxmlformats-officedocument.presentationml.presentation'
 | 
					 | 
				
			||||||
		'.rar':    'application/vnd.rar'
 | 
					 | 
				
			||||||
		'.rtf':    'application/rtf'
 | 
					 | 
				
			||||||
		'.sh':     'application/x-sh'
 | 
					 | 
				
			||||||
		'.svg':    'image/svg+xml'
 | 
					 | 
				
			||||||
		'.swf':    'application/x-shockwave-flash'
 | 
					 | 
				
			||||||
		'.tar':    'application/x-tar'
 | 
					 | 
				
			||||||
		'.tif':    'image/tiff'
 | 
					 | 
				
			||||||
		'.tiff':   'image/tiff'
 | 
					 | 
				
			||||||
		'.ts':     'video/mp2t'
 | 
					 | 
				
			||||||
		'.ttf':    'font/ttf'
 | 
					 | 
				
			||||||
		'.txt':    'text/plain'
 | 
					 | 
				
			||||||
		'.vsd':    'application/vnd.visio'
 | 
					 | 
				
			||||||
		'.wav':    'audio/wav'
 | 
					 | 
				
			||||||
		'.weba':   'audio/webm'
 | 
					 | 
				
			||||||
		'.webm':   'video/webm'
 | 
					 | 
				
			||||||
		'.webp':   'image/webp'
 | 
					 | 
				
			||||||
		'.woff':   'font/woff'
 | 
					 | 
				
			||||||
		'.woff2':  'font/woff2'
 | 
					 | 
				
			||||||
		'.xhtml':  'application/xhtml+xml'
 | 
					 | 
				
			||||||
		'.xls':    'application/vnd.ms-excel'
 | 
					 | 
				
			||||||
		'.xlsx':   'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
 | 
					 | 
				
			||||||
		'.xml':    'application/xml'
 | 
					 | 
				
			||||||
		'.xul':    'application/vnd.mozilla.xul+xml'
 | 
					 | 
				
			||||||
		'.zip':    'application/zip'
 | 
					 | 
				
			||||||
		'.3gp':    'video/3gpp'
 | 
					 | 
				
			||||||
		'.3g2':    'video/3gpp2'
 | 
					 | 
				
			||||||
		'.7z':     'application/x-7z-compressed'
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	max_http_post_size = 1024 * 1024
 | 
					 | 
				
			||||||
	default_port       = 8080
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// The Context struct represents the Context which hold the HTTP request and response.
 | 
					// The Context struct represents the Context which hold the HTTP request and response.
 | 
				
			||||||
// It has fields for the query, form, files.
 | 
					// It has fields for the query, form, files.
 | 
				
			||||||
pub struct Context {
 | 
					pub struct Context {
 | 
				
			||||||
mut:
 | 
					 | 
				
			||||||
	content_type string      = 'text/plain'
 | 
					 | 
				
			||||||
	status       http.Status = http.Status.ok
 | 
					 | 
				
			||||||
pub:
 | 
					pub:
 | 
				
			||||||
	// HTTP Request
 | 
						// HTTP Request
 | 
				
			||||||
	req http.Request
 | 
						req http.Request
 | 
				
			||||||
 | 
						// API key used when authenticating requests
 | 
				
			||||||
 | 
						api_key string
 | 
				
			||||||
	// TODO Response
 | 
						// TODO Response
 | 
				
			||||||
pub mut:
 | 
					pub mut:
 | 
				
			||||||
	done bool
 | 
						// TCP connection to client.
 | 
				
			||||||
 | 
						// But beware, do not store it for further use, after request processing web will close connection.
 | 
				
			||||||
 | 
						conn &net.TcpConn
 | 
				
			||||||
 | 
						// Gives access to a shared logger object
 | 
				
			||||||
 | 
						logger shared log.Log
 | 
				
			||||||
	// time.ticks() from start of web connection handle.
 | 
						// time.ticks() from start of web connection handle.
 | 
				
			||||||
	// You can use it to determine how much time is spent on your request.
 | 
						// You can use it to determine how much time is spent on your request.
 | 
				
			||||||
	page_gen_start i64
 | 
						page_gen_start i64
 | 
				
			||||||
	// TCP connection to client.
 | 
						// REQUEST
 | 
				
			||||||
	// But beware, do not store it for further use, after request processing web will close connection.
 | 
					 | 
				
			||||||
	conn              &net.TcpConn
 | 
					 | 
				
			||||||
	static_files      map[string]string
 | 
						static_files      map[string]string
 | 
				
			||||||
	static_mime_types map[string]string
 | 
						static_mime_types map[string]string
 | 
				
			||||||
	// Map containing query params for the route.
 | 
						// Map containing query params for the route.
 | 
				
			||||||
| 
						 | 
					@ -161,14 +40,13 @@ pub mut:
 | 
				
			||||||
	form map[string]string
 | 
						form map[string]string
 | 
				
			||||||
	// Files from multipart-form.
 | 
						// Files from multipart-form.
 | 
				
			||||||
	files map[string][]http.FileData
 | 
						files map[string][]http.FileData
 | 
				
			||||||
 | 
					 | 
				
			||||||
	header http.Header // response headers
 | 
					 | 
				
			||||||
	// ? It doesn't seem to be used anywhere
 | 
					 | 
				
			||||||
	form_error string
 | 
					 | 
				
			||||||
	// Allows reading the request body
 | 
						// Allows reading the request body
 | 
				
			||||||
	reader io.BufferedReader
 | 
						reader io.BufferedReader
 | 
				
			||||||
	// Gives access to a shared logger object
 | 
						// RESPONSE
 | 
				
			||||||
	logger shared log.Log
 | 
						status       http.Status = http.Status.ok
 | 
				
			||||||
 | 
						content_type string      = 'text/plain'
 | 
				
			||||||
 | 
						// response headers
 | 
				
			||||||
 | 
						header http.Header
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct FileData {
 | 
					struct FileData {
 | 
				
			||||||
| 
						 | 
					@ -188,50 +66,92 @@ struct Route {
 | 
				
			||||||
// Probably you can use it for check user session cookie or add header.
 | 
					// Probably you can use it for check user session cookie or add header.
 | 
				
			||||||
pub fn (ctx Context) before_request() {}
 | 
					pub fn (ctx Context) before_request() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// send_string
 | 
					// send_string writes the given string to the TCP connection socket.
 | 
				
			||||||
fn send_string(mut conn net.TcpConn, s string) ? {
 | 
					fn (mut ctx Context) send_string(s string) ? {
 | 
				
			||||||
	conn.write(s.bytes())?
 | 
						ctx.conn.write(s.bytes())?
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// send_response_to_client sends a response to the client
 | 
					// send_reader reads at most `size` bytes from the given reader & writes them
 | 
				
			||||||
[manualfree]
 | 
					// to the TCP connection socket. Internally, a 10KB buffer is used, to avoid
 | 
				
			||||||
pub fn (mut ctx Context) send_response_to_client(mimetype string, res string) bool {
 | 
					// having to store all bytes in memory at once.
 | 
				
			||||||
	if ctx.done {
 | 
					fn (mut ctx Context) send_reader(mut reader io.Reader, size u64) ? {
 | 
				
			||||||
		return false
 | 
						mut buf := []u8{len: 10_000}
 | 
				
			||||||
	}
 | 
						mut bytes_left := size
 | 
				
			||||||
	ctx.done = true
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// build header
 | 
						// Repeat as long as the stream still has data
 | 
				
			||||||
	header := http.new_header_from_map({
 | 
						for bytes_left > 0 {
 | 
				
			||||||
		http.CommonHeader.content_type:   mimetype
 | 
							bytes_read := reader.read(mut buf)?
 | 
				
			||||||
		http.CommonHeader.content_length: res.len.str()
 | 
							bytes_left -= u64(bytes_read)
 | 
				
			||||||
	}).join(ctx.header)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	mut resp := http.Response{
 | 
							mut to_write := bytes_read
 | 
				
			||||||
		header: header.join(web.headers_close)
 | 
					
 | 
				
			||||||
		body: res
 | 
							for to_write > 0 {
 | 
				
			||||||
 | 
								bytes_written := ctx.conn.write(buf[bytes_read - to_write..bytes_read]) or { break }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								to_write = to_write - bytes_written
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	resp.set_version(.v1_1)
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// send_custom_response sends the given http.Response to the client. It can be
 | 
				
			||||||
 | 
					// used to overwrite the Context object & send a completely custom
 | 
				
			||||||
 | 
					// http.Response instead.
 | 
				
			||||||
 | 
					fn (mut ctx Context) send_custom_response(resp &http.Response) ? {
 | 
				
			||||||
 | 
						ctx.send_string(resp.bytestr())?
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// send_response_header constructs a valid HTTP response with an empty body &
 | 
				
			||||||
 | 
					// sends it to the client.
 | 
				
			||||||
 | 
					pub fn (mut ctx Context) send_response_header() ? {
 | 
				
			||||||
 | 
						mut resp := http.new_response(
 | 
				
			||||||
 | 
							header: ctx.header.join(headers_close)
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						resp.header.add(.content_type, ctx.content_type)
 | 
				
			||||||
	resp.set_status(ctx.status)
 | 
						resp.set_status(ctx.status)
 | 
				
			||||||
	send_string(mut ctx.conn, resp.bytestr()) or { return false }
 | 
					
 | 
				
			||||||
 | 
						ctx.send_custom_response(resp)?
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// send is a convenience function for sending the HTTP response with an empty
 | 
				
			||||||
 | 
					// body.
 | 
				
			||||||
 | 
					pub fn (mut ctx Context) send() bool {
 | 
				
			||||||
 | 
						return ctx.send_response('')
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// send_response constructs the resulting HTTP response with the given body
 | 
				
			||||||
 | 
					// string & sends it to the client.
 | 
				
			||||||
 | 
					pub fn (mut ctx Context) send_response(res string) bool {
 | 
				
			||||||
 | 
						ctx.send_response_header() or { return false }
 | 
				
			||||||
 | 
						ctx.send_string(res) or { return false }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return true
 | 
						return true
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// text responds to a request with some plaintext.
 | 
					// send_reader_response constructs the resulting HTTP response with the given
 | 
				
			||||||
pub fn (mut ctx Context) text(status http.Status, s string) Result {
 | 
					// body & streams the reader's contents to the client.
 | 
				
			||||||
	ctx.status = status
 | 
					pub fn (mut ctx Context) send_reader_response(mut reader io.Reader, size u64) bool {
 | 
				
			||||||
 | 
						ctx.send_response_header() or { return false }
 | 
				
			||||||
 | 
						ctx.send_reader(mut reader, size) or { return false }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ctx.send_response_to_client('text/plain', s)
 | 
						return true
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return Result{}
 | 
					// is_authenticated checks whether the request passes a correct API key.
 | 
				
			||||||
 | 
					pub fn (ctx &Context) is_authenticated() bool {
 | 
				
			||||||
 | 
						if provided_key := ctx.req.header.get_custom('X-Api-Key') {
 | 
				
			||||||
 | 
							return provided_key == ctx.api_key
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// json<T> HTTP_OK with json_s as payload with content-type `application/json`
 | 
					// json<T> HTTP_OK with json_s as payload with content-type `application/json`
 | 
				
			||||||
pub fn (mut ctx Context) json<T>(status http.Status, j T) Result {
 | 
					pub fn (mut ctx Context) json<T>(status http.Status, j T) Result {
 | 
				
			||||||
	ctx.status = status
 | 
						ctx.status = status
 | 
				
			||||||
 | 
						ctx.content_type = 'application/json'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	json_s := json.encode(j)
 | 
						json_s := json.encode(j)
 | 
				
			||||||
	ctx.send_response_to_client('application/json', json_s)
 | 
						ctx.send_response(json_s)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return Result{}
 | 
						return Result{}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -239,119 +159,112 @@ pub fn (mut ctx Context) json<T>(status http.Status, j T) Result {
 | 
				
			||||||
// file Response HTTP_OK with file as payload
 | 
					// file Response HTTP_OK with file as payload
 | 
				
			||||||
// This function manually implements responses because it needs to stream the file contents
 | 
					// This function manually implements responses because it needs to stream the file contents
 | 
				
			||||||
pub fn (mut ctx Context) file(f_path string) Result {
 | 
					pub fn (mut ctx Context) file(f_path string) Result {
 | 
				
			||||||
	if ctx.done {
 | 
						// If the file doesn't exist, just respond with a 404
 | 
				
			||||||
 | 
						if !os.is_file(f_path) {
 | 
				
			||||||
 | 
							ctx.status = .not_found
 | 
				
			||||||
 | 
							ctx.send()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		return Result{}
 | 
							return Result{}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if !os.is_file(f_path) {
 | 
						ctx.header.add(.accept_ranges, 'bytes')
 | 
				
			||||||
		return ctx.not_found()
 | 
					
 | 
				
			||||||
 | 
						file_size := os.file_size(f_path)
 | 
				
			||||||
 | 
						ctx.header.add(http.CommonHeader.content_length, file_size.str())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// A HEAD request only returns the size of the file.
 | 
				
			||||||
 | 
						if ctx.req.method == .head {
 | 
				
			||||||
 | 
							ctx.send()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return Result{}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// ext := os.file_ext(f_path)
 | 
						mut file := os.open(f_path) or {
 | 
				
			||||||
	// data := os.read_file(f_path) or {
 | 
					 | 
				
			||||||
	// 	eprint(err.msg())
 | 
					 | 
				
			||||||
	// 	ctx.server_error(500)
 | 
					 | 
				
			||||||
	// 	return Result{}
 | 
					 | 
				
			||||||
	// }
 | 
					 | 
				
			||||||
	// content_type := web.mime_types[ext]
 | 
					 | 
				
			||||||
	// if content_type == '' {
 | 
					 | 
				
			||||||
	// 	eprintln('no MIME type found for extension $ext')
 | 
					 | 
				
			||||||
	// 	ctx.server_error(500)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// 	return Result{}
 | 
					 | 
				
			||||||
	// }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// First, we return the headers for the request
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// We open the file before sending the headers in case reading fails
 | 
					 | 
				
			||||||
	file_size := os.file_size(f_path)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	file := os.open(f_path) or {
 | 
					 | 
				
			||||||
		eprintln(err.msg())
 | 
							eprintln(err.msg())
 | 
				
			||||||
		ctx.server_error(500)
 | 
							ctx.server_error(500)
 | 
				
			||||||
		return Result{}
 | 
							return Result{}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// build header
 | 
						defer {
 | 
				
			||||||
	header := http.new_header_from_map({
 | 
							file.close()
 | 
				
			||||||
		// http.CommonHeader.content_type:   content_type
 | 
					 | 
				
			||||||
		http.CommonHeader.content_length: file_size.str()
 | 
					 | 
				
			||||||
	}).join(ctx.header)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	mut resp := http.Response{
 | 
					 | 
				
			||||||
		header: header.join(web.headers_close)
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	resp.set_version(.v1_1)
 | 
					 | 
				
			||||||
	resp.set_status(ctx.status)
 | 
					 | 
				
			||||||
	send_string(mut ctx.conn, resp.bytestr()) or { return Result{} }
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	mut buf := []u8{len: 1_000_000}
 | 
						// Currently, this only supports a single provided range, e.g.
 | 
				
			||||||
	mut bytes_left := file_size
 | 
						// bytes=0-1023, and not multiple ranges, e.g. bytes=0-50, 100-150
 | 
				
			||||||
 | 
						if range_str := ctx.req.header.get(.range) {
 | 
				
			||||||
 | 
							mut parts := range_str.split_nth('=', 2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Repeat as long as the stream still has data
 | 
							// We only support the 'bytes' range type
 | 
				
			||||||
	for bytes_left > 0 {
 | 
							if parts[0] != 'bytes' {
 | 
				
			||||||
		// TODO check if just breaking here is safe
 | 
								ctx.status = .requested_range_not_satisfiable
 | 
				
			||||||
		bytes_read := file.read(mut buf) or { break }
 | 
								ctx.header.delete(.content_length)
 | 
				
			||||||
		bytes_left -= u64(bytes_read)
 | 
								ctx.send()
 | 
				
			||||||
 | 
								return Result{}
 | 
				
			||||||
		mut to_write := bytes_read
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		for to_write > 0 {
 | 
					 | 
				
			||||||
			// TODO don't just loop infinitely here
 | 
					 | 
				
			||||||
			bytes_written := ctx.conn.write(buf[bytes_read - to_write..bytes_read]) or { continue }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			to_write = to_write - bytes_written
 | 
					 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							parts = parts[1].split_nth('-', 2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							start := parts[0].i64()
 | 
				
			||||||
 | 
							end := if parts[1] == '' { file_size - 1 } else { parts[1].u64() }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Either the actual number 0 or the result of an invalid integer
 | 
				
			||||||
 | 
							if end == 0 {
 | 
				
			||||||
 | 
								ctx.status = .requested_range_not_satisfiable
 | 
				
			||||||
 | 
								ctx.header.delete(.content_length)
 | 
				
			||||||
 | 
								ctx.send()
 | 
				
			||||||
 | 
								return Result{}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Move cursor to start of data to read
 | 
				
			||||||
 | 
							file.seek(start, .start) or {
 | 
				
			||||||
 | 
								ctx.server_error(500)
 | 
				
			||||||
 | 
								return Result{}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							length := end - u64(start) + 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ctx.status = .partial_content
 | 
				
			||||||
 | 
							ctx.header.set(.content_length, length.str())
 | 
				
			||||||
 | 
							ctx.send_reader_response(mut file, length)
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							ctx.send_reader_response(mut file, file_size)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ctx.done = true
 | 
					 | 
				
			||||||
	return Result{}
 | 
						return Result{}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// status responds with an empty textual response, essentially only returning
 | 
					// status responds with an empty textual response, essentially only returning
 | 
				
			||||||
// the given status code.
 | 
					// the given status code.
 | 
				
			||||||
pub fn (mut ctx Context) status(status http.Status) Result {
 | 
					pub fn (mut ctx Context) status(status http.Status) Result {
 | 
				
			||||||
	return ctx.text(status, '')
 | 
						ctx.status = status
 | 
				
			||||||
 | 
						ctx.send()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return Result{}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// server_error Response a server error
 | 
					// server_error Response a server error
 | 
				
			||||||
pub fn (mut ctx Context) server_error(ecode int) Result {
 | 
					pub fn (mut ctx Context) server_error(ecode int) Result {
 | 
				
			||||||
	$if debug {
 | 
						ctx.send_custom_response(http_500) or {}
 | 
				
			||||||
		eprintln('> ctx.server_error ecode: $ecode')
 | 
					
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	if ctx.done {
 | 
					 | 
				
			||||||
		return Result{}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	send_string(mut ctx.conn, web.http_500.bytestr()) or {}
 | 
					 | 
				
			||||||
	return Result{}
 | 
						return Result{}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// redirect Redirect to an url
 | 
					// redirect Redirect to an url
 | 
				
			||||||
pub fn (mut ctx Context) redirect(url string) Result {
 | 
					pub fn (mut ctx Context) redirect(url string) Result {
 | 
				
			||||||
	if ctx.done {
 | 
						mut resp := http_302
 | 
				
			||||||
		return Result{}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	ctx.done = true
 | 
					 | 
				
			||||||
	mut resp := web.http_302
 | 
					 | 
				
			||||||
	resp.header = resp.header.join(ctx.header)
 | 
						resp.header = resp.header.join(ctx.header)
 | 
				
			||||||
	resp.header.add(.location, url)
 | 
						resp.header.add(.location, url)
 | 
				
			||||||
	send_string(mut ctx.conn, resp.bytestr()) or { return Result{} }
 | 
					
 | 
				
			||||||
 | 
						ctx.send_custom_response(resp) or {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return Result{}
 | 
						return Result{}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// not_found Send an not_found response
 | 
					// not_found Send an not_found response
 | 
				
			||||||
pub fn (mut ctx Context) not_found() Result {
 | 
					pub fn (mut ctx Context) not_found() Result {
 | 
				
			||||||
	return ctx.status(http.Status.not_found)
 | 
						ctx.send_custom_response(http_404) or {}
 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
// add_header Adds an header to the response with key and val
 | 
						return Result{}
 | 
				
			||||||
pub fn (mut ctx Context) add_header(key string, val string) {
 | 
					 | 
				
			||||||
	ctx.header.add_custom(key, val) or {}
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// get_header Returns the header data from the key
 | 
					 | 
				
			||||||
pub fn (ctx &Context) get_header(key string) string {
 | 
					 | 
				
			||||||
	return ctx.req.header.get_custom(key) or { '' }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
interface DbInterface {
 | 
					interface DbInterface {
 | 
				
			||||||
| 
						 | 
					@ -478,6 +391,7 @@ fn handle_conn<T>(mut conn net.TcpConn, mut app T, routes map[string]Route) {
 | 
				
			||||||
		static_mime_types: app.static_mime_types
 | 
							static_mime_types: app.static_mime_types
 | 
				
			||||||
		reader: reader
 | 
							reader: reader
 | 
				
			||||||
		logger: app.logger
 | 
							logger: app.logger
 | 
				
			||||||
 | 
							api_key: app.api_key
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Calling middleware...
 | 
						// Calling middleware...
 | 
				
			||||||
| 
						 | 
					@ -496,31 +410,27 @@ fn handle_conn<T>(mut conn net.TcpConn, mut app T, routes map[string]Route) {
 | 
				
			||||||
				// Used for route matching
 | 
									// Used for route matching
 | 
				
			||||||
				route_words := route.path.split('/').filter(it != '')
 | 
									route_words := route.path.split('/').filter(it != '')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
				// Route immediate matches first
 | 
									// Route immediate matches & index files first
 | 
				
			||||||
				// For example URL `/register` matches route `/:user`, but `fn register()`
 | 
									// For example URL `/register` matches route `/:user`, but `fn register()`
 | 
				
			||||||
				// should be called first.
 | 
									// should be called first.
 | 
				
			||||||
				if !route.path.contains('/:') && url_words == route_words {
 | 
									if (!route.path.contains('/:') && url_words == route_words)
 | 
				
			||||||
					// We found a match
 | 
										|| (url_words.len == 0 && route_words == ['index'] && method.name == 'index') {
 | 
				
			||||||
					if head.method == .post && method.args.len > 0 {
 | 
										// Check whether the request is authorised
 | 
				
			||||||
						// TODO implement POST requests
 | 
										if 'auth' in method.attrs && !app.is_authenticated() {
 | 
				
			||||||
						// Populate method args with form values
 | 
											conn.write(http_401.bytes()) or {}
 | 
				
			||||||
						// mut args := []string{cap: method.args.len}
 | 
											return
 | 
				
			||||||
						// for param in method.args {
 | 
					 | 
				
			||||||
						// 	args << form[param.name]
 | 
					 | 
				
			||||||
						// }
 | 
					 | 
				
			||||||
						// app.$method(args)
 | 
					 | 
				
			||||||
					} else {
 | 
					 | 
				
			||||||
						app.$method()
 | 
					 | 
				
			||||||
					}
 | 
										}
 | 
				
			||||||
					return
 | 
					 | 
				
			||||||
				}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
				if url_words.len == 0 && route_words == ['index'] && method.name == 'index' {
 | 
										// We found a match
 | 
				
			||||||
					app.$method()
 | 
										app.$method()
 | 
				
			||||||
					return
 | 
										return
 | 
				
			||||||
				}
 | 
									} else if params := route_matches(url_words, route_words) {
 | 
				
			||||||
 | 
										// Check whether the request is authorised
 | 
				
			||||||
 | 
										if 'auth' in method.attrs && !app.is_authenticated() {
 | 
				
			||||||
 | 
											conn.write(http_401.bytes()) or {}
 | 
				
			||||||
 | 
											return
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
				if params := route_matches(url_words, route_words) {
 | 
					 | 
				
			||||||
					method_args := params.clone()
 | 
										method_args := params.clone()
 | 
				
			||||||
					if method_args.len != method.args.len {
 | 
										if method_args.len != method.args.len {
 | 
				
			||||||
						eprintln('warning: uneven parameters count ($method.args.len) in `$method.name`, compared to the web route `$method.attrs` ($method_args.len)')
 | 
											eprintln('warning: uneven parameters count ($method.args.len) in `$method.name`, compared to the web route `$method.attrs` ($method_args.len)')
 | 
				
			||||||
| 
						 | 
					@ -532,7 +442,7 @@ fn handle_conn<T>(mut conn net.TcpConn, mut app T, routes map[string]Route) {
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	// Route not found
 | 
						// Route not found
 | 
				
			||||||
	conn.write(web.http_404.bytes()) or {}
 | 
						conn.write(http_404.bytes()) or {}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// route_matches returns wether a route matches
 | 
					// route_matches returns wether a route matches
 | 
				
			||||||
| 
						 | 
					@ -578,28 +488,6 @@ fn route_matches(url_words []string, route_words []string) ?[]string {
 | 
				
			||||||
	return params
 | 
						return params
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ip Returns the ip address from the current user
 | 
					 | 
				
			||||||
pub fn (ctx &Context) ip() string {
 | 
					 | 
				
			||||||
	mut ip := ctx.req.header.get(.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
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// error Set s to the form error
 | 
					 | 
				
			||||||
pub fn (mut ctx Context) error(s string) {
 | 
					 | 
				
			||||||
	println('web error: $s')
 | 
					 | 
				
			||||||
	ctx.form_error = s
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// filter Do not delete.
 | 
					// filter Do not delete.
 | 
				
			||||||
// It used by `vlib/v/gen/c/str_intp.v:130` for string interpolation inside web templates
 | 
					// It used by `vlib/v/gen/c/str_intp.v:130` for string interpolation inside web templates
 | 
				
			||||||
// TODO: move it to template render
 | 
					// TODO: move it to template render
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue