mirror of
				https://github.com/Ysurac/openmptcprouter-feeds.git
				synced 2025-03-09 15:40:03 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			268 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| -- Copyright 2008 Steven Barth <steven@midlink.org>
 | |
| -- Licensed to the public under the Apache License 2.0.
 | |
| 
 | |
| local ltn12 = require "luci.ltn12"
 | |
| local protocol = require "luci.http.protocol"
 | |
| local util  = require "luci.util"
 | |
| local string = require "string"
 | |
| local coroutine = require "coroutine"
 | |
| local table = require "table"
 | |
| 
 | |
| local ipairs, pairs, next, type, tostring, error =
 | |
| 	ipairs, pairs, next, type, tostring, error
 | |
| 
 | |
| module "luci.http"
 | |
| 
 | |
| context = util.threadlocal()
 | |
| 
 | |
| Request = util.class()
 | |
| function Request.__init__(self, env, sourcein, sinkerr)
 | |
| 	self.input = sourcein
 | |
| 	self.error = sinkerr
 | |
| 
 | |
| 
 | |
| 	-- File handler nil by default to let .content() work
 | |
| 	self.filehandler = nil
 | |
| 
 | |
| 	-- HTTP-Message table
 | |
| 	self.message = {
 | |
| 		env = env,
 | |
| 		headers = {},
 | |
| 		params = protocol.urldecode_params(env.QUERY_STRING or ""),
 | |
| 	}
 | |
| 
 | |
| 	self.parsed_input = false
 | |
| end
 | |
| 
 | |
| function Request.formvalue(self, name, noparse)
 | |
| 	if not noparse and not self.parsed_input then
 | |
| 		self:_parse_input()
 | |
| 	end
 | |
| 
 | |
| 	if name then
 | |
| 		return self.message.params[name]
 | |
| 	else
 | |
| 		return self.message.params
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function Request.formvaluetable(self, prefix)
 | |
| 	local vals = {}
 | |
| 	prefix = prefix and prefix .. "." or "."
 | |
| 
 | |
| 	if not self.parsed_input then
 | |
| 		self:_parse_input()
 | |
| 	end
 | |
| 
 | |
| 	local void = self.message.params[nil]
 | |
| 	for k, v in pairs(self.message.params) do
 | |
| 		if k:find(prefix, 1, true) == 1 then
 | |
| 			vals[k:sub(#prefix + 1)] = tostring(v)
 | |
| 		end
 | |
| 	end
 | |
| 
 | |
| 	return vals
 | |
| end
 | |
| 
 | |
| function Request.content(self)
 | |
| 	if not self.parsed_input then
 | |
| 		self:_parse_input()
 | |
| 	end
 | |
| 
 | |
| 	return self.message.content, self.message.content_length
 | |
| end
 | |
| 
 | |
| function Request.getcookie(self, name)
 | |
|   local c = string.gsub(";" .. (self:getenv("HTTP_COOKIE") or "") .. ";", "%s*;%s*", ";")
 | |
|   local p = ";" .. name .. "=(.-);"
 | |
|   local i, j, value = c:find(p)
 | |
|   return value and urldecode(value)
 | |
| end
 | |
| 
 | |
| function Request.getenv(self, name)
 | |
| 	if name then
 | |
| 		return self.message.env[name]
 | |
| 	else
 | |
| 		return self.message.env
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function Request.setfilehandler(self, callback)
 | |
| 	self.filehandler = callback
 | |
| 
 | |
| 	-- If input has already been parsed then any files are either in temporary files
 | |
| 	-- or are in self.message.params[key]
 | |
| 	if self.parsed_input then
 | |
| 		for param, value in pairs(self.message.params) do
 | |
| 		repeat
 | |
| 			-- We're only interested in files
 | |
| 			if (not value["file"]) then break end
 | |
| 			-- If we were able to write to temporary file
 | |
| 			if (value["fd"]) then 
 | |
| 				fd = value["fd"]
 | |
| 				local eof = false
 | |
| 				repeat	
 | |
| 					filedata = fd:read(1024)
 | |
| 					if (filedata:len() < 1024) then
 | |
| 						eof = true
 | |
| 					end
 | |
| 					callback({ name=value["name"], file=value["file"] }, filedata, eof)
 | |
| 				until (eof)
 | |
| 				fd:close()
 | |
| 				value["fd"] = nil
 | |
| 			-- We had to read into memory
 | |
| 			else
 | |
| 				-- There should only be one numbered value in table - the data
 | |
| 				for k, v in ipairs(value) do
 | |
| 					callback({ name=value["name"], file=value["file"] }, v, true)
 | |
| 				end
 | |
| 			end
 | |
| 		until true
 | |
| 		end
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function Request._parse_input(self)
 | |
| 	protocol.parse_message_body(
 | |
| 		 self.input,
 | |
| 		 self.message,
 | |
| 		 self.filehandler
 | |
| 	)
 | |
| 	self.parsed_input = true
 | |
| end
 | |
| 
 | |
| function close()
 | |
| 	if not context.eoh then
 | |
| 		context.eoh = true
 | |
| 		coroutine.yield(3)
 | |
| 	end
 | |
| 
 | |
| 	if not context.closed then
 | |
| 		context.closed = true
 | |
| 		coroutine.yield(5)
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function content()
 | |
| 	return context.request:content()
 | |
| end
 | |
| 
 | |
| function formvalue(name, noparse)
 | |
| 	return context.request:formvalue(name, noparse)
 | |
| end
 | |
| 
 | |
| function formvaluetable(prefix)
 | |
| 	return context.request:formvaluetable(prefix)
 | |
| end
 | |
| 
 | |
| function getcookie(name)
 | |
| 	return context.request:getcookie(name)
 | |
| end
 | |
| 
 | |
| -- or the environment table itself.
 | |
| function getenv(name)
 | |
| 	return context.request:getenv(name)
 | |
| end
 | |
| 
 | |
| function setfilehandler(callback)
 | |
| 	return context.request:setfilehandler(callback)
 | |
| end
 | |
| 
 | |
| function header(key, value)
 | |
| 	if not context.headers then
 | |
| 		context.headers = {}
 | |
| 	end
 | |
| 	context.headers[key:lower()] = value
 | |
| 	coroutine.yield(2, key, value)
 | |
| end
 | |
| 
 | |
| function prepare_content(mime)
 | |
| 	if not context.headers or not context.headers["content-type"] then
 | |
| 		if mime == "application/xhtml+xml" then
 | |
| 			if not getenv("HTTP_ACCEPT") or
 | |
| 			  not getenv("HTTP_ACCEPT"):find("application/xhtml+xml", nil, true) then
 | |
| 				mime = "text/html; charset=UTF-8"
 | |
| 			end
 | |
| 			header("Vary", "Accept")
 | |
| 		end
 | |
| 		header("Content-Type", mime)
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function source()
 | |
| 	return context.request.input
 | |
| end
 | |
| 
 | |
| function status(code, message)
 | |
| 	code = code or 200
 | |
| 	message = message or "OK"
 | |
| 	context.status = code
 | |
| 	coroutine.yield(1, code, message)
 | |
| end
 | |
| 
 | |
| -- This function is as a valid LTN12 sink.
 | |
| -- If the content chunk is nil this function will automatically invoke close.
 | |
| function write(content, src_err)
 | |
| 	if not content then
 | |
| 		if src_err then
 | |
| 			error(src_err)
 | |
| 		else
 | |
| 			close()
 | |
| 		end
 | |
| 		return true
 | |
| 	elseif #content == 0 then
 | |
| 		return true
 | |
| 	else
 | |
| 		if not context.eoh then
 | |
| 			if not context.status then
 | |
| 				status()
 | |
| 			end
 | |
| 			if not context.headers or not context.headers["content-type"] then
 | |
| 				header("Content-Type", "text/html; charset=utf-8")
 | |
| 			end
 | |
| 			if not context.headers["cache-control"] then
 | |
| 				header("Cache-Control", "no-cache")
 | |
| 				header("Expires", "0")
 | |
| 			end
 | |
| 
 | |
| 
 | |
| 			context.eoh = true
 | |
| 			coroutine.yield(3)
 | |
| 		end
 | |
| 		coroutine.yield(4, content)
 | |
| 		return true
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function splice(fd, size)
 | |
| 	coroutine.yield(6, fd, size)
 | |
| end
 | |
| 
 | |
| function redirect(url)
 | |
| 	if url == "" then url = "/" end
 | |
| 	status(302, "Found")
 | |
| 	header("Location", url)
 | |
| 	close()
 | |
| end
 | |
| 
 | |
| function build_querystring(q)
 | |
| 	local s = { "?" }
 | |
| 
 | |
| 	for k, v in pairs(q) do
 | |
| 		if #s > 1 then s[#s+1] = "&" end
 | |
| 
 | |
| 		s[#s+1] = urldecode(k)
 | |
| 		s[#s+1] = "="
 | |
| 		s[#s+1] = urldecode(v)
 | |
| 	end
 | |
| 
 | |
| 	return table.concat(s, "")
 | |
| end
 | |
| 
 | |
| urldecode = protocol.urldecode
 | |
| 
 | |
| urlencode = protocol.urlencode
 | |
| 
 | |
| function write_json(x)
 | |
| 	util.serialize_json(x, write)
 | |
| end
 |