Temporary in case i fuck up
This commit is contained in:
commit
ac022435a4
5
README.md
Normal file
5
README.md
Normal file
@ -0,0 +1,5 @@
|
||||
# Oasis
|
||||
|
||||
A local development tool for the laravel framework. Inspired by Valet and Herd.
|
||||
|
||||
I want to make a tool that also worked on linux, and has some extra features I want.
|
161
fastcgi/client.go
Normal file
161
fastcgi/client.go
Normal file
@ -0,0 +1,161 @@
|
||||
package fastcgi
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"bytes"
|
||||
"encoding/binary"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/http/httputil"
|
||||
"net/textproto"
|
||||
"strconv"
|
||||
"strings"
|
||||
"sync"
|
||||
)
|
||||
|
||||
type FCGIClient struct {
|
||||
mutex sync.Mutex
|
||||
rwc io.ReadWriteCloser
|
||||
h Header
|
||||
buf bytes.Buffer
|
||||
keepAlive bool
|
||||
reqId uint16
|
||||
}
|
||||
|
||||
// Close fcgi connnection
|
||||
func (client *FCGIClient) Close() {
|
||||
client.rwc.Close()
|
||||
}
|
||||
|
||||
func (client *FCGIClient) writeRecord(recType FCGIRequestType, content []byte) (err error) {
|
||||
client.mutex.Lock()
|
||||
defer client.mutex.Unlock()
|
||||
client.buf.Reset()
|
||||
// Initialize the record
|
||||
header := Header{}
|
||||
header.init(recType, 1, len(content))
|
||||
rec := Record{
|
||||
Header: header,
|
||||
Content: content,
|
||||
}
|
||||
|
||||
// Write the record to the connection
|
||||
b, err := rec.toBytes()
|
||||
_, err = client.rwc.Write(b)
|
||||
return err
|
||||
}
|
||||
|
||||
func (client *FCGIClient) writeEndRequest(appStatus int, protocolStatus uint8) error {
|
||||
b := make([]byte, 8)
|
||||
binary.BigEndian.PutUint32(b, uint32(appStatus))
|
||||
b[4] = protocolStatus
|
||||
return client.writeRecord(FCGI_END_REQUEST, b)
|
||||
}
|
||||
|
||||
func (client *FCGIClient) writePairs(recType FCGIRequestType, pairs map[string]string) error {
|
||||
w := newWriter(client, recType)
|
||||
b := make([]byte, 8)
|
||||
nn := 0
|
||||
for k, v := range pairs {
|
||||
m := 8 + len(k) + len(v)
|
||||
if m > maxWrite {
|
||||
// param data size exceed 65535 bytes"
|
||||
vl := maxWrite - 8 - len(k)
|
||||
v = v[:vl]
|
||||
}
|
||||
n := encodeSize(b, uint32(len(k)))
|
||||
n += encodeSize(b[n:], uint32(len(v)))
|
||||
m = n + len(k) + len(v)
|
||||
if (nn + m) > maxWrite {
|
||||
w.Flush()
|
||||
nn = 0
|
||||
}
|
||||
nn += m
|
||||
if _, err := w.Write(b[:n]); err != nil {
|
||||
return err
|
||||
}
|
||||
if _, err := w.WriteString(k); err != nil {
|
||||
return err
|
||||
}
|
||||
if _, err := w.WriteString(v); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
w.Close()
|
||||
return nil
|
||||
}
|
||||
|
||||
// Do made the request and returns a io.Reader that translates the data read
|
||||
// from fcgi responder out of fcgi packet before returning it.
|
||||
func (client *FCGIClient) Do(req *FCGIRequest) (http.Response, error) {
|
||||
beginRequestRecord := NewBeginRequestRecord()
|
||||
err := client.writeRecord(beginRequestRecord.Header.Type, beginRequestRecord.Content)
|
||||
if err != nil {
|
||||
return http.Response{}, err
|
||||
}
|
||||
|
||||
err = client.writePairs(FCGI_PARAMS, req.Context)
|
||||
if err != nil {
|
||||
return http.Response{}, err
|
||||
}
|
||||
|
||||
// body := newWriter(client, FCGI_STDIN)
|
||||
// if req != nil {
|
||||
// io.Copy(body, req)
|
||||
// }
|
||||
// body.Close()
|
||||
|
||||
r := &streamReader{c: client}
|
||||
rb := bufio.NewReader(r)
|
||||
tp := textproto.NewReader(rb)
|
||||
resp := new(http.Response)
|
||||
// Parse the first line of the response.
|
||||
line, err := tp.ReadLine()
|
||||
if err != nil {
|
||||
if err == io.EOF {
|
||||
err = io.ErrUnexpectedEOF
|
||||
}
|
||||
return http.Response{}, err
|
||||
}
|
||||
if i := strings.IndexByte(line, ' '); i == -1 {
|
||||
err = &badStringError{"malformed HTTP response", line}
|
||||
} else {
|
||||
resp.Proto = line[:i]
|
||||
resp.Status = strings.TrimLeft(line[i+1:], " ")
|
||||
}
|
||||
statusCode := resp.Status
|
||||
if i := strings.IndexByte(resp.Status, ' '); i != -1 {
|
||||
statusCode = resp.Status[:i]
|
||||
}
|
||||
if len(statusCode) != 3 {
|
||||
err = &badStringError{"malformed HTTP status code", statusCode}
|
||||
}
|
||||
resp.StatusCode, err = strconv.Atoi(statusCode)
|
||||
if err != nil || resp.StatusCode < 0 {
|
||||
err = &badStringError{"malformed HTTP status code", statusCode}
|
||||
}
|
||||
var ok bool
|
||||
if resp.ProtoMajor, resp.ProtoMinor, ok = http.ParseHTTPVersion(resp.Proto); !ok {
|
||||
err = &badStringError{"malformed HTTP version", resp.Proto}
|
||||
}
|
||||
// Parse the response headers.
|
||||
mimeHeader, err := tp.ReadMIMEHeader()
|
||||
if err != nil {
|
||||
if err == io.EOF {
|
||||
err = io.ErrUnexpectedEOF
|
||||
}
|
||||
return http.Response{}, err
|
||||
}
|
||||
resp.Header = http.Header(mimeHeader)
|
||||
// TODO: fixTransferEncoding ?
|
||||
resp.TransferEncoding = resp.Header["Transfer-Encoding"]
|
||||
resp.ContentLength, _ = strconv.ParseInt(resp.Header.Get("Content-Length"), 10, 64)
|
||||
|
||||
if chunked(resp.TransferEncoding) {
|
||||
resp.Body = io.NopCloser(httputil.NewChunkedReader(rb))
|
||||
} else {
|
||||
resp.Body = io.NopCloser(rb)
|
||||
}
|
||||
|
||||
return *resp, nil
|
||||
}
|
140
fastcgi/fastcgi.go
Normal file
140
fastcgi/fastcgi.go
Normal file
@ -0,0 +1,140 @@
|
||||
package fastcgi
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"encoding/binary"
|
||||
"fmt"
|
||||
"io"
|
||||
"net"
|
||||
)
|
||||
|
||||
type FCGIRequestType uint8
|
||||
|
||||
const FCGI_LISTENSOCK_FILENO uint8 = 0
|
||||
const FCGI_HEADER_LEN uint8 = 8
|
||||
const VERSION_1 uint8 = 1
|
||||
const FCGI_NULL_REQUEST_ID uint8 = 0
|
||||
const FCGI_KEEP_CONN uint8 = 1
|
||||
const doubleCRLF = "\r\n\r\n"
|
||||
|
||||
const (
|
||||
FCGI_BEGIN_REQUEST FCGIRequestType = iota + 1
|
||||
FCGI_ABORT_REQUEST
|
||||
FCGI_END_REQUEST
|
||||
FCGI_PARAMS
|
||||
FCGI_STDIN
|
||||
FCGI_STDOUT
|
||||
FCGI_STDERR
|
||||
FCGI_DATA
|
||||
FCGI_GET_VALUES
|
||||
FCGI_GET_VALUES_RESULT
|
||||
FCGI_UNKNOWN_TYPE
|
||||
FCGI_MAXTYPE = FCGI_UNKNOWN_TYPE
|
||||
)
|
||||
|
||||
const (
|
||||
FCGI_RESPONDER uint8 = iota + 1
|
||||
FCGI_AUTHORIZER
|
||||
FCGI_FILTER
|
||||
)
|
||||
|
||||
const (
|
||||
FCGI_REQUEST_COMPLETE uint8 = iota
|
||||
FCGI_CANT_MPX_CONN
|
||||
FCGI_OVERLOADED
|
||||
FCGI_UNKNOWN_ROLE
|
||||
)
|
||||
|
||||
const (
|
||||
FCGI_MAX_CONNS string = "MAX_CONNS"
|
||||
FCGI_MAX_REQS string = "MAX_REQS"
|
||||
FCGI_MPXS_CONNS string = "MPXS_CONNS"
|
||||
)
|
||||
|
||||
const (
|
||||
maxWrite = 65500 // 65530 may work, but for compatibility
|
||||
maxPad = 255
|
||||
)
|
||||
|
||||
// Connects to the fcgi responder at the specified network address.
|
||||
// See func net.Dial for a description of the network and address parameters.
|
||||
func Dial(network, address string) (fcgi *FCGIClient, err error) {
|
||||
var conn net.Conn
|
||||
|
||||
conn, err = net.Dial(network, address)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
fcgi = &FCGIClient{
|
||||
rwc: conn,
|
||||
keepAlive: false,
|
||||
reqId: 1,
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func readSize(s []byte) (uint32, int) {
|
||||
if len(s) == 0 {
|
||||
return 0, 0
|
||||
}
|
||||
size, n := uint32(s[0]), 1
|
||||
if size&(1<<7) != 0 {
|
||||
if len(s) < 4 {
|
||||
return 0, 0
|
||||
}
|
||||
n = 4
|
||||
size = binary.BigEndian.Uint32(s)
|
||||
size &^= 1 << 31
|
||||
}
|
||||
return size, n
|
||||
}
|
||||
|
||||
func readString(s []byte, size uint32) string {
|
||||
if size > uint32(len(s)) {
|
||||
return ""
|
||||
}
|
||||
return string(s[:size])
|
||||
}
|
||||
|
||||
func encodeSize(b []byte, size uint32) int {
|
||||
if size > 127 {
|
||||
size |= 1 << 31
|
||||
binary.BigEndian.PutUint32(b, size)
|
||||
return 4
|
||||
}
|
||||
b[0] = byte(size)
|
||||
return 1
|
||||
}
|
||||
|
||||
// bufWriter encapsulates bufio.Writer but also closes the underlying stream when
|
||||
// Closed.
|
||||
type bufWriter struct {
|
||||
closer io.Closer
|
||||
*bufio.Writer
|
||||
}
|
||||
|
||||
func (w *bufWriter) Close() error {
|
||||
if err := w.Writer.Flush(); err != nil {
|
||||
w.closer.Close()
|
||||
return err
|
||||
}
|
||||
return w.closer.Close()
|
||||
}
|
||||
|
||||
func newWriter(c *FCGIClient, recType FCGIRequestType) *bufWriter {
|
||||
s := &streamWriter{c: c, recType: recType}
|
||||
w := bufio.NewWriterSize(s, maxWrite)
|
||||
return &bufWriter{s, w}
|
||||
}
|
||||
|
||||
type badStringError struct {
|
||||
what string
|
||||
str string
|
||||
}
|
||||
|
||||
func (e *badStringError) Error() string { return fmt.Sprintf("%s %q", e.what, e.str) }
|
||||
|
||||
// Checks whether chunked is part of the encodings stack
|
||||
func chunked(te []string) bool { return len(te) > 0 && te[0] == "chunked" }
|
244
fastcgi/request.go
Normal file
244
fastcgi/request.go
Normal file
@ -0,0 +1,244 @@
|
||||
package fastcgi
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"strconv"
|
||||
)
|
||||
|
||||
// -- TYPES --
|
||||
// These are the types needed to house the data for an FCGI request
|
||||
type Header struct {
|
||||
Version uint8
|
||||
Type FCGIRequestType
|
||||
Id uint16
|
||||
ContentLength uint16
|
||||
PaddingLength uint8
|
||||
Reserved uint8
|
||||
}
|
||||
|
||||
func (h *Header) init(reqType FCGIRequestType, reqId uint16, l int) {
|
||||
h.Version = 1
|
||||
h.Type = reqType
|
||||
h.Id = reqId
|
||||
h.ContentLength = uint16(l)
|
||||
h.PaddingLength = uint8(-l & 7)
|
||||
}
|
||||
|
||||
// A record is essentially a "packet" in FastCGI.
|
||||
// The header lets the server know what type
|
||||
// of data is being sent, and it expects
|
||||
// a certain structure depending on
|
||||
// the type. For example, the
|
||||
// FCGI_BEGIN_REQUEST record should
|
||||
// have a body of 8 bytes with:
|
||||
// - The first byte being the role
|
||||
// - The second byte being also the role
|
||||
// - The third byte being the flags
|
||||
// - The last five bytes are reserved for future use
|
||||
type Record struct {
|
||||
Header Header
|
||||
Content []byte
|
||||
Buffer bytes.Buffer // Buffer to use when writing to the network
|
||||
ReadBuffer []byte // Buffer to use when reading a response
|
||||
}
|
||||
|
||||
// Turn a record into a byte array so it can be
|
||||
// sent over the network socket
|
||||
func (r *Record) toBytes() ([]byte, error) {
|
||||
// client.h.init(recType, client.reqId, len(content))
|
||||
if err := binary.Write(&r.Buffer, binary.BigEndian, r.Header); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if _, err := r.Buffer.Write(r.Content); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if _, err := r.Buffer.Write(pad[:r.Header.PaddingLength]); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return r.Buffer.Bytes(), nil
|
||||
}
|
||||
|
||||
func (rec *Record) read(r io.Reader) (buf []byte, err error) {
|
||||
if err = binary.Read(r, binary.BigEndian, &rec.Header); err != nil {
|
||||
return
|
||||
}
|
||||
if rec.Header.Version != 1 {
|
||||
err = errors.New("fcgi: invalid header version")
|
||||
return
|
||||
}
|
||||
if rec.Header.Type == FCGI_END_REQUEST {
|
||||
err = io.EOF
|
||||
return
|
||||
}
|
||||
n := int(rec.Header.ContentLength) + int(rec.Header.PaddingLength)
|
||||
if len(rec.ReadBuffer) < n {
|
||||
rec.ReadBuffer = make([]byte, n)
|
||||
}
|
||||
if n, err = io.ReadFull(r, rec.ReadBuffer[:n]); err != nil {
|
||||
return
|
||||
}
|
||||
buf = rec.ReadBuffer[:int(rec.Header.ContentLength)]
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
// for padding so we don't have to allocate all the time
|
||||
// not synchronized because we don't care what the contents are
|
||||
var pad [maxPad]byte
|
||||
|
||||
type FCGIRequest struct {
|
||||
Id uint16
|
||||
Type FCGIRequestType
|
||||
Context map[string]string
|
||||
Records []Record
|
||||
}
|
||||
|
||||
func RequestFromHttp(r *http.Request) *FCGIRequest {
|
||||
c := FCGIRequest{}
|
||||
c.Context = make(map[string]string)
|
||||
c.Context["SERVER_SOFTWARE"] = "oasis / fastcgi"
|
||||
c.Context["QUERY_STRING"] = r.URL.RawQuery
|
||||
c.Context["REMOTE_ADDR"] = "127.0.0.1"
|
||||
|
||||
return &c
|
||||
}
|
||||
|
||||
func NewBeginRequestRecord() *Record {
|
||||
role := uint16(FCGI_RESPONDER)
|
||||
flags := byte(0)
|
||||
// Create an 8-byte array as per the FastCGI specification.
|
||||
var b [8]byte
|
||||
|
||||
// Split the 16-bit role into two bytes and assign them.
|
||||
b[0] = byte(role >> 8) // High byte
|
||||
b[1] = byte(role) // Low byte
|
||||
|
||||
// Set the flags.
|
||||
b[2] = flags
|
||||
|
||||
// The reserved bytes (b[3] to b[7]) will remain zero by default.
|
||||
|
||||
// Return a begin request record
|
||||
h := Header{}
|
||||
h.init(FCGI_BEGIN_REQUEST, 1, len(b))
|
||||
return &Record{
|
||||
Header: h,
|
||||
Content: b[:],
|
||||
}
|
||||
}
|
||||
|
||||
func (req *FCGIRequest) Script(path string) {
|
||||
req.Context["SCRIPT_FILENAME"] = path
|
||||
}
|
||||
|
||||
func (req *FCGIRequest) Method(m string) {
|
||||
req.Context["REQUEST_METHOD"] = m
|
||||
}
|
||||
|
||||
// Get issues a GET request to the fcgi responder.
|
||||
func (r *FCGIRequest) TypeGet() {
|
||||
|
||||
r.Context["REQUEST_METHOD"] = "GET"
|
||||
r.Context["CONTENT_LENGTH"] = "0"
|
||||
}
|
||||
|
||||
// Get issues a Post request to the fcgi responder. with request body
|
||||
// in the format that bodyType specified
|
||||
func (r *FCGIRequest) TypePost(bodyType string, body io.Reader, l int) {
|
||||
|
||||
if len(r.Context["REQUEST_METHOD"]) == 0 || r.Context["REQUEST_METHOD"] == "GET" {
|
||||
r.Context["REQUEST_METHOD"] = "POST"
|
||||
}
|
||||
r.Context["CONTENT_LENGTH"] = strconv.Itoa(l)
|
||||
if len(bodyType) > 0 {
|
||||
r.Context["CONTENT_TYPE"] = bodyType
|
||||
} else {
|
||||
r.Context["CONTENT_TYPE"] = "application/x-www-form-urlencoded"
|
||||
}
|
||||
}
|
||||
|
||||
// PostForm issues a POST to the fcgi responder, with form
|
||||
// as a string key to a list values (url.Values)
|
||||
func (r *FCGIRequest) TypePostForm(data url.Values) {
|
||||
body := bytes.NewReader([]byte(data.Encode()))
|
||||
r.TypePost("application/x-www-form-urlencoded", body, body.Len())
|
||||
}
|
||||
|
||||
// PostFile issues a POST to the fcgi responder in multipart(RFC 2046) standard,
|
||||
// with form as a string key to a list values (url.Values),
|
||||
// and/or with file as a string key to a list file path.
|
||||
// func (r *FCGIRequest) PostFile(p map[string]string, data url.Values, file map[string]string) {
|
||||
// buf := &bytes.Buffer{}
|
||||
// writer := multipart.NewWriter(buf)
|
||||
// bodyType := writer.FormDataContentType()
|
||||
//
|
||||
// for key, val := range data {
|
||||
// for _, v0 := range val {
|
||||
// err = writer.WriteField(key, v0)
|
||||
// if err != nil {
|
||||
// return
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// for key, val := range file {
|
||||
// fd, e := os.Open(val)
|
||||
// if e != nil {
|
||||
// return nil, e
|
||||
// }
|
||||
// defer fd.Close()
|
||||
//
|
||||
// part, e := writer.CreateFormFile(key, filepath.Base(val))
|
||||
// if e != nil {
|
||||
// return nil, e
|
||||
// }
|
||||
// _, err = io.Copy(part, fd)
|
||||
// }
|
||||
//
|
||||
// err = writer.Close()
|
||||
// if err != nil {
|
||||
// return
|
||||
// }
|
||||
//
|
||||
// // return client.Post(p, bodyType, buf, buf.Len())
|
||||
// }
|
||||
|
||||
// Format the context for sending
|
||||
// func (r *FCGIRequest) writePairs() error {
|
||||
// w := newWriter(client, recType)
|
||||
// b := make([]byte, 8)
|
||||
// nn := 0
|
||||
// for k, v := range r.Context {
|
||||
// m := 8 + len(k) + len(v)
|
||||
// if m > maxWrite {
|
||||
// // param data size exceed 65535 bytes"
|
||||
// vl := maxWrite - 8 - len(k)
|
||||
// v = v[:vl]
|
||||
// }
|
||||
// n := encodeSize(b, uint32(len(k)))
|
||||
// n += encodeSize(b[n:], uint32(len(v)))
|
||||
// m = n + len(k) + len(v)
|
||||
// if (nn + m) > maxWrite {
|
||||
// w.Flush()
|
||||
// nn = 0
|
||||
// }
|
||||
// nn += m
|
||||
// if _, err := w.Write(b[:n]); err != nil {
|
||||
// return err
|
||||
// }
|
||||
// if _, err := w.WriteString(k); err != nil {
|
||||
// return err
|
||||
// }
|
||||
// if _, err := w.WriteString(v); err != nil {
|
||||
// return err
|
||||
// }
|
||||
// }
|
||||
// w.Close()
|
||||
// return nil
|
||||
// }
|
28
fastcgi/streamreader.go
Normal file
28
fastcgi/streamreader.go
Normal file
@ -0,0 +1,28 @@
|
||||
package fastcgi
|
||||
|
||||
type streamReader struct {
|
||||
c *FCGIClient
|
||||
buf []byte
|
||||
}
|
||||
|
||||
func (w *streamReader) Read(p []byte) (n int, err error) {
|
||||
|
||||
if len(p) > 0 {
|
||||
if len(w.buf) == 0 {
|
||||
rec := &Record{}
|
||||
w.buf, err = rec.read(w.c.rwc)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
n = len(p)
|
||||
if n > len(w.buf) {
|
||||
n = len(w.buf)
|
||||
}
|
||||
copy(p, w.buf[:n])
|
||||
w.buf = w.buf[n:]
|
||||
}
|
||||
|
||||
return
|
||||
}
|
29
fastcgi/streamwriter.go
Normal file
29
fastcgi/streamwriter.go
Normal file
@ -0,0 +1,29 @@
|
||||
package fastcgi
|
||||
|
||||
// streamWriter abstracts out the separation of a stream into discrete records.
|
||||
// It only writes maxWrite bytes at a time.
|
||||
type streamWriter struct {
|
||||
c *FCGIClient
|
||||
recType FCGIRequestType
|
||||
}
|
||||
|
||||
func (w *streamWriter) Write(p []byte) (int, error) {
|
||||
nn := 0
|
||||
for len(p) > 0 {
|
||||
n := len(p)
|
||||
if n > maxWrite {
|
||||
n = maxWrite
|
||||
}
|
||||
if err := w.c.writeRecord(w.recType, p[:n]); err != nil {
|
||||
return nn, err
|
||||
}
|
||||
nn += n
|
||||
p = p[n:]
|
||||
}
|
||||
return nn, nil
|
||||
}
|
||||
|
||||
func (w *streamWriter) Close() error {
|
||||
// send empty record to close the stream
|
||||
return w.c.writeRecord(w.recType, nil)
|
||||
}
|
14
index.php
Normal file
14
index.php
Normal file
@ -0,0 +1,14 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title></title>
|
||||
</head>
|
||||
<body>
|
||||
<?php
|
||||
var_dump($_GET);
|
||||
?>
|
||||
<h1>Name is: <?php echo $_GET['name'] ?? 'NOT SET' ?></h1>
|
||||
<h2>SAPI NAME: <?php echo php_sapi_name() ?>
|
||||
</body>
|
||||
</html>
|
11
installer/installer.go
Normal file
11
installer/installer.go
Normal file
@ -0,0 +1,11 @@
|
||||
package installer
|
||||
|
||||
// This package handle installing any
|
||||
// necessary programs we need to run
|
||||
// the oasis environment such as:
|
||||
// - Nginx
|
||||
// - DNSMasq
|
||||
// It uses the package manager for
|
||||
// the OS and ensures any config
|
||||
// files needed go in the right
|
||||
// place.
|
5
installer/os.go
Normal file
5
installer/os.go
Normal file
@ -0,0 +1,5 @@
|
||||
package installer
|
||||
|
||||
// Return the correct package manager
|
||||
// and config paths for the current
|
||||
// operating system.
|
47
main.go
Normal file
47
main.go
Normal file
@ -0,0 +1,47 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"log"
|
||||
"net/http"
|
||||
|
||||
"github.com/javif89/oasis/fastcgi"
|
||||
)
|
||||
|
||||
func handleRequest(w http.ResponseWriter, r *http.Request) {
|
||||
fmt.Println("Request received")
|
||||
req := fastcgi.RequestFromHttp(r)
|
||||
req.Script("/home/javi/projects/oasis/index.php")
|
||||
req.TypeGet()
|
||||
|
||||
fcgiClient, err := fastcgi.Dial("unix", "/var/run/php/php8.3-fpm.sock")
|
||||
if err != nil {
|
||||
log.Println("err:", err)
|
||||
}
|
||||
|
||||
resp, err := fcgiClient.Do(req)
|
||||
if err != nil {
|
||||
log.Println("err:", err)
|
||||
}
|
||||
|
||||
content, err := io.ReadAll(resp.Body)
|
||||
if err != nil {
|
||||
log.Println("err:", err)
|
||||
}
|
||||
|
||||
w.Write(content)
|
||||
}
|
||||
|
||||
func main() {
|
||||
|
||||
// Server
|
||||
http.HandleFunc("/", handleRequest)
|
||||
|
||||
fmt.Println("Starting server")
|
||||
|
||||
err := http.ListenAndServe(":8000", nil)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user