-/*
-linksexp -- Texinfo/XBEL/OPML/urls autogeneration from recfile bookmark
-Copyright (C) 2021 Sergey Matveev <stargrave@stargrave.org>
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU General Public License as published by
-the Free Software Foundation, version 3 of the License.
-
-This program is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with this program. If not, see <http://www.gnu.org/licenses/>.
-*/
+// linksexp -- Texinfo/XBEL/OPML/SWG/urls autogeneration from recfile bookmark
+// Copyright (C) 2021-2025 Sergey Matveev <stargrave@stargrave.org>
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, version 3 of the License.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
package main
"strings"
"time"
- "go.cypherpunks.ru/recfile"
+ "go.cypherpunks.su/recfile/v2"
)
type ByTitle []map[string][]string
func (a ByTitle) Len() int { return len(a) }
func (a ByTitle) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
func (a ByTitle) Less(i, j int) bool {
- return strings.Compare(a[i]["Title"][0], a[j]["Title"][0]) < 0
+ return strings.Compare(a[i]["title"][0], a[j]["title"][0]) < 0
}
func main() {
doXBEL := flag.Bool("xbel", false, "Make XBEL")
doOPML := flag.Bool("opml", false, "Make OPML")
doURLS := flag.Bool("urls", false, "Make newsboat urls")
+ doSWG := flag.Bool("swg", false, "Make SWG txtar")
flag.Parse()
r := recfile.NewReader(os.Stdin)
if m["%rec"] != nil {
continue
}
- sort.Strings(m["Category"])
- cats := strings.Join(m["Category"], " ")
- for _, f := range m["Feed"] {
+ sort.Strings(m["cat"])
+ cats := strings.Join(m["cat"], " ")
+ for _, f := range m["feed"] {
+ if strings.HasPrefix(f, "gemini://") {
+ f = "https://gemini/" + f
+ }
fmt.Println(f, cats)
}
}
if m["%rec"] != nil {
continue
}
- sort.Strings(m["Category"])
+ sort.Strings(m["cat"])
data = append(data, m)
}
sort.Sort(ByTitle(data))
if m["%rec"] != nil {
continue
}
- sort.Strings(m["Category"])
- if cs := m["Category"]; len(cs) == 0 {
+ sort.Strings(m["cat"])
+ if cs := m["cat"]; len(cs) == 0 {
data["Uncategorized"] = append(data["Uncategorized"], m)
} else {
for _, cat := range cs {
os.Exit(0)
}
+ if *doSWG {
+ swg(cats, data)
+ os.Exit(0)
+ }
+
fmt.Println("Updated:", time.Now().Format(time.RFC3339))
fmt.Println("@menu")
ents := data[cat]
for n, ent := range ents {
catsOther := make([]string, 0)
- for _, c := range ent["Category"] {
+ for _, c := range ent["cat"] {
if c != cat {
catsOther = append(catsOther, c)
}
}
var note string
- if len(ent["Note"]) > 0 {
- note = "(" + strings.Trim(ent["Note"][0], " \n") + ")"
+ if len(ent["note"]) > 0 {
+ note = "(" + strings.Trim(ent["note"][0], " \n") + ")"
}
fmt.Printf(
" @item %d @tab @url{%s,, %s} %s @tab %s @tab\n",
- n, ent["URL"][0], ent["Title"][0], note,
+ n,
+ strings.ReplaceAll(ent["url"][0], "@", "@@"),
+ strings.ReplaceAll(ent["title"][0], "@", "@@"),
+ note,
strings.Join(catsOther, ", "),
)
- switch feeds := ent["Feed"]; len(feeds) {
+ switch feeds := ent["feed"]; len(feeds) {
case 0:
fmt.Printf(" @emph{STATIC}\n")
case 1:
)
default:
for i, feed := range feeds {
- fmt.Printf(" @url{%s, feed%d}\n", feed, i)
+ fmt.Printf(
+ " @url{%s, feed%d}\n",
+ strings.ReplaceAll(feed, "@", "@@"), i,
+ )
}
}
}