Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| be10f5238e | |||
| 78488d2f41 | |||
| 7399fbc5ff | |||
| f62499f6b5 | |||
| 16006c1f1b | |||
| 92c293205d | |||
| 6473b48a5b | |||
| 385397c980 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -1,4 +1,3 @@
|
|||||||
badger/
|
badger/
|
||||||
dist/
|
dist/
|
||||||
tmp/
|
tmp/
|
||||||
linkwallet
|
|
||||||
|
|||||||
@@ -11,11 +11,15 @@ builds:
|
|||||||
- windows
|
- windows
|
||||||
- darwin
|
- darwin
|
||||||
dockers:
|
dockers:
|
||||||
- image_templates:
|
- goos: linux
|
||||||
|
goarch: amd64
|
||||||
|
image_templates:
|
||||||
- "tardisx/linkwallet:{{ .Tag }}"
|
- "tardisx/linkwallet:{{ .Tag }}"
|
||||||
- "tardisx/linkwallet:v{{ .Major }}"
|
- "tardisx/linkwallet:v{{ .Major }}"
|
||||||
- "tardisx/linkwallet:v{{ .Major }}.{{ .Minor }}"
|
- "tardisx/linkwallet:v{{ .Major }}.{{ .Minor }}"
|
||||||
- "tardisx/linkwallet"
|
- "tardisx/linkwallet"
|
||||||
|
build_flag_templates:
|
||||||
|
- "--platform=linux/amd64"
|
||||||
archives:
|
archives:
|
||||||
- replacements:
|
- replacements:
|
||||||
darwin: Darwin
|
darwin: Darwin
|
||||||
@@ -33,3 +37,4 @@ changelog:
|
|||||||
exclude:
|
exclude:
|
||||||
- '^docs:'
|
- '^docs:'
|
||||||
- '^test:'
|
- '^test:'
|
||||||
|
- '^[Bb]ump'
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
FROM scratch
|
FROM alpine:3.16
|
||||||
ENTRYPOINT ["/linkwallet"]
|
ENTRYPOINT ["/linkwallet"]
|
||||||
COPY linkwallet /
|
COPY linkwallet /
|
||||||
19
README.md
19
README.md
@@ -1 +1,20 @@
|
|||||||
# linkwallet
|
# linkwallet
|
||||||
|
|
||||||
|
A self-hosted bookmark database with full-text page content search.
|
||||||
|
|
||||||
|
# Feature list
|
||||||
|
|
||||||
|
* Simple cross-platform single binary deployment
|
||||||
|
* or docker if you prefer
|
||||||
|
* Full-text search
|
||||||
|
* Bookmark content is scraped and indexed locally
|
||||||
|
* Page content periodically refreshed automatically
|
||||||
|
* Interactively search across titles and content
|
||||||
|
* Rippingly fast results, as you type
|
||||||
|
* Embedded database, no separate database system required
|
||||||
|
* Light on resources
|
||||||
|
|
||||||
|
# Roadmap
|
||||||
|
|
||||||
|
* Bookmarklet
|
||||||
|
* Tags
|
||||||
|
|||||||
25
cmd/linkwallet/linkwallet.go
Normal file
25
cmd/linkwallet/linkwallet.go
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
|
||||||
|
"github.com/tardisx/linkwallet/db"
|
||||||
|
"github.com/tardisx/linkwallet/version"
|
||||||
|
"github.com/tardisx/linkwallet/web"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
|
||||||
|
dbh := db.DB{}
|
||||||
|
dbh.Open("badger")
|
||||||
|
bmm := db.NewBookmarkManager(&dbh)
|
||||||
|
|
||||||
|
go func() { version.UpdateVersionInfo() }()
|
||||||
|
|
||||||
|
log.Printf("linkallet verison %s starting", version.Is())
|
||||||
|
|
||||||
|
server := web.Create(bmm)
|
||||||
|
go bmm.RunQueue()
|
||||||
|
go bmm.UpdateContent()
|
||||||
|
server.Start()
|
||||||
|
}
|
||||||
@@ -2,6 +2,7 @@ package db
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
@@ -48,6 +49,19 @@ func (m *BookmarkManager) ListBookmarks() ([]entity.Bookmark, error) {
|
|||||||
return bookmarks, nil
|
return bookmarks, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ExportBookmarks exports all bookmarks to an io.Writer
|
||||||
|
func (m *BookmarkManager) ExportBookmarks(w io.Writer) error {
|
||||||
|
bms := []entity.Bookmark{}
|
||||||
|
err := m.db.store.Find(&bms, &badgerhold.Query{})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("could not export bookmarks: %w", err)
|
||||||
|
}
|
||||||
|
for _, bm := range bms {
|
||||||
|
w.Write([]byte(bm.URL + "\n"))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (m *BookmarkManager) SaveBookmark(bm *entity.Bookmark) error {
|
func (m *BookmarkManager) SaveBookmark(bm *entity.Bookmark) error {
|
||||||
err := m.db.store.Update(bm.ID, &bm)
|
err := m.db.store.Update(bm.ID, &bm)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import (
|
|||||||
"golang.org/x/mod/semver"
|
"golang.org/x/mod/semver"
|
||||||
)
|
)
|
||||||
|
|
||||||
const Tag = "v0.0.3"
|
const Tag = "v0.0.7"
|
||||||
|
|
||||||
var versionInfo struct {
|
var versionInfo struct {
|
||||||
Local struct {
|
Local struct {
|
||||||
|
|||||||
@@ -22,6 +22,7 @@
|
|||||||
<a href="#">Admin</a>
|
<a href="#">Admin</a>
|
||||||
<ul class="menu vertical">
|
<ul class="menu vertical">
|
||||||
<li><a href="/manage">Manage links</a></li>
|
<li><a href="/manage">Manage links</a></li>
|
||||||
|
<li><a href="/export">Export all URLs</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|||||||
10
web/web.go
10
web/web.go
@@ -189,6 +189,16 @@ func Create(bmm *db.BookmarkManager) *Server {
|
|||||||
c.String(http.StatusOK, "queued")
|
c.String(http.StatusOK, "queued")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
r.GET("/export", func(c *gin.Context) {
|
||||||
|
c.Writer.Header().Set("Content-Type", "text/plain")
|
||||||
|
c.Writer.Header().Set("Content-Disposition", "attachment; filename=\"bookmarks.txt\"")
|
||||||
|
err := bmm.ExportBookmarks(c.Writer)
|
||||||
|
// this is a bit late, but we already added headers, so at least log it.
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("got error when exporting: %s", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
return server
|
return server
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user