Compare commits
33 Commits
ece0e31320
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5470ba1298 | ||
|
|
13fae1c23f | ||
|
|
21719a6cf7 | ||
| addaade269 | |||
|
|
6ee6c9c8d9 | ||
|
|
a0133e0981 | ||
| 70c668bdf1 | |||
|
|
68c89de1a4 | ||
|
|
0b7de76874 | ||
|
|
797e115191 | ||
|
|
16a65df664 | ||
|
|
4e87c77ccb | ||
|
|
94b3f4c0f2 | ||
|
|
f574a00a8f | ||
|
|
5bce49eed4 | ||
|
|
51b788bb50 | ||
|
|
7f43eb43e0 | ||
|
|
81b960c231 | ||
|
|
26f1008dc4 | ||
|
|
f6558e3cd7 | ||
|
|
60f5495f79 | ||
|
|
fbbd78ce72 | ||
|
|
e5c7dc6f44 | ||
|
|
55112c96dc | ||
|
|
3fd6ab5675 | ||
|
|
3396109e00 | ||
|
|
fb6afc2ffe | ||
|
|
808ce89dfc | ||
|
|
e627d02d08 | ||
|
|
1e7bbc6e16 | ||
|
|
231ceea80a | ||
|
|
5b95af791e | ||
|
|
4e6f47bff9 |
26
README.md
26
README.md
@@ -2,7 +2,7 @@
|
||||
|
||||
I am using the api of [diyhrt.market](https://diyhrt.market/api/) to get the current stats data of estrogen stocks.
|
||||
|
||||
# Installation
|
||||
## Installation
|
||||
|
||||
To install the new tab page you can use `go install`
|
||||
|
||||
@@ -13,14 +13,26 @@ go install gitea.elara.ws/Hazel/transfem-startpage
|
||||
Then you can run the program `transfem-startpage`
|
||||
|
||||
```sh
|
||||
transfem-startpage
|
||||
transfem-startpage help
|
||||
```
|
||||
|
||||
To configure this new tab page as website, you can install the firefox extension [New Tab Override](https://addons.mozilla.org/en-US/firefox/addon/new-tab-override/). Then just configure the url as `http://127.0.0.1:{port}/`. The default port should be `5500` but it will also print it out when starting the server. Make sure to check the box `Set focus to the web page instead of the address bar` in the extension settings, because the new tab page auto focuses the search bar.
|
||||
|
||||
## CLI
|
||||
|
||||
```sh
|
||||
transfem-startpage {program} {...args}
|
||||
```
|
||||
|
||||
program | args | description
|
||||
---|---|---
|
||||
`help` | `program:optional` | get more information on how the cli or one program works
|
||||
`start` | `profile:optional` | start the webserver for a certain profile
|
||||
`cache` | `action:emum(clear;clean)` | so something with the cache
|
||||
|
||||
## Config and Profiles
|
||||
|
||||
This tool works with profiles. The default profile is `startpage`. If you want to load another profile just write it as command line arg after the command. To write a config File you can create the files here:
|
||||
This tool works with profiles. The default profile is `default`. If you want to load another profile just write it as command line arg after the command. To write a config File you can create the files here:
|
||||
|
||||
- `{profile}.toml`
|
||||
- `.{profile}.toml`
|
||||
@@ -43,8 +55,10 @@ air dev
|
||||
|
||||
## TODO
|
||||
|
||||
- implement templating for every one of the frontend files
|
||||
- implement functionality to clear and clean cache
|
||||
- host this website on a demo page
|
||||
- implement ctl
|
||||
- implement fetching in intervals
|
||||
- actually building and figuring out how I should do that
|
||||
- writing documentation
|
||||
- implement autocomplete with a nice go backend and fast communication. Since it all runs locally nobody should have privacy concerns NEEDS TO BE ABLE TO TOGGLED OFF FOR DEMO PAGE
|
||||
|
||||
WRITE DOCUMENTATION
|
||||
|
||||
17
demo.toml
Normal file
17
demo.toml
Normal file
@@ -0,0 +1,17 @@
|
||||
[Server]
|
||||
Port = 1312
|
||||
|
||||
[DiyHrt]
|
||||
FetchIntervals = 60
|
||||
|
||||
[Template]
|
||||
ActiveCard = "listings"
|
||||
PageTitle = "TransfemStartpage demo"
|
||||
HeaderPhrases = [
|
||||
"GirlJuice.Inject();",
|
||||
"You.Cute = true;",
|
||||
"You.Gay = true;",
|
||||
"Nazi.Punch();",
|
||||
"Dolls.GiveGuns();",
|
||||
"Firefox > Chrome"
|
||||
]
|
||||
11
dev.toml
11
dev.toml
@@ -3,3 +3,14 @@ Port = 1234
|
||||
|
||||
[Template]
|
||||
ActiveCard = "websites"
|
||||
|
||||
[[Template.Websites]]
|
||||
Name = "Transfem Startpage"
|
||||
Url = "https://gitea.elara.ws/Hazel/transfem-startpage"
|
||||
ImageUrl = "https://gitea.elara.ws/assets/img/logo.svg"
|
||||
|
||||
[[Template.Websites]]
|
||||
Name = "GoLang"
|
||||
Url = "https://go.dev/"
|
||||
ImageUrl = "https://go.dev/images/gophers/motorcycle.svg"
|
||||
|
||||
|
||||
@@ -26,7 +26,7 @@ body {
|
||||
display: grid;
|
||||
gap: 5em;
|
||||
|
||||
grid-template-rows: 10em 4em 15em;
|
||||
grid-template-rows: 10em 4em 17em;
|
||||
}
|
||||
|
||||
.search {
|
||||
@@ -49,16 +49,6 @@ body {
|
||||
color: black;
|
||||
}
|
||||
|
||||
@media (max-height: 300px) {
|
||||
.search-grid {
|
||||
grid-template-rows: 4em;
|
||||
}
|
||||
|
||||
.search-logo {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
||||
.cards {
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
@@ -70,6 +60,20 @@ body {
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
@media (max-height: 500px) {
|
||||
.search-grid {
|
||||
grid-template-rows: 4em;
|
||||
}
|
||||
|
||||
.search-logo {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.cards {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
||||
.card {
|
||||
background-color: rgba(255, 255, 255, 0.5);
|
||||
width: 10em;
|
||||
@@ -81,6 +85,7 @@ body {
|
||||
gap: 1em;
|
||||
padding: 1em;
|
||||
|
||||
height: 15em;
|
||||
border-radius: 1em;
|
||||
}
|
||||
|
||||
@@ -96,9 +101,8 @@ body {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
|
||||
.card-image {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
aspect-ratio: 1/1;
|
||||
}
|
||||
|
||||
|
||||
@@ -18,8 +18,9 @@
|
||||
|
||||
{{ if eq .ActiveCard "stores" }}
|
||||
<div class="cards" id="stores">
|
||||
{{ $T := .DiyHrtTarget }}
|
||||
{{range $Store := .Stores }}
|
||||
<a target="_blank" href="{{ $Store.Url }}" class="card">
|
||||
<a target="{{ $T }}" href="{{ $Store.Url }}" class="card">
|
||||
<h3>{{ $Store.Name }}</h3>
|
||||
</a>
|
||||
{{- end }}
|
||||
@@ -28,8 +29,9 @@
|
||||
|
||||
{{ if eq .ActiveCard "listings" }}
|
||||
<div class="cards" id="listings">
|
||||
{{ $T := .DiyHrtTarget }}
|
||||
{{range $Listing := .Listings }}
|
||||
<a target="_blank" href="{{ $Listing.Url }}" class="card {{ if $Listing.InStock }}in-stock{{ end }}">
|
||||
<a target="{{ $T }}" href="{{ $Listing.Url }}" class="card {{ if $Listing.InStock }}in-stock{{ end }}">
|
||||
<h3>{{ $Listing.ProductName }}</h3>
|
||||
<p>{{ $Listing.StoreName }} - {{ $Listing.Price }} {{ $Listing.PriceCurrency }}</p>
|
||||
</a>
|
||||
@@ -39,8 +41,9 @@
|
||||
|
||||
{{ if eq .ActiveCard "websites" }}
|
||||
<div class="cards" id="websites">
|
||||
{{ $T := .WebsiteTarget }}
|
||||
{{range $Website := .Websites }}
|
||||
<a target="_blank" href="{{ $Website.Url }}" class="card">
|
||||
<a href="{{ $Website.Url }}" class="card" target="{{ $T }}">
|
||||
<h3>{{ $Website.Name }}</h3>
|
||||
<img class="card-image" src="{{ $Website.ImageUrl }}" alt="{{ $Website.Name }} picture">
|
||||
</a>
|
||||
|
||||
@@ -4,90 +4,94 @@ const form = document.getElementById("search-form");
|
||||
const input = document.getElementById("search-input");
|
||||
|
||||
// https://stackoverflow.com/a/3809435/16804841
|
||||
const expression = /https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/gi;
|
||||
const expression =
|
||||
/https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/gi;
|
||||
const urlRegex = new RegExp(expression);
|
||||
|
||||
const searchEngines = {
|
||||
"g": {
|
||||
action: "https://www.google.com/search",
|
||||
name: "q",
|
||||
},
|
||||
"d": {
|
||||
action: "https://duckduckgo.com/",
|
||||
name: "q",
|
||||
},
|
||||
"y": {
|
||||
action: "https://yandex.com/search/",
|
||||
name: "text",
|
||||
},
|
||||
"lure": {
|
||||
action: "https://lure.sh/pkgs",
|
||||
name: "q",
|
||||
},
|
||||
g: {
|
||||
action: "https://www.google.com/search",
|
||||
name: "q",
|
||||
},
|
||||
d: {
|
||||
action: "https://duckduckgo.com/",
|
||||
name: "q",
|
||||
},
|
||||
y: {
|
||||
action: "https://www.youtube.com/results",
|
||||
name: "search_query",
|
||||
},
|
||||
ya: {
|
||||
action: "https://yandex.com/search/",
|
||||
name: "text",
|
||||
},
|
||||
lure: {
|
||||
action: "https://lure.sh/pkgs",
|
||||
name: "q",
|
||||
},
|
||||
};
|
||||
|
||||
const translationPrefixes = [
|
||||
"t",
|
||||
"translation",
|
||||
]
|
||||
|
||||
const translationPrefixes = ["t", "translation"];
|
||||
|
||||
function getDeepLUrl(s) {
|
||||
const parts = s.split("-")
|
||||
if (parts.length != 3) {
|
||||
return undefined
|
||||
}
|
||||
const parts = s.split("-");
|
||||
if (parts.length != 3) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
return `https://www.deepl.com/en/translator?/#${encodeURIComponent(parts[0].trim())}/${encodeURIComponent(parts[1].trim())}/${encodeURIComponent(parts[2].trim())}`;
|
||||
return `https://www.deepl.com/en/translator?/#${encodeURIComponent(
|
||||
parts[0].trim()
|
||||
)}/${encodeURIComponent(parts[1].trim())}/${encodeURIComponent(
|
||||
parts[2].trim()
|
||||
)}`;
|
||||
}
|
||||
|
||||
form.addEventListener("submit", (event) => {
|
||||
event.preventDefault();
|
||||
|
||||
form.addEventListener("submit", event => {
|
||||
event.preventDefault();
|
||||
s = input.value;
|
||||
|
||||
s = input.value;
|
||||
// check if url
|
||||
if (s.match(urlRegex)) {
|
||||
window.open(s, "_self");
|
||||
return;
|
||||
}
|
||||
|
||||
// check if url
|
||||
if (s.match(urlRegex)) {
|
||||
window.open(s, "_self");
|
||||
return
|
||||
// deepl translations
|
||||
let doTranslation = false;
|
||||
for (const value of translationPrefixes) {
|
||||
const prefix = `!${value} `;
|
||||
if (s.startsWith(prefix)) {
|
||||
doTranslation = true;
|
||||
s = s.slice(prefix.length); // Remove the !{key} prefix
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// deepl translations
|
||||
let doTranslation = false;
|
||||
for (const value of translationPrefixes) {
|
||||
const prefix = `!${value} `;
|
||||
if (s.startsWith(prefix)) {
|
||||
doTranslation = true;
|
||||
s = s.slice(prefix.length); // Remove the !{key} prefix
|
||||
break;
|
||||
}
|
||||
if (doTranslation) {
|
||||
const url = getDeepLUrl(s);
|
||||
if (url) {
|
||||
window.open(url.toString(), "_self");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (doTranslation) {
|
||||
const url = getDeepLUrl(s);
|
||||
if (url) {
|
||||
window.open(url.toString(), "_self");
|
||||
return;
|
||||
}
|
||||
// Check if the string starts with ! followed by a key from searchEngines
|
||||
let selectedEngine = {
|
||||
action: form.getAttribute("action"),
|
||||
name: input.getAttribute("name"),
|
||||
};
|
||||
|
||||
for (const [key, value] of Object.entries(searchEngines)) {
|
||||
const prefix = `!${key} `;
|
||||
if (s.startsWith(prefix)) {
|
||||
selectedEngine = value;
|
||||
s = s.slice(prefix.length); // Remove the !{key} prefix
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Check if the string starts with ! followed by a key from searchEngines
|
||||
let selectedEngine = {
|
||||
action: form.getAttribute("action"),
|
||||
name: input.getAttribute("name"),
|
||||
};
|
||||
|
||||
for (const [key, value] of Object.entries(searchEngines)) {
|
||||
const prefix = `!${key} `;
|
||||
if (s.startsWith(prefix)) {
|
||||
selectedEngine = value;
|
||||
s = s.slice(prefix.length); // Remove the !{key} prefix
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
const url = new URL(selectedEngine.action);
|
||||
url.searchParams.set(selectedEngine.name, s.trim());
|
||||
window.open(url.toString(), "_self");
|
||||
const url = new URL(selectedEngine.action);
|
||||
url.searchParams.set(selectedEngine.name, s.trim());
|
||||
window.open(url.toString(), "_self");
|
||||
});
|
||||
|
||||
2
go.mod
2
go.mod
@@ -5,7 +5,7 @@ go 1.24.2
|
||||
require github.com/labstack/echo/v4 v4.13.4
|
||||
|
||||
require (
|
||||
github.com/kirsle/configdir v0.0.0-20170128060238-e45d2f54772f // indirect
|
||||
github.com/TwiN/go-color v1.4.1 // indirect
|
||||
github.com/labstack/gommon v0.4.2 // indirect
|
||||
github.com/mattn/go-colorable v0.1.14 // indirect
|
||||
github.com/mattn/go-isatty v0.0.20 // indirect
|
||||
|
||||
2
go.sum
2
go.sum
@@ -1,5 +1,7 @@
|
||||
github.com/BurntSushi/toml v1.5.0 h1:W5quZX/G/csjUnuI8SUYlsHs9M38FC7znL0lIO+DvMg=
|
||||
github.com/BurntSushi/toml v1.5.0/go.mod h1:ukJfTF/6rtPPRCnwkur4qwRxa8vTRFBF0uk2lLoLwho=
|
||||
github.com/TwiN/go-color v1.4.1 h1:mqG0P/KBgHKVqmtL5ye7K0/Gr4l6hTksPgTgMk3mUzc=
|
||||
github.com/TwiN/go-color v1.4.1/go.mod h1:WcPf/jtiW95WBIsEeY1Lc/b8aaWoiqQpu5cf8WFxu+s=
|
||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||
github.com/kirsle/configdir v0.0.0-20170128060238-e45d2f54772f h1:dKccXx7xA56UNqOcFIbuqFjAWPVtP688j5QMgmo6OHU=
|
||||
|
||||
98
internal/cache/cache.go
vendored
Normal file
98
internal/cache/cache.go
vendored
Normal file
@@ -0,0 +1,98 @@
|
||||
package cache
|
||||
|
||||
import (
|
||||
"crypto/sha1"
|
||||
"encoding/hex"
|
||||
"errors"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"path/filepath"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/utils"
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
type Cache struct {
|
||||
CacheDir string
|
||||
Disabled bool
|
||||
}
|
||||
|
||||
func getCacheDir() (string, error) {
|
||||
baseDir, err := os.UserCacheDir()
|
||||
if err != nil {
|
||||
baseDir = "/tmp"
|
||||
}
|
||||
cacheDir := filepath.Join(baseDir, utils.Name)
|
||||
err = os.MkdirAll(cacheDir, 0o755)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return cacheDir, nil
|
||||
}
|
||||
|
||||
func getProfileCacheDir(profile string) (string, error) {
|
||||
var profileCacheDir string
|
||||
|
||||
cacheDir, err := getCacheDir()
|
||||
if err != nil {
|
||||
return profileCacheDir, err
|
||||
}
|
||||
|
||||
profileCacheDir = filepath.Join(cacheDir, profile)
|
||||
err = os.MkdirAll(cacheDir, 0o755)
|
||||
return profileCacheDir, err
|
||||
}
|
||||
|
||||
func NewCache(profile string) Cache {
|
||||
cacheDir, err := getProfileCacheDir(profile)
|
||||
|
||||
return Cache{
|
||||
CacheDir: cacheDir,
|
||||
Disabled: err != nil,
|
||||
}
|
||||
}
|
||||
|
||||
const baseCacheUrl = "cache"
|
||||
|
||||
func (c Cache) StartStaticServer(e *echo.Echo) error {
|
||||
e.Static("/"+baseCacheUrl, c.CacheDir)
|
||||
return nil
|
||||
}
|
||||
|
||||
func hashUrl(url string) string {
|
||||
h := sha1.New()
|
||||
io.WriteString(h, url)
|
||||
return hex.EncodeToString(h.Sum(nil))
|
||||
}
|
||||
|
||||
func (c Cache) CacheUrl(urlString string) (string, error) {
|
||||
filename := hashUrl(urlString) + filepath.Ext(urlString)
|
||||
targetPath := filepath.Join(c.CacheDir, filename)
|
||||
|
||||
// if the file was already downloaded it doesn't need to be downloaded again
|
||||
if _, err := os.Stat(targetPath); errors.Is(err, os.ErrNotExist) {
|
||||
resp, err := http.Get(urlString)
|
||||
if !errors.Is(err, os.ErrNotExist) {
|
||||
return urlString, err
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
file, err := os.Create(targetPath)
|
||||
if err != nil {
|
||||
return urlString, err
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
_, err = io.Copy(file, resp.Body)
|
||||
|
||||
if err != nil {
|
||||
return urlString, err
|
||||
}
|
||||
} else {
|
||||
return url.JoinPath(baseCacheUrl, filename)
|
||||
}
|
||||
|
||||
return url.JoinPath(baseCacheUrl, filename)
|
||||
}
|
||||
9
internal/cli/cache.go
Normal file
9
internal/cli/cache.go
Normal file
@@ -0,0 +1,9 @@
|
||||
package cli
|
||||
|
||||
import "log"
|
||||
|
||||
func Cache() error {
|
||||
log.Println("running cache")
|
||||
log.Panicln("not implemented yet")
|
||||
return nil
|
||||
}
|
||||
102
internal/cli/cli.go
Normal file
102
internal/cli/cli.go
Normal file
@@ -0,0 +1,102 @@
|
||||
package cli
|
||||
|
||||
import (
|
||||
"log"
|
||||
"os"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/utils"
|
||||
"github.com/TwiN/go-color"
|
||||
)
|
||||
|
||||
type ProgramFunction func() error
|
||||
type Program struct {
|
||||
Name string
|
||||
Function ProgramFunction
|
||||
ShortDescription string
|
||||
LongDescription string
|
||||
Arguments []Argument
|
||||
}
|
||||
type Argument struct {
|
||||
Name string
|
||||
Type string
|
||||
Required bool
|
||||
Description string
|
||||
}
|
||||
|
||||
var HelpHeader = `This is the help page of ` + utils.Name + `.
|
||||
` + color.Purple + utils.BinaryName + ` {program} {...args}` + color.Reset + `
|
||||
The following Programs are available:`
|
||||
var Programs = []Program{
|
||||
{
|
||||
Name: "help",
|
||||
ShortDescription: "get more information on how the cli in general or a specific program works",
|
||||
LongDescription: "What did you expect to find here?",
|
||||
Arguments: []Argument{
|
||||
{
|
||||
Name: "program",
|
||||
Type: "string",
|
||||
Required: false,
|
||||
Description: "defines the program you want to know more about",
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
Name: "start",
|
||||
Function: Start,
|
||||
ShortDescription: "start the webserver",
|
||||
LongDescription: `The start program starts the webserver.
|
||||
It loads the config file of the according profile.
|
||||
It uses the default values if no config file was found.`,
|
||||
Arguments: []Argument{
|
||||
{
|
||||
Name: "profile",
|
||||
Type: "string",
|
||||
Required: false,
|
||||
Description: "tells the program which config to load, default is 'default'",
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
Name: "cache",
|
||||
Function: Cache,
|
||||
ShortDescription: "do something with the cache",
|
||||
LongDescription: `Does something with the cache.
|
||||
- clear: delete the whole cache
|
||||
- clean: delete all files that aren't used by any program.`,
|
||||
Arguments: []Argument{
|
||||
{
|
||||
Name: "action",
|
||||
Type: "enum(clear;clean)",
|
||||
Required: true,
|
||||
Description: "defines what to do with the cache",
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
func GetProgram(programName string) Program {
|
||||
for i, p := range Programs {
|
||||
if p.Name == programName {
|
||||
return Programs[i]
|
||||
}
|
||||
}
|
||||
|
||||
log.Panicln("couldn't find program", programName, ". EXITING")
|
||||
return Program{}
|
||||
}
|
||||
|
||||
func Cli() {
|
||||
// getting around initialization cycle
|
||||
Programs[0].Function = Help
|
||||
|
||||
programName := "help"
|
||||
if len(os.Args) > 1 {
|
||||
programName = os.Args[1]
|
||||
}
|
||||
|
||||
var selectedProgram Program = GetProgram(programName)
|
||||
err := selectedProgram.Function()
|
||||
if err != nil {
|
||||
log.Panicln(err)
|
||||
}
|
||||
}
|
||||
90
internal/cli/help.go
Normal file
90
internal/cli/help.go
Normal file
@@ -0,0 +1,90 @@
|
||||
package cli
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"strings"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/utils"
|
||||
"github.com/TwiN/go-color"
|
||||
)
|
||||
|
||||
func padString(s string, n int) string {
|
||||
missing := n - len(s)
|
||||
if missing <= 0 {
|
||||
return s
|
||||
}
|
||||
|
||||
for _ = range missing {
|
||||
s = s + " "
|
||||
}
|
||||
|
||||
return s
|
||||
}
|
||||
|
||||
func getSingleArgumentString(a Argument) string {
|
||||
requiredString := ""
|
||||
if a.Required {
|
||||
requiredString = "*"
|
||||
}
|
||||
return requiredString + a.Name + ":" + a.Type
|
||||
}
|
||||
|
||||
func getArgumentString(arguments []Argument) string {
|
||||
argumentString := color.Blue
|
||||
for _, a := range arguments {
|
||||
argumentString = argumentString + " [" + getSingleArgumentString(a) + "]"
|
||||
}
|
||||
return argumentString + color.Reset
|
||||
}
|
||||
|
||||
func generalHelp() error {
|
||||
fmt.Println()
|
||||
fmt.Println(HelpHeader)
|
||||
fmt.Println()
|
||||
|
||||
for _, p := range Programs {
|
||||
fmt.Print(color.Bold + padString(p.Name, 7) + color.Reset)
|
||||
fmt.Print(padString(getArgumentString(p.Arguments), 40) + p.ShortDescription + "\n")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func specificHelp(programName string) error {
|
||||
program := GetProgram(programName)
|
||||
|
||||
fmt.Println(color.Bold + "MAN PAGE FOR " + strings.ToUpper(programName) + color.Reset)
|
||||
fmt.Println()
|
||||
|
||||
fmt.Println(color.Purple + utils.BinaryName + " " + programName + color.Reset + getArgumentString(program.Arguments))
|
||||
fmt.Println()
|
||||
|
||||
fmt.Println(color.Bold + "arguments" + color.Reset)
|
||||
|
||||
argumentStrings := make([]string, len(program.Arguments))
|
||||
maxArgumentString := 0
|
||||
for i, a := range program.Arguments {
|
||||
s := getSingleArgumentString(a)
|
||||
argumentStrings[i] = s
|
||||
if len(s) > maxArgumentString {
|
||||
maxArgumentString = len(s)
|
||||
}
|
||||
}
|
||||
|
||||
for i, a := range program.Arguments {
|
||||
fmt.Println(padString(argumentStrings[i], maxArgumentString+4) + a.Description)
|
||||
}
|
||||
|
||||
fmt.Println()
|
||||
fmt.Println(program.LongDescription)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func Help() error {
|
||||
if len(os.Args) > 2 {
|
||||
return specificHelp(os.Args[2])
|
||||
}
|
||||
|
||||
return generalHelp()
|
||||
}
|
||||
18
internal/cli/start.go
Normal file
18
internal/cli/start.go
Normal file
@@ -0,0 +1,18 @@
|
||||
package cli
|
||||
|
||||
import (
|
||||
"log"
|
||||
"os"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/server"
|
||||
)
|
||||
|
||||
func Start() error {
|
||||
profile := "default"
|
||||
if len(os.Args) > 2 {
|
||||
profile = os.Args[2]
|
||||
}
|
||||
log.Println("starting server with profile " + profile)
|
||||
|
||||
return server.Start(profile)
|
||||
}
|
||||
9
internal/diyhrt/diy_config.go
Normal file
9
internal/diyhrt/diy_config.go
Normal file
@@ -0,0 +1,9 @@
|
||||
package diyhrt
|
||||
|
||||
type DiyHrtConfig struct {
|
||||
ApiKey string
|
||||
FetchIntervals int
|
||||
|
||||
StoreFilter StoreFilter
|
||||
ListingFilter ListingFilter
|
||||
}
|
||||
@@ -4,16 +4,14 @@ import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"net/http"
|
||||
"os"
|
||||
"time"
|
||||
)
|
||||
|
||||
const endpoint = "https://diyhrt.market/api/listings"
|
||||
|
||||
func GetListings() ([]Listing, error) {
|
||||
apiKey := os.Getenv("API_KEY")
|
||||
func GetListings(apiKey string) ([]Listing, error) {
|
||||
if apiKey == "" {
|
||||
return nil, errors.New("API_KEY environment variable not set")
|
||||
return nil, errors.New("diyhrt API_KEY key not set. Set it as env or in DiyHrt.ApiKey")
|
||||
}
|
||||
|
||||
// Create HTTP client
|
||||
|
||||
@@ -2,14 +2,12 @@ package rendering
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"maps"
|
||||
"log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"slices"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/diyhrt"
|
||||
"github.com/kirsle/configdir"
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/utils"
|
||||
"github.com/pelletier/go-toml"
|
||||
)
|
||||
|
||||
@@ -21,6 +19,15 @@ const (
|
||||
Websites ActiveCard = "websites"
|
||||
)
|
||||
|
||||
type AnchorTarget string
|
||||
|
||||
const (
|
||||
OpenInNewTab AnchorTarget = "_blank"
|
||||
OpenInCurrentTab AnchorTarget = "_self"
|
||||
OpenInParent AnchorTarget = "_parent"
|
||||
OpenInTopWindow AnchorTarget = "_top"
|
||||
)
|
||||
|
||||
type ServerConfig struct {
|
||||
Port int
|
||||
}
|
||||
@@ -34,20 +41,20 @@ type TemplateConfig struct {
|
||||
SearchFormAction string
|
||||
SearchInputName string
|
||||
|
||||
StoreFilter diyhrt.StoreFilter
|
||||
ListingFilter diyhrt.ListingFilter
|
||||
|
||||
Listings []diyhrt.Listing
|
||||
Stores []diyhrt.Store
|
||||
Listings []diyhrt.Listing
|
||||
Stores []diyhrt.Store
|
||||
DiyHrtTarget AnchorTarget
|
||||
|
||||
ActiveCard ActiveCard
|
||||
|
||||
Websites []Website
|
||||
Websites []Website
|
||||
WebsiteTarget AnchorTarget
|
||||
}
|
||||
|
||||
type Config struct {
|
||||
Server ServerConfig
|
||||
Template TemplateConfig
|
||||
DiyHrt diyhrt.DiyHrtConfig
|
||||
}
|
||||
|
||||
func NewConfig() Config {
|
||||
@@ -55,6 +62,17 @@ func NewConfig() Config {
|
||||
Server: ServerConfig{
|
||||
Port: 5500,
|
||||
},
|
||||
DiyHrt: diyhrt.DiyHrtConfig{
|
||||
ApiKey: os.Getenv("API_KEY"),
|
||||
FetchIntervals: 60, // fetch every hour
|
||||
StoreFilter: diyhrt.StoreFilter{
|
||||
Limit: 0,
|
||||
IncludeIds: []int{7},
|
||||
},
|
||||
ListingFilter: diyhrt.ListingFilter{
|
||||
FromStores: []int{7},
|
||||
},
|
||||
},
|
||||
Template: TemplateConfig{
|
||||
HeaderPhrases: []string{
|
||||
"GirlJuice.Inject();",
|
||||
@@ -74,41 +92,26 @@ func NewConfig() Config {
|
||||
|
||||
ActiveCard: DiyHrtListings,
|
||||
|
||||
StoreFilter: diyhrt.StoreFilter{
|
||||
Limit: 0,
|
||||
IncludeIds: []int{7},
|
||||
},
|
||||
|
||||
ListingFilter: diyhrt.ListingFilter{
|
||||
FromStores: []int{7},
|
||||
},
|
||||
DiyHrtTarget: OpenInCurrentTab,
|
||||
|
||||
Websites: []Website{
|
||||
{Url: "https://gitea.elara.ws/Hazel/transfem-startpage", Name: "Transfem Startpage", ImageUrl: "https://gitea.elara.ws/assets/img/logo.svg"},
|
||||
},
|
||||
WebsiteTarget: OpenInCurrentTab,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func (rc *TemplateConfig) LoadDiyHrt(listings []diyhrt.Listing) {
|
||||
existingStores := make(map[int]diyhrt.Store)
|
||||
|
||||
for _, listing := range listings {
|
||||
existingStores[listing.Store.Id] = listing.Store
|
||||
}
|
||||
|
||||
rc.Listings = rc.ListingFilter.Filter(listings)
|
||||
rc.Stores = rc.StoreFilter.Filter(slices.Collect(maps.Values(existingStores)))
|
||||
}
|
||||
|
||||
func (rc *Config) ScanForConfigFile(profile string) error {
|
||||
profileFile := profile + ".toml"
|
||||
|
||||
configPath := configdir.LocalConfig("startpage")
|
||||
configFile := filepath.Join(configPath, profileFile)
|
||||
baseDir, cacheDirErr := os.UserConfigDir()
|
||||
if cacheDirErr == nil {
|
||||
configFile := filepath.Join(baseDir, utils.Name, profileFile)
|
||||
|
||||
if err := rc.LoadConfigFile(configFile); !errors.Is(err, os.ErrNotExist) {
|
||||
return err
|
||||
if err := rc.LoadConfigFile(configFile); !errors.Is(err, os.ErrNotExist) {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
if err := rc.LoadConfigFile(profileFile); !errors.Is(err, os.ErrNotExist) {
|
||||
@@ -119,7 +122,7 @@ func (rc *Config) ScanForConfigFile(profile string) error {
|
||||
return err
|
||||
}
|
||||
|
||||
return errors.New("No config file found")
|
||||
return errors.New("no config file found")
|
||||
}
|
||||
|
||||
func (rc *Config) LoadConfigFile(file string) error {
|
||||
@@ -127,7 +130,7 @@ func (rc *Config) LoadConfigFile(file string) error {
|
||||
return err
|
||||
}
|
||||
|
||||
fmt.Println("loading config file: " + file)
|
||||
log.Println("loading config file", file)
|
||||
|
||||
content, err := os.ReadFile(file)
|
||||
|
||||
|
||||
28
internal/rendering/diyhrt.go
Normal file
28
internal/rendering/diyhrt.go
Normal file
@@ -0,0 +1,28 @@
|
||||
package rendering
|
||||
|
||||
import (
|
||||
"maps"
|
||||
"slices"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/diyhrt"
|
||||
)
|
||||
|
||||
func (c *Config) LoadDiyHrt(listings []diyhrt.Listing) {
|
||||
existingStores := make(map[int]diyhrt.Store)
|
||||
|
||||
for _, listing := range listings {
|
||||
existingStores[listing.Store.Id] = listing.Store
|
||||
}
|
||||
|
||||
c.Template.Listings = c.DiyHrt.ListingFilter.Filter(listings)
|
||||
c.Template.Stores = c.DiyHrt.StoreFilter.Filter(slices.Collect(maps.Values(existingStores)))
|
||||
}
|
||||
|
||||
func (c *Config) FetchDiyHrt() error {
|
||||
l, err := diyhrt.GetListings(c.DiyHrt.ApiKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
c.LoadDiyHrt(l)
|
||||
return nil
|
||||
}
|
||||
@@ -1,7 +1,80 @@
|
||||
package rendering
|
||||
|
||||
import (
|
||||
"crypto/sha1"
|
||||
"encoding/hex"
|
||||
"errors"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
type Website struct {
|
||||
Url string
|
||||
Name string
|
||||
ImageUrl string
|
||||
Url string
|
||||
Name string
|
||||
ImageUrl string
|
||||
IsFetched bool
|
||||
}
|
||||
|
||||
const CacheUrl = "cache"
|
||||
|
||||
func GetCacheDir() (string, error) {
|
||||
baseDir, err := os.UserCacheDir()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
cacheDir := filepath.Join(baseDir, "startpage")
|
||||
err = os.MkdirAll(cacheDir, 0o755)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return cacheDir, nil
|
||||
}
|
||||
|
||||
func hashUrl(url string) string {
|
||||
h := sha1.New()
|
||||
io.WriteString(h, url)
|
||||
return hex.EncodeToString(h.Sum(nil))
|
||||
}
|
||||
|
||||
func (w *Website) Cache() error {
|
||||
if w.IsFetched {
|
||||
return nil
|
||||
}
|
||||
|
||||
cacheDir, err := GetCacheDir()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
filename := hashUrl(w.ImageUrl) + filepath.Ext(w.ImageUrl)
|
||||
targetPath := filepath.Join(cacheDir, filename)
|
||||
|
||||
// if the file was already downloaded it doesn't need to be downloaded again
|
||||
if _, err := os.Stat(targetPath); errors.Is(err, os.ErrNotExist) {
|
||||
resp, err := http.Get(w.ImageUrl)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
file, err := os.Create(targetPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
_, err = io.Copy(file, resp.Body)
|
||||
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
// set the value in the struct to the current file
|
||||
w.ImageUrl, _ = url.JoinPath(CacheUrl, filename)
|
||||
w.IsFetched = true
|
||||
return nil
|
||||
}
|
||||
|
||||
42
internal/server/embed.go
Normal file
42
internal/server/embed.go
Normal file
@@ -0,0 +1,42 @@
|
||||
package server
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"embed"
|
||||
"io/fs"
|
||||
"log"
|
||||
"net/http"
|
||||
"text/template"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
var FrontendFiles embed.FS
|
||||
|
||||
func getFileContent() string {
|
||||
content, err := FrontendFiles.ReadFile("frontend/index.html")
|
||||
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
return string(content)
|
||||
}
|
||||
|
||||
func getIndex(c echo.Context) error {
|
||||
IndexTemplate := template.Must(template.New("index").Parse(getFileContent()))
|
||||
|
||||
var tpl bytes.Buffer
|
||||
IndexTemplate.Execute(&tpl, Config.Template)
|
||||
|
||||
return c.HTML(http.StatusOK, tpl.String())
|
||||
}
|
||||
|
||||
func getFileSystem() http.FileSystem {
|
||||
fsys, err := fs.Sub(FrontendFiles, "frontend")
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
return http.FS(fsys)
|
||||
}
|
||||
67
internal/server/server.go
Normal file
67
internal/server/server.go
Normal file
@@ -0,0 +1,67 @@
|
||||
package server
|
||||
|
||||
import (
|
||||
"log"
|
||||
"net/http"
|
||||
"strconv"
|
||||
"time"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/cache"
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/rendering"
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
var Config = rendering.NewConfig()
|
||||
|
||||
func StartFetching() {
|
||||
for {
|
||||
log.Println("Fetch DiyHrt data...")
|
||||
Config.FetchDiyHrt()
|
||||
time.Sleep(time.Duration(Config.DiyHrt.FetchIntervals) * time.Second)
|
||||
|
||||
if Config.DiyHrt.FetchIntervals == 0 {
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func Start(profile string) error {
|
||||
err := Config.ScanForConfigFile(profile)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
go StartFetching()
|
||||
|
||||
err = Config.FetchDiyHrt()
|
||||
if err != nil {
|
||||
log.Println(err)
|
||||
}
|
||||
|
||||
e := echo.New()
|
||||
|
||||
// statically serve the file
|
||||
cache := cache.NewCache(profile)
|
||||
if !cache.Disabled {
|
||||
cache.StartStaticServer(e)
|
||||
|
||||
log.Println("downloading website icons...")
|
||||
for i, w := range Config.Template.Websites {
|
||||
u, err := cache.CacheUrl(w.ImageUrl)
|
||||
if err != nil {
|
||||
log.Println(err)
|
||||
}
|
||||
Config.Template.Websites[i].ImageUrl = u
|
||||
Config.Template.Websites[i].IsFetched = true
|
||||
}
|
||||
}
|
||||
|
||||
// https://echo.labstack.com/docs/cookbook/embed-resources
|
||||
staticHandler := http.FileServer(getFileSystem())
|
||||
e.GET("/assets/*", echo.WrapHandler(http.StripPrefix("/", staticHandler)))
|
||||
e.GET("/scripts/*", echo.WrapHandler(http.StripPrefix("/", staticHandler)))
|
||||
e.GET("/", getIndex)
|
||||
|
||||
e.Logger.Fatal(e.Start(":" + strconv.Itoa(Config.Server.Port)))
|
||||
return nil
|
||||
}
|
||||
6
internal/utils/meta.go
Normal file
6
internal/utils/meta.go
Normal file
@@ -0,0 +1,6 @@
|
||||
package utils
|
||||
|
||||
import "os"
|
||||
|
||||
var Name = "transfem-startpage"
|
||||
var BinaryName = os.Args[0]
|
||||
85
main.go
85
main.go
@@ -1,91 +1,16 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"embed"
|
||||
"fmt"
|
||||
"html/template"
|
||||
"io/fs"
|
||||
"log"
|
||||
"net/http"
|
||||
"os"
|
||||
"strconv"
|
||||
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/diyhrt"
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/rendering"
|
||||
"github.com/labstack/echo/v4"
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/cli"
|
||||
"gitea.elara.ws/Hazel/transfem-startpage/internal/server"
|
||||
)
|
||||
|
||||
var CurrentConfig = rendering.NewConfig()
|
||||
|
||||
func FetchDiyHrt() error {
|
||||
fmt.Println("Fetch DiyHrt Marketplaces...")
|
||||
|
||||
l, err := diyhrt.GetListings()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
CurrentConfig.Template.LoadDiyHrt(l)
|
||||
return nil
|
||||
}
|
||||
|
||||
//go:embed frontend/*
|
||||
var frontendFiles embed.FS
|
||||
|
||||
func getFileContent() string {
|
||||
content, err := frontendFiles.ReadFile("frontend/index.html")
|
||||
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
return string(content)
|
||||
}
|
||||
|
||||
var IndexTemplate = template.Must(template.New("index").Parse(getFileContent()))
|
||||
|
||||
func getIndex(c echo.Context) error {
|
||||
var tpl bytes.Buffer
|
||||
IndexTemplate.Execute(&tpl, CurrentConfig.Template)
|
||||
|
||||
return c.HTML(http.StatusOK, tpl.String())
|
||||
}
|
||||
|
||||
func getFileSystem() http.FileSystem {
|
||||
fsys, err := fs.Sub(frontendFiles, "frontend")
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
return http.FS(fsys)
|
||||
}
|
||||
var FrontendFiles embed.FS
|
||||
|
||||
func main() {
|
||||
fmt.Println("running transfem startpage")
|
||||
|
||||
profile := "startpage"
|
||||
if len(os.Args) > 1 {
|
||||
profile = os.Args[1]
|
||||
}
|
||||
fmt.Println("loading profile " + profile + "...")
|
||||
|
||||
err := CurrentConfig.ScanForConfigFile(profile)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
|
||||
err = FetchDiyHrt()
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
|
||||
e := echo.New()
|
||||
|
||||
// https://echo.labstack.com/docs/cookbook/embed-resources
|
||||
staticHandler := http.FileServer(getFileSystem())
|
||||
e.GET("/assets/*", echo.WrapHandler(http.StripPrefix("/", staticHandler)))
|
||||
e.GET("/scripts/*", echo.WrapHandler(http.StripPrefix("/", staticHandler)))
|
||||
e.GET("/", getIndex)
|
||||
|
||||
e.Logger.Fatal(e.Start(":" + strconv.Itoa(CurrentConfig.Server.Port)))
|
||||
server.FrontendFiles = FrontendFiles
|
||||
cli.Cli()
|
||||
}
|
||||
|
||||
1
tmp/build-errors.log
Normal file
1
tmp/build-errors.log
Normal file
@@ -0,0 +1 @@
|
||||
exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1exit status 1
|
||||
Reference in New Issue
Block a user