mirror of
https://code.rocketnine.space/tslocum/gmitohtml.git
synced 2024-11-27 11:18:13 +01:00
Initial commit
This commit is contained in:
commit
260018cc9d
7 changed files with 373 additions and 0 deletions
5
.gitignore
vendored
Normal file
5
.gitignore
vendored
Normal file
|
@ -0,0 +1,5 @@
|
|||
.idea/
|
||||
dist/
|
||||
vendor/
|
||||
*.sh
|
||||
/gmitohtml
|
21
.gitlab-ci.yml
Normal file
21
.gitlab-ci.yml
Normal file
|
@ -0,0 +1,21 @@
|
|||
image: golang:latest
|
||||
|
||||
stages:
|
||||
- validate
|
||||
- build
|
||||
|
||||
fmt:
|
||||
stage: validate
|
||||
script:
|
||||
- gofmt -l -s -e .
|
||||
- exit $(gofmt -l -s -e . | wc -l)
|
||||
|
||||
vet:
|
||||
stage: validate
|
||||
script:
|
||||
- go vet -composites=false ./...
|
||||
|
||||
test:
|
||||
stage: validate
|
||||
script:
|
||||
- go test -race -v ./...
|
21
LICENSE
Normal file
21
LICENSE
Normal file
|
@ -0,0 +1,21 @@
|
|||
MIT License
|
||||
|
||||
Copyright (c) 2020 Trevor Slocum <trevor@rocketnine.space>
|
||||
|
||||
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.
|
37
README.md
Normal file
37
README.md
Normal file
|
@ -0,0 +1,37 @@
|
|||
# gmitohtml
|
||||
[![CI status](https://gitlab.com/tslocum/gmitohtml/badges/master/pipeline.svg)](https://gitlab.com/tslocum/gmitohtml/commits/master)
|
||||
[![Donate](https://img.shields.io/liberapay/receives/rocketnine.space.svg?logo=liberapay)](https://liberapay.com/rocketnine.space)
|
||||
|
||||
[Gemini](https://gemini.circumlunar.space) to HTML conversion tool and daemon
|
||||
|
||||
## Download
|
||||
|
||||
gmitohtml is written in [Go](https://golang.org). Run the following command to
|
||||
download and build gmitohtml from source.
|
||||
|
||||
```bash
|
||||
go get gitlab.com/tslocum/gmitohtml
|
||||
```
|
||||
|
||||
The resulting binary is available as `~/go/bin/gmitohtml`.
|
||||
|
||||
## Usage
|
||||
|
||||
Convert a single document:
|
||||
|
||||
```bash
|
||||
gmitohtml < document.gmi
|
||||
```
|
||||
|
||||
Run as daemon at `http://localhost:8080`:
|
||||
|
||||
```bash
|
||||
# Start the daemon:
|
||||
gmitohtml --daemon=localhost:8080
|
||||
# Access via browser:
|
||||
xdg-open http://localhost:8080/gemini/twins.rocketnine.space/
|
||||
```
|
||||
|
||||
## Support
|
||||
|
||||
Please share issues and suggestions [here](https://gitlab.com/tslocum/gmitohtml/issues).
|
3
go.mod
Normal file
3
go.mod
Normal file
|
@ -0,0 +1,3 @@
|
|||
module gitlab.com/tslocum/gmitohtml
|
||||
|
||||
go 1.15
|
69
main.go
Normal file
69
main.go
Normal file
|
@ -0,0 +1,69 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
"gitlab.com/tslocum/gmitohtml/pkg/gmitohtml"
|
||||
|
||||
"flag"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"net/url"
|
||||
"os"
|
||||
"os/exec"
|
||||
"runtime"
|
||||
)
|
||||
|
||||
func openBrowser(url string) {
|
||||
var err error
|
||||
switch runtime.GOOS {
|
||||
case "linux":
|
||||
err = exec.Command("xdg-open", url).Start()
|
||||
case "windows":
|
||||
err = exec.Command("rundll32", "url.dll,FileProtocolHandler", url).Start()
|
||||
case "darwin":
|
||||
err = exec.Command("open", url).Start()
|
||||
default:
|
||||
err = fmt.Errorf("unsupported platform")
|
||||
}
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
func main() {
|
||||
var view bool
|
||||
var daemon string
|
||||
flag.BoolVar(&view, "view", false, "open web browser")
|
||||
flag.StringVar(&daemon, "daemon", "", "start daemon on specified address")
|
||||
// TODO option to include response header in page
|
||||
flag.Parse()
|
||||
|
||||
if daemon != "" {
|
||||
err := gmitohtml.StartDaemon(daemon)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
if view {
|
||||
openBrowser("http://" + daemon)
|
||||
}
|
||||
|
||||
select {} //TODO
|
||||
}
|
||||
|
||||
data, err := ioutil.ReadAll(os.Stdin)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
data = gmitohtml.Convert(data, "")
|
||||
|
||||
data = append([]byte("<!DOCTYPE html>\n<html>\n<body>\n"), data...)
|
||||
data = append(data, []byte("\n</body>\n</html>")...)
|
||||
|
||||
if view {
|
||||
openBrowser(string(append([]byte("data:text/html,"), []byte(url.PathEscape(string(data)))...)))
|
||||
return
|
||||
}
|
||||
fmt.Print(gmitohtml.Convert(data, ""))
|
||||
}
|
217
pkg/gmitohtml/convert.go
Normal file
217
pkg/gmitohtml/convert.go
Normal file
|
@ -0,0 +1,217 @@
|
|||
package gmitohtml
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"bytes"
|
||||
"crypto/tls"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"path"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// ErrInvalidURL is the error returned when the URL is invalid.
|
||||
var ErrInvalidURL = errors.New("invalid URL")
|
||||
|
||||
var daemonAddress string
|
||||
|
||||
func rewriteURL(u string, loc *url.URL) string {
|
||||
if daemonAddress != "" {
|
||||
if strings.HasPrefix(u, "gemini://") {
|
||||
return "http://" + daemonAddress + "/gemini/" + u[9:]
|
||||
} else if strings.Contains(u, "://") {
|
||||
return u
|
||||
} else if loc != nil && len(u) > 0 && !strings.HasPrefix(u, "//") {
|
||||
newPath := u
|
||||
if u[0] != '/' {
|
||||
newPath = path.Join(loc.Path, u)
|
||||
}
|
||||
return "http://" + daemonAddress + "/gemini/" + loc.Host + newPath
|
||||
}
|
||||
return "http://" + daemonAddress + "/gemini/" + u
|
||||
}
|
||||
return u
|
||||
}
|
||||
|
||||
// Convert converts text/gemini to text/html.
|
||||
func Convert(page []byte, u string) []byte {
|
||||
var result []byte
|
||||
|
||||
var lastPreformatted bool
|
||||
var preformatted bool
|
||||
|
||||
parsedURL, err := url.Parse(u)
|
||||
if err != nil {
|
||||
parsedURL = nil
|
||||
err = nil
|
||||
}
|
||||
|
||||
scanner := bufio.NewScanner(bytes.NewReader(page))
|
||||
for scanner.Scan() {
|
||||
line := scanner.Bytes()
|
||||
l := len(line)
|
||||
if l >= 3 && string(line[0:3]) == "```" {
|
||||
preformatted = !preformatted
|
||||
continue
|
||||
}
|
||||
|
||||
if preformatted != lastPreformatted {
|
||||
if preformatted {
|
||||
result = append(result, []byte("<pre>\n")...)
|
||||
} else {
|
||||
result = append(result, []byte("</pre>\n")...)
|
||||
}
|
||||
lastPreformatted = preformatted
|
||||
}
|
||||
|
||||
if preformatted {
|
||||
result = append(result, line...)
|
||||
result = append(result, []byte("\n")...)
|
||||
continue
|
||||
}
|
||||
|
||||
if l >= 7 && bytes.HasPrefix(line, []byte("=> ")) {
|
||||
split := bytes.SplitN(line[3:], []byte(" "), 2)
|
||||
if len(split) == 2 {
|
||||
link := append([]byte(`<a href="`), rewriteURL(string(split[0]), parsedURL)...)
|
||||
link = append(link, []byte(`">`)...)
|
||||
link = append(link, split[1]...)
|
||||
link = append(link, []byte(`</a>`)...)
|
||||
result = append(result, link...)
|
||||
result = append(result, []byte("<br>")...)
|
||||
continue
|
||||
}
|
||||
}
|
||||
|
||||
heading := 0
|
||||
for i := 0; i < l; i++ {
|
||||
if line[i] == '#' {
|
||||
heading++
|
||||
} else {
|
||||
break
|
||||
}
|
||||
}
|
||||
if heading > 0 {
|
||||
result = append(result, []byte(fmt.Sprintf("<h%d>%s</h%d>", heading, line, heading))...)
|
||||
continue
|
||||
}
|
||||
|
||||
result = append(result, line...)
|
||||
result = append(result, []byte("<br>")...)
|
||||
}
|
||||
|
||||
return result
|
||||
}
|
||||
|
||||
// Fetch downloads and converts a Gemini page.
|
||||
func Fetch(u string, clientCertFile string, clientCertKey string) ([]byte, []byte, error) {
|
||||
if u == "" {
|
||||
return nil, nil, ErrInvalidURL
|
||||
}
|
||||
|
||||
requestURL, err := url.ParseRequestURI(u)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
if requestURL.Scheme == "" {
|
||||
requestURL.Scheme = "gemini"
|
||||
}
|
||||
if strings.IndexRune(requestURL.Host, ':') == -1 {
|
||||
requestURL.Host += ":1965"
|
||||
}
|
||||
|
||||
tlsConfig := &tls.Config{}
|
||||
|
||||
conn, err := tls.Dial("tcp", requestURL.Host, tlsConfig)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
// Send request header
|
||||
conn.Write([]byte(requestURL.String() + "\r\n"))
|
||||
|
||||
data, err := ioutil.ReadAll(conn)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
firstNewLine := -1
|
||||
l := len(data)
|
||||
if l > 2 {
|
||||
for i := 1; i < l; i++ {
|
||||
if data[i] == '\n' && data[i-1] == '\r' {
|
||||
firstNewLine = i
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
var header []byte
|
||||
if firstNewLine > -1 {
|
||||
header = data[:firstNewLine]
|
||||
data = data[firstNewLine+1:]
|
||||
}
|
||||
|
||||
if bytes.HasPrefix(header, []byte("text/html")) {
|
||||
return header, data, nil
|
||||
}
|
||||
return header, Convert(data, requestURL.String()), nil
|
||||
}
|
||||
|
||||
func handleRequest(writer http.ResponseWriter, request *http.Request) {
|
||||
defer request.Body.Close()
|
||||
if request.URL == nil {
|
||||
return
|
||||
}
|
||||
|
||||
pathSplit := strings.Split(request.URL.Path, "/")
|
||||
if len(pathSplit) < 2 || pathSplit[1] != "gemini" {
|
||||
writer.Write([]byte("Error: invalid protocol, only Gemini is supported"))
|
||||
return
|
||||
}
|
||||
|
||||
u, err := url.ParseRequestURI("gemini://" + strings.Join(pathSplit[2:], "/"))
|
||||
if err != nil {
|
||||
writer.Write([]byte("Error: invalid URL"))
|
||||
return
|
||||
}
|
||||
|
||||
header, data, err := Fetch(u.String(), "", "")
|
||||
if err != nil {
|
||||
fmt.Fprintf(writer, "Error: failed to fetch %s: %s", u, err)
|
||||
return
|
||||
}
|
||||
|
||||
if len(header) > 0 && header[0] == '3' {
|
||||
split := bytes.SplitN(header, []byte(" "), 2)
|
||||
if len(split) == 2 {
|
||||
writer.Header().Set("Location", rewriteURL(string(split[1]), request.URL))
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
if len(header) > 3 && !bytes.HasPrefix(header[3:], []byte("text/gemini")) {
|
||||
writer.Header().Set("Content-type", string(header[3:]))
|
||||
} else {
|
||||
writer.Header().Set("Content-type", "text/html; charset=utf-8")
|
||||
}
|
||||
|
||||
writer.Write([]byte("<!DOCTYPE html>\n<html>\n<body>\n"))
|
||||
writer.Write(data)
|
||||
writer.Write([]byte("\n</body>\n</html>"))
|
||||
}
|
||||
|
||||
// StartDaemon starts the page conversion daemon.
|
||||
func StartDaemon(address string) error {
|
||||
daemonAddress = address
|
||||
|
||||
http.HandleFunc("/", handleRequest)
|
||||
|
||||
go func() {
|
||||
log.Fatal(http.ListenAndServe(address, nil))
|
||||
}()
|
||||
return nil
|
||||
}
|
Loading…
Reference in a new issue