mirror of
https://github.com/pldubouilh/gossa
synced 2025-12-06 08:22:32 +01:00
commit
f427a0fefd
8 changed files with 140 additions and 88 deletions
19
Makefile
19
Makefile
|
|
@ -1,30 +1,29 @@
|
|||
build:
|
||||
run:
|
||||
make embed
|
||||
go vet && go fmt
|
||||
CGO_ENABLED=0 go build gossa.go
|
||||
rm gossa.go
|
||||
./gossa fixture
|
||||
|
||||
watch:
|
||||
ls src/* | entr -rc make run
|
||||
|
||||
embed:
|
||||
echo "embedding css and js into binary"
|
||||
cp src/main.go gossa.go
|
||||
perl -pe 's/template_will_be_here/`cat src\/template.go`/ge' -i gossa.go
|
||||
perl -pe 's/css_will_be_here/`cat src\/style.css`/ge' -i gossa.go
|
||||
perl -pe 's/theme_will_be_here/`cat src\/theme.css`/ge' -i gossa.go
|
||||
perl -pe 's/js_will_be_here/`cat src\/script.js`/ge' -i gossa.go
|
||||
perl -pe 's/favicon_will_be_here/`base64 -w0 src\/favicon.png`/ge' -i gossa.go
|
||||
|
||||
run:
|
||||
make build
|
||||
./gossa fixture
|
||||
|
||||
ci:
|
||||
cd src && go vet && go fmt
|
||||
timeout 10 make run &
|
||||
cd src && sleep 5 && go test
|
||||
cp src/gossa_test.go . && sleep 5 && go test; rm gossa gossa_test.go
|
||||
|
||||
ci-watch:
|
||||
ls src/* | entr -rc make ci
|
||||
|
||||
watch:
|
||||
ls src/* | entr -rc make run
|
||||
|
||||
build-all:
|
||||
make embed
|
||||
env GOOS=linux GOARCH=amd64 go build gossa.go
|
||||
|
|
|
|||
|
|
@ -5,13 +5,14 @@ gossa
|
|||
|
||||
[](https://travis-ci.org/pldubouilh/gossa)
|
||||
|
||||
🎶 A fast and simple webserver for your files, that's dependency-free and with under 240 lines for the server code, easily code-reviewable.
|
||||
🎶 A fast and simple webserver for your files, that's dependency-free and with under 210 lines for the server code, easily code-reviewable.
|
||||
|
||||
### features
|
||||
* browse through files/directories
|
||||
* upload with drag-and-drop
|
||||
* move/rename/delete files
|
||||
* browse through pictures with a full-screen carousel
|
||||
* stream videos directly from the browser
|
||||
* simple keyboard navigation/shortcuts
|
||||
* fast ; fills my 80MB/s AC wifi link
|
||||
|
||||
|
|
@ -32,12 +33,13 @@ make
|
|||
|-------------|-------------|
|
||||
|Arrows/Enter | browse through files/directories and pictures|
|
||||
|Ctrl/Meta + C | copy URL to clipboard|
|
||||
|Ctrl/Meta + B | toggle theme (dark/clear)|
|
||||
|\<any letter\> | search|
|
||||
|Ctrl/Meta + E | rename file/folder|
|
||||
|Ctrl/Meta + Del | delete file/folder|
|
||||
|Ctrl/Meta + D | create a new directory|
|
||||
|Ctrl/Meta + X | cut selected path|
|
||||
|Ctrl/Meta + V | paste previously selected paths to directory|
|
||||
|\<any letter\> | search|
|
||||
|
||||
### ui shortcuts
|
||||
|shortcut | action|
|
||||
|
|
|
|||
|
|
@ -77,7 +77,7 @@ func testDefaults(t *testing.T, url string) string {
|
|||
t.Fatal("error 中文 folder")
|
||||
}
|
||||
|
||||
if !strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-types icon-blank"></i></td> <td class="file-size"><code>0.2k</code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="custom_mime_type.types">custom_mime_type.types</a></td> </tr>`) {
|
||||
if !strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-types icon-blank"></i></td> <td class="file-size"><code>211.0B</code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="custom_mime_type.types">custom_mime_type.types</a></td> </tr>`) {
|
||||
t.Fatal("error row custom_mime_type")
|
||||
}
|
||||
|
||||
|
|
@ -116,7 +116,7 @@ func TestGetFolder(t *testing.T) {
|
|||
}
|
||||
|
||||
bodyStr = testDefaults(t, "http://127.0.0.1:8001/")
|
||||
if !strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-folder icon-blank"></i></td> <td class="file-size"><code>0</code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="AAA">AAA/</a></td> </tr>`) {
|
||||
if !strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-folder icon-blank"></i></td> <td class="file-size"><code></code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="AAA">AAA/</a></td> </tr>`) {
|
||||
t.Fatal("error new folder created")
|
||||
}
|
||||
|
||||
|
|
@ -140,7 +140,7 @@ func TestGetFolder(t *testing.T) {
|
|||
}
|
||||
|
||||
bodyStr = testDefaults(t, "http://127.0.0.1:8001/")
|
||||
if strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-folder icon-blank"></i></td> <td class="file-size"><code>0</code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="AAA">AAA/</a></td> </tr>`) {
|
||||
if strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-folder icon-blank"></i></td> <td class="file-size"><code></code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="AAA">AAA/</a></td> </tr>`) {
|
||||
t.Fatal("error folder moved")
|
||||
}
|
||||
|
||||
|
|
@ -157,7 +157,7 @@ func TestGetFolder(t *testing.T) {
|
|||
}
|
||||
|
||||
bodyStr = testDefaults(t, "http://127.0.0.1:8001/")
|
||||
if !strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-하 하 icon-blank"></i></td> <td class="file-size"><code>0.0k</code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="%E1%84%92%E1%85%A1%20%E1%84%92%E1%85%A1">하 하</a></td> </tr>`) {
|
||||
if !strings.Contains(bodyStr, `<tr> <td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-하 하 icon-blank"></i></td> <td class="file-size"><code>9.0B</code></td> <td class="arrow"><i class="arrow-icon"></i></td> <td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="%E1%84%92%E1%85%A1%20%E1%84%92%E1%85%A1">하 하</a></td> </tr>`) {
|
||||
t.Fatal("error checking new file row")
|
||||
}
|
||||
|
||||
99
src/main.go
99
src/main.go
|
|
@ -6,6 +6,7 @@ import (
|
|||
"flag"
|
||||
"fmt"
|
||||
"html"
|
||||
"html/template"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
|
|
@ -21,14 +22,23 @@ var host = flag.String("h", "127.0.0.1", "host to listen to")
|
|||
var port = flag.String("p", "8001", "port to listen to")
|
||||
var verb = flag.Bool("verb", true, "verbosity")
|
||||
var skipHidden = flag.Bool("k", true, "skip hidden files")
|
||||
|
||||
var initPath = ""
|
||||
var css = `css_will_be_here` // js will be embedded here
|
||||
var js = `js_will_be_here` // id. css
|
||||
var favicon = "data:image/png;base64,favicon_will_be_here" // id. b64 favicon
|
||||
var units = [8]string{"k", "M", "G", "T", "P", "E", "Z", "Y"}
|
||||
|
||||
var fs http.Handler
|
||||
var page, _ = template.New("pageTemplate").Parse(`template_will_be_here`)
|
||||
|
||||
type rowTemplate struct {
|
||||
Name string
|
||||
Href template.HTML
|
||||
Size string
|
||||
Ext string
|
||||
}
|
||||
|
||||
type pageTemplate struct {
|
||||
Title template.HTML
|
||||
RowsFiles []rowTemplate
|
||||
RowsFolders []rowTemplate
|
||||
}
|
||||
|
||||
type rpcCall struct {
|
||||
Call string `json:"call"`
|
||||
|
|
@ -47,31 +57,17 @@ func logVerb(s ...interface{}) {
|
|||
}
|
||||
}
|
||||
|
||||
func sizeToString(bytes float64) string {
|
||||
if bytes == 0 {
|
||||
return "0"
|
||||
}
|
||||
var u = -1
|
||||
func sizeToString(bytes int64) string {
|
||||
units := [9]string{"B", "k", "M", "G", "T", "P", "E", "Z", "Y"}
|
||||
b := float64(bytes)
|
||||
u := 0
|
||||
for {
|
||||
bytes = bytes / 1024
|
||||
if b < 1024 {
|
||||
return strconv.FormatFloat(b, 'f', 1, 64) + units[u]
|
||||
}
|
||||
b = b / 1024
|
||||
u++
|
||||
if bytes < 1024 {
|
||||
return strconv.FormatFloat(bytes, 'f', 1, 64) + units[u]
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func row(name string, href string, size float64, ext string) string {
|
||||
if strings.HasPrefix(href, "/") {
|
||||
href = strings.Replace(href, "/", "", 1)
|
||||
}
|
||||
|
||||
return `<tr>
|
||||
<td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-` + strings.ToLower(ext) + ` icon-blank"></i></td>
|
||||
<td class="file-size"><code>` + sizeToString(size) + `</code></td>
|
||||
<td class="arrow"><i class="arrow-icon"></i></td>
|
||||
<td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="` + url.PathEscape(href) + `">` + name + `</a></td>
|
||||
</tr>`
|
||||
}
|
||||
|
||||
func replyList(w http.ResponseWriter, path string) {
|
||||
|
|
@ -79,52 +75,34 @@ func replyList(w http.ResponseWriter, path string) {
|
|||
path += "/"
|
||||
}
|
||||
|
||||
var head = `<!doctype html><html>
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width">
|
||||
<title>` + html.EscapeString(path) + `</title>
|
||||
<link href="` + favicon + `" rel="icon" type="image/png"/>
|
||||
<script>window.onload = function(){` + js + `}</script>
|
||||
<style type="text/css">` + css + `</style>
|
||||
</head>
|
||||
<body>
|
||||
<div id="drop-grid"> Drop here to upload </div>
|
||||
<h1>.` + html.EscapeString(path) + `</h1>
|
||||
<div class="icHolder"><div style="display:none;" class="ic icon-large-images" onclick="window.picsToggle()"></div>
|
||||
<div class="ic icon-large-folder" onclick="window.mkdirBtn()"></div></div>
|
||||
<div id="pics" style="display:none;"> <div onclick="window.picsToggle()" id="picsToggleCinema"></div> <img onclick="window.picsNav()" id="picsHolder"/> <span id="picsLabel"></span> </div>
|
||||
<table>`
|
||||
|
||||
_files, err := ioutil.ReadDir(initPath + path)
|
||||
check(err)
|
||||
|
||||
p := pageTemplate{}
|
||||
if path != "/" {
|
||||
head += row("../", "../", 0, "folder")
|
||||
p.RowsFolders = append(p.RowsFolders, rowTemplate{"../", "../", "", "folder"})
|
||||
}
|
||||
|
||||
var dirs = ""
|
||||
var files = ""
|
||||
|
||||
for _, el := range _files {
|
||||
var name = el.Name()
|
||||
name := el.Name()
|
||||
href := url.PathEscape(name)
|
||||
if *skipHidden && strings.HasPrefix(name, ".") {
|
||||
continue
|
||||
}
|
||||
|
||||
if el.IsDir() && strings.HasPrefix(href, "/") {
|
||||
href = strings.Replace(href, "/", "", 1)
|
||||
}
|
||||
if el.IsDir() {
|
||||
dirs += row(name+"/", name, 0, "folder")
|
||||
p.RowsFolders = append(p.RowsFolders, rowTemplate{name + "/", template.HTML(href), "", "folder"})
|
||||
} else {
|
||||
var sl = strings.Split(name, ".")
|
||||
var ext = sl[len(sl)-1]
|
||||
files += row(name, name, float64(el.Size()), ext)
|
||||
sl := strings.Split(name, ".")
|
||||
ext := strings.ToLower(sl[len(sl)-1])
|
||||
p.RowsFiles = append(p.RowsFiles, rowTemplate{name, template.HTML(href), sizeToString(el.Size()), ext})
|
||||
}
|
||||
}
|
||||
|
||||
w.Write([]byte(head + dirs + files + `</table>
|
||||
<br><address><a href="https://github.com/pldubouilh/gossa">Gossa 🎶</a></address>
|
||||
<div id="progress" style="display:none;"><span id="dlBarName"></span><div id="dlBarPc">1%</div></div>
|
||||
</body></html>`))
|
||||
p.Title = template.HTML(html.EscapeString(path))
|
||||
page.Execute(w, p)
|
||||
}
|
||||
|
||||
func doContent(w http.ResponseWriter, r *http.Request) {
|
||||
|
|
@ -206,7 +184,6 @@ func checkPath(p string) (string, error) {
|
|||
|
||||
func main() {
|
||||
flag.Parse()
|
||||
|
||||
if len(flag.Args()) == 0 {
|
||||
initPath = "."
|
||||
} else {
|
||||
|
|
@ -217,11 +194,11 @@ func main() {
|
|||
initPath, err = filepath.Abs(initPath)
|
||||
check(err)
|
||||
|
||||
var hostString = *host + ":" + *port
|
||||
hostString := *host + ":" + *port
|
||||
fmt.Println("Gossa startig on directory " + initPath)
|
||||
fmt.Println("Listening on http://" + hostString)
|
||||
|
||||
var root = http.Dir(initPath)
|
||||
root := http.Dir(initPath)
|
||||
fs = http.StripPrefix("/", http.FileServer(root))
|
||||
|
||||
http.HandleFunc("/rpc", rpc)
|
||||
|
|
|
|||
|
|
@ -163,6 +163,11 @@ const setBackgroundLinks = t => { t.style.backgroundColor = 'rgba(123, 123, 123,
|
|||
|
||||
const getLink = e => e.target.parentElement.querySelectorAll('a.list-links')[0]
|
||||
|
||||
upGrid.ondragleave = e => {
|
||||
cancelDefault(e)
|
||||
upGrid.style.display = 'none'
|
||||
}
|
||||
|
||||
document.ondragenter = e => {
|
||||
if (isPicMode()) { return }
|
||||
cancelDefault(e)
|
||||
|
|
@ -181,10 +186,7 @@ document.ondragenter = e => {
|
|||
}
|
||||
}
|
||||
|
||||
upGrid.ondragleave = e => {
|
||||
cancelDefault(e)
|
||||
upGrid.style.display = 'none'
|
||||
}
|
||||
document.ondragend = e => resetBackgroundLinks()
|
||||
|
||||
document.ondragover = e => {
|
||||
cancelDefault(e)
|
||||
|
|
@ -402,7 +404,7 @@ document.body.addEventListener('keydown', e => {
|
|||
return prevent(e) || picsNav(false) || prevPage()
|
||||
|
||||
case 'Escape':
|
||||
return prevent(e) || picsOff()
|
||||
return prevent(e) || resetBackgroundLinks() || picsOff()
|
||||
}
|
||||
|
||||
// Ctrl keys
|
||||
|
|
@ -426,11 +428,14 @@ document.body.addEventListener('keydown', e => {
|
|||
|
||||
case 'KeyD':
|
||||
return prevent(e) || isPicMode() || window.mkdirBtn()
|
||||
|
||||
case 'KeyB':
|
||||
return prevent(e) || toggleTheme()
|
||||
}
|
||||
}
|
||||
|
||||
// text search
|
||||
if (e.code.includes('Key')) {
|
||||
if (e.code.includes('Key') && !e.ctrlKey && !e.metaKey) {
|
||||
typedPath += e.code.replace('Key', '').toLocaleLowerCase()
|
||||
clearTimeout(typedToken)
|
||||
typedToken = setTimeout(() => { typedPath = '' }, 1000)
|
||||
|
|
|
|||
|
|
@ -1,3 +1,9 @@
|
|||
a {
|
||||
text-decoration: none;
|
||||
background-color: transparent !important;
|
||||
/* border-bottom: .01em solid #dfe6e9; */
|
||||
}
|
||||
|
||||
.icon {
|
||||
display: block;
|
||||
height: 16px;
|
||||
|
|
@ -75,7 +81,6 @@ h1 {
|
|||
}
|
||||
|
||||
#progress {
|
||||
background-color: white;
|
||||
width: 99%;
|
||||
left: 0.5%;
|
||||
right: 0.5%;
|
||||
|
|
@ -83,7 +88,6 @@ h1 {
|
|||
bottom: 0px;
|
||||
padding-bottom: 10px;
|
||||
max-height: 50%;
|
||||
overflow-y: scroll;
|
||||
overflow-x: hidden;
|
||||
}
|
||||
|
||||
|
|
@ -118,10 +122,9 @@ h1 {
|
|||
text-align: center;
|
||||
font-size: 4em;
|
||||
font-family: Helvetica;
|
||||
background-color: white;
|
||||
color: green;
|
||||
opacity: 0.8;
|
||||
backdrop-filter: grayscale(100%);
|
||||
background-color: rgba(123, 123, 123, 0.2)
|
||||
}
|
||||
|
||||
#pics {
|
||||
|
|
|
|||
59
src/template.go
Normal file
59
src/template.go
Normal file
|
|
@ -0,0 +1,59 @@
|
|||
<!doctype html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width">
|
||||
<title>{{.Title}}</title>
|
||||
<link href="data:image/png;base64,favicon_will_be_here" rel="icon" type="image/png" />
|
||||
<style type="text/css">css_will_be_here</style>
|
||||
<style type="text/css" id="theme">theme_will_be_here</style>
|
||||
<script>
|
||||
const theme = document.getElementById('theme')
|
||||
const themeNow = () => localStorage.getItem('theme')
|
||||
const setTheme = () => { theme.disabled = themeNow() === 'regular' }
|
||||
const toggleTheme = () => localStorage.setItem('theme', (themeNow() === 'regular' ? 'alt' : 'regular')) || setTheme()
|
||||
setTheme()
|
||||
</script>
|
||||
<script>window.onload = function () { js_will_be_here }</script>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div id="drop-grid"> Drop here to upload </div>
|
||||
|
||||
<h1>.{{.Title}}</h1>
|
||||
|
||||
<div class="icHolder">
|
||||
<div style="display:none;" class="ic icon-large-images" onclick="window.picsToggle()"></div>
|
||||
<div class="ic icon-large-folder" onclick="window.mkdirBtn()"></div>
|
||||
</div>
|
||||
|
||||
<div id="pics" style="display:none;">
|
||||
<div onclick="window.picsToggle()" id="picsToggleCinema"></div> <img onclick="window.picsNav()" id="picsHolder" />
|
||||
<span id="picsLabel"></span>
|
||||
</div>
|
||||
|
||||
<table>
|
||||
{{range .RowsFolders}}
|
||||
<tr>
|
||||
<td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-{{.Ext}} icon-blank"></i></td>
|
||||
<td class="file-size"><code>{{.Size}}</code></td>
|
||||
<td class="arrow"><i class="arrow-icon"></i></td>
|
||||
<td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="{{.Href}}">{{.Name}}</a></td>
|
||||
</tr>
|
||||
{{end}}
|
||||
{{range .RowsFiles}}
|
||||
<tr>
|
||||
<td><i ondblclick="return rm(event)" onclick="return rename(event)" class="btn icon icon-{{.Ext}} icon-blank"></i></td>
|
||||
<td class="file-size"><code>{{.Size}}</code></td>
|
||||
<td class="arrow"><i class="arrow-icon"></i></td>
|
||||
<td class="display-name"><a class="list-links" onclick="return onClickLink(event)" href="{{.Href}}">{{.Name}}</a></td>
|
||||
</tr>
|
||||
{{end}}
|
||||
</table>
|
||||
|
||||
<div id="progress" style="display:none;">
|
||||
<span id="dlBarName"></span>
|
||||
<div id="dlBarPc">1%</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
7
src/theme.css
Normal file
7
src/theme.css
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
html, a {
|
||||
background-color: #2d3436; color: #dfe6e9;
|
||||
}
|
||||
|
||||
.arrow, .icon-large-images {
|
||||
filter: invert(100%) !important;
|
||||
}
|
||||
Loading…
Reference in a new issue