mirror of https://github.com/hunshcn/gh-proxy
commit
bdab7ddd67
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2020 hunshcn
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
@ -0,0 +1,109 @@
|
||||
'use strict'
|
||||
|
||||
/** @type {RequestInit} */
|
||||
const PREFLIGHT_INIT = {
|
||||
status: 204,
|
||||
headers: new Headers({
|
||||
'access-control-allow-origin': '*',
|
||||
'access-control-allow-methods': 'GET,POST,PUT,PATCH,TRACE,DELETE,HEAD,OPTIONS',
|
||||
'access-control-max-age': '1728000',
|
||||
}),
|
||||
}
|
||||
|
||||
const PREFIX = 'https://example.com/'
|
||||
|
||||
|
||||
/**
|
||||
* @param {any} body
|
||||
* @param {number} status
|
||||
* @param {Object<string, string>} headers
|
||||
*/
|
||||
function makeRes(body, status = 200, headers = {}) {
|
||||
headers['access-control-allow-origin'] = '*'
|
||||
return new Response(body, {status, headers})
|
||||
}
|
||||
|
||||
|
||||
addEventListener('fetch', e => {
|
||||
const ret = fetchHandler(e)
|
||||
.catch(err => makeRes('cfworker error:\n' + err.stack, 502))
|
||||
e.respondWith(ret)
|
||||
})
|
||||
|
||||
|
||||
/**
|
||||
* @param {FetchEvent} e
|
||||
*/
|
||||
async function fetchHandler(e) {
|
||||
const req = e.request
|
||||
const urlStr = req.url
|
||||
const urlObj = new URL(urlStr)
|
||||
let path = urlStr.substring(urlObj.origin.length + 1).replace(/^(https?):\/+/, '$1://')
|
||||
return httpHandler(req, path)
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param {Request} req
|
||||
* @param {string} pathname
|
||||
*/
|
||||
function httpHandler(req, pathname) {
|
||||
const reqHdrRaw = req.headers
|
||||
|
||||
// preflight
|
||||
if (req.method === 'OPTIONS' &&
|
||||
reqHdrRaw.has('access-control-request-headers')
|
||||
) {
|
||||
return new Response(null, PREFLIGHT_INIT)
|
||||
}
|
||||
|
||||
const reqHdrNew = new Headers(reqHdrRaw)
|
||||
|
||||
/** @type {RequestInit} */
|
||||
const reqInit = {
|
||||
method: req.method,
|
||||
headers: reqHdrNew,
|
||||
redirect: 'manual',
|
||||
body: req.body
|
||||
}
|
||||
return proxy(pathname, reqInit)
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} path
|
||||
* @param {RequestInit} reqInit
|
||||
*/
|
||||
async function proxy(path, reqInit) {
|
||||
const res = await fetch(path, reqInit)
|
||||
const resHdrOld = res.headers
|
||||
const resHdrNew = new Headers(resHdrOld)
|
||||
|
||||
const status = res.status
|
||||
|
||||
if (resHdrNew.has('location')) {
|
||||
let _location = resHdrNew.get('location')
|
||||
resHdrNew.set('location', PREFIX + _location)
|
||||
}
|
||||
resHdrNew.set('access-control-expose-headers', '*')
|
||||
resHdrNew.set('access-control-allow-origin', '*')
|
||||
|
||||
resHdrNew.delete('content-security-policy')
|
||||
resHdrNew.delete('content-security-policy-report-only')
|
||||
resHdrNew.delete('clear-site-data')
|
||||
if (path.substring(0, path.lastIndexOf("/")).endsWith('/info/lfs/objects')) {
|
||||
let text = await res.text()
|
||||
text = text.replaceAll('"href":"https://', '"href":"' + PREFIX + 'https://')
|
||||
resHdrNew.delete("content-length")
|
||||
return new Response(text, {
|
||||
status,
|
||||
headers: resHdrNew,
|
||||
})
|
||||
}
|
||||
|
||||
return new Response(res.body, {
|
||||
status,
|
||||
headers: resHdrNew,
|
||||
})
|
||||
}
|
@ -0,0 +1,144 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"net/http/httputil"
|
||||
"net/url"
|
||||
"sort"
|
||||
"strings"
|
||||
)
|
||||
|
||||
const HostPrefix = "http://127.0.0.1:8080/"
|
||||
|
||||
func NewServer(enableLog bool) *Server {
|
||||
director := func(req *http.Request) {
|
||||
if _, ok := req.Header["User-Agent"]; !ok {
|
||||
// explicitly disable User-Agent so it's not set to default value
|
||||
req.Header.Set("User-Agent", "")
|
||||
}
|
||||
}
|
||||
return &Server{proxy: &httputil.ReverseProxy{Director: director}, enableLog: enableLog}
|
||||
}
|
||||
|
||||
type Server struct {
|
||||
proxy *httputil.ReverseProxy
|
||||
enableLog bool
|
||||
}
|
||||
|
||||
func (s Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||
targetUrl := r.URL.RequestURI()
|
||||
targetUrl = targetUrl[1:]
|
||||
target, err := url.Parse(targetUrl)
|
||||
if err != nil {
|
||||
w.WriteHeader(500)
|
||||
return
|
||||
}
|
||||
r.Host = target.Host
|
||||
r.URL = target
|
||||
r.URL.Host = target.Host
|
||||
r.URL.Path = target.Path
|
||||
r.URL.RawPath = target.RawPath
|
||||
mw := &MWriter{Next: s.proxy.ServeHTTP, EnableLog: s.enableLog}
|
||||
|
||||
if strings.HasSuffix(targetUrl[:strings.LastIndexByte(targetUrl, '/')], "/info/lfs/objects") {
|
||||
mw.ReplaceFunc = func(mw *MWriter) {
|
||||
for k, v := range mw.cache.header {
|
||||
if k == "Content-Length" {
|
||||
continue
|
||||
}
|
||||
mw.writer.Header()[k] = v
|
||||
}
|
||||
if mw.cache.statusCode != 0 {
|
||||
mw.writer.WriteHeader(mw.cache.statusCode)
|
||||
}
|
||||
content := mw.cache.body.String()
|
||||
content = strings.ReplaceAll(content, "\"href\":\"https://", "\"href\":\""+HostPrefix+"https://")
|
||||
_, _ = mw.writer.Write([]byte(content))
|
||||
}
|
||||
}
|
||||
mw.ServeHTTP(w, r)
|
||||
}
|
||||
|
||||
type MWriter struct {
|
||||
writer http.ResponseWriter
|
||||
request *http.Request
|
||||
cache struct {
|
||||
header http.Header
|
||||
body bytes.Buffer
|
||||
statusCode int
|
||||
}
|
||||
wroteLog bool
|
||||
|
||||
Next func(http.ResponseWriter, *http.Request)
|
||||
ReplaceFunc func(*MWriter)
|
||||
EnableLog bool
|
||||
}
|
||||
|
||||
func (mw *MWriter) Log() {
|
||||
if mw.EnableLog && !mw.wroteLog {
|
||||
if mw.cache.statusCode == 0 {
|
||||
mw.cache.statusCode = http.StatusOK
|
||||
}
|
||||
fmt.Println(mw.request.Method, mw.request.URL.String(), mw.cache.statusCode)
|
||||
header := mw.writer.Header()
|
||||
if mw.ReplaceFunc != nil {
|
||||
header = mw.cache.header
|
||||
}
|
||||
var keys []string
|
||||
for key := range header {
|
||||
keys = append(keys, key)
|
||||
}
|
||||
sort.Strings(keys)
|
||||
for _, key := range keys {
|
||||
for _, value := range header[key] {
|
||||
fmt.Printf("%s: %s\n", key, value)
|
||||
}
|
||||
}
|
||||
fmt.Println()
|
||||
mw.wroteLog = true
|
||||
}
|
||||
}
|
||||
|
||||
func (mw *MWriter) Write(data []byte) (int, error) {
|
||||
mw.Log()
|
||||
if mw.ReplaceFunc == nil {
|
||||
return mw.writer.Write(data)
|
||||
}
|
||||
return mw.cache.body.Write(data)
|
||||
}
|
||||
|
||||
func (mw *MWriter) WriteHeader(statusCode int) {
|
||||
for i := range mw.Header()["Location"] {
|
||||
mw.Header()["Location"][i] = HostPrefix + mw.Header()["Location"][0]
|
||||
}
|
||||
mw.Log()
|
||||
if mw.ReplaceFunc == nil {
|
||||
mw.writer.WriteHeader(statusCode)
|
||||
}
|
||||
mw.cache.statusCode = statusCode
|
||||
}
|
||||
|
||||
func (mw *MWriter) Header() http.Header {
|
||||
if mw.ReplaceFunc == nil {
|
||||
return mw.writer.Header()
|
||||
}
|
||||
if mw.cache.header == nil {
|
||||
mw.cache.header = mw.writer.Header().Clone()
|
||||
}
|
||||
return mw.cache.header
|
||||
}
|
||||
|
||||
func (mw *MWriter) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||
mw.writer = w
|
||||
mw.request = r
|
||||
mw.Next(mw, r)
|
||||
if mw.ReplaceFunc != nil {
|
||||
mw.ReplaceFunc(mw)
|
||||
}
|
||||
}
|
||||
|
||||
func main() {
|
||||
http.ListenAndServe(":8080", NewServer(true))
|
||||
}
|
Loading…
Reference in new issue