-/*
-linksexp -- Texinfo/XBEL/OPML 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 *doURLS {
+ for {
+ m, err := r.NextMapWithSlice()
+ if err == io.EOF {
+ break
+ }
+ if err != nil {
+ panic(err)
+ }
+ if m["%rec"] != nil {
+ continue
+ }
+ 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)
+ }
+ }
+ os.Exit(0)
+ }
+
if *doOPML {
data := make([]map[string][]string, 0)
for {
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("@table @strong")
+
+ fmt.Println("@menu")
+ fmt.Println("Categories:")
+ for _, cat := range cats {
+ fmt.Printf("* %s (%d items): LinksCat%s\n", cat, len(data[cat]), cat)
+ }
+ fmt.Println("@end menu")
+
for _, cat := range cats {
- fmt.Println("@item", cat)
+ fmt.Println("@node", "LinksCat"+cat)
+ fmt.Println("@section Links category:", cat)
fmt.Println("@multitable @columnfractions .05 .8 .1 .05")
+ fmt.Println("@headitem @tab @tab Other categories @tab Feed URLs")
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 = "(" + ent["Note"][0] + ")"
+ 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,
+ " @item %d @tab @url{%s,, %s} %s @tab %s @tab\n",
+ 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:
- fmt.Printf(" @url{%s, feed}\n", feeds[0])
+ fmt.Printf(
+ " @url{%s, feed}\n",
+ strings.ReplaceAll(feeds[0], "@", "@@"),
+ )
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,
+ )
}
}
}
fmt.Println("@end multitable")
}
- fmt.Println("@end table")
}