import opml

This commit is contained in:
Nazar Kanaev 2020-07-12 16:38:27 +01:00
parent 9583b4b8d7
commit ea4775839f
4 changed files with 67 additions and 9 deletions

View File

@ -5,16 +5,15 @@ import (
"github.com/mmcdole/gofeed" "github.com/mmcdole/gofeed"
"net/http" "net/http"
"encoding/json" "encoding/json"
"encoding/xml"
"os" "os"
"log" "log"
"io" "io"
"io/ioutil"
"mime" "mime"
"strings" "strings"
"path/filepath" "path/filepath"
"strconv" "strconv"
"math" "math"
"fmt"
) )
func IndexHandler(rw http.ResponseWriter, req *http.Request) { func IndexHandler(rw http.ResponseWriter, req *http.Request) {
@ -351,6 +350,33 @@ func SettingsHandler(rw http.ResponseWriter, req *http.Request) {
} }
} }
type opml struct {
XMLName xml.Name `xml:"opml"`
Version string `xml:"version,attr"`
Outlines []outline `xml:"body>outline"`
}
type outline struct {
Type string `xml:"type,attr,omitempty"`
Title string `xml:"text,attr"`
FeedURL string `xml:"xmlUrl,attr,omitempty"`
SiteURL string `xml:"htmlUrl,attr,omitempty"`
Description string `xml:"description,attr,omitempty"`
Outlines []outline `xml:"outline,omitempty"`
}
func (o outline) AllFeeds() []outline {
result := make([]outline, 0)
for _, sub := range o.Outlines {
if sub.Type == "rss" {
result = append(result, sub)
} else {
result = append(result, sub.AllFeeds()...)
}
}
return result
}
func OPMLImportHandler(rw http.ResponseWriter, req *http.Request) { func OPMLImportHandler(rw http.ResponseWriter, req *http.Request) {
if req.Method == "POST" { if req.Method == "POST" {
file, _, err := req.FormFile("opml") file, _, err := req.FormFile("opml")
@ -358,12 +384,25 @@ func OPMLImportHandler(rw http.ResponseWriter, req *http.Request) {
log.Print(err) log.Print(err)
return return
} }
content, err := ioutil.ReadAll(file) feeds := new(opml)
decoder := xml.NewDecoder(file)
decoder.Entity = xml.HTMLEntity
decoder.Strict = false
err = decoder.Decode(&feeds)
if err != nil { if err != nil {
log.Print(err) log.Print(err)
return return
} }
fmt.Println(string(content)) for _, outline := range feeds.Outlines {
if outline.Type == "rss" {
db(req).CreateFeed(outline.Title, outline.Description, outline.SiteURL, outline.FeedURL, "", nil)
} else {
folder := db(req).CreateFolder(outline.Title)
for _, o := range outline.AllFeeds() {
db(req).CreateFeed(o.Title, o.Description, o.SiteURL, o.FeedURL, "", &folder.Id)
}
}
}
} else { } else {
rw.WriteHeader(http.StatusMethodNotAllowed) rw.WriteHeader(http.StatusMethodNotAllowed)
} }

View File

@ -13,17 +13,34 @@ type Folder struct {
func (s *Storage) CreateFolder(title string) *Folder { func (s *Storage) CreateFolder(title string) *Folder {
expanded := true expanded := true
result, err := s.db.Exec(` result, err := s.db.Exec(`
insert into folders (title, is_expanded) values (?, ?)`, insert into folders (title, is_expanded) values (?, ?)
on conflict (title) do nothing`,
title, expanded, title, expanded,
) )
if err != nil { if err != nil {
fmt.Println(err) fmt.Println(err)
return nil return nil
} }
id, idErr := result.LastInsertId()
if idErr != nil { var id int64
numrows, err := result.RowsAffected()
if err != nil {
s.log.Print(err)
return nil return nil
} }
if numrows == 1 {
id, err = result.LastInsertId()
if err != nil {
s.log.Print(err)
return nil
}
} else {
err = s.db.QueryRow(`select id, is_expanded from folders where title=?`, title).Scan(&id, &expanded)
if err != nil {
s.log.Print(err)
return nil
}
}
return &Folder{Id: id, Title: title, IsExpanded: expanded} return &Folder{Id: id, Title: title, IsExpanded: expanded}
} }

View File

@ -14,6 +14,8 @@ create table if not exists folders (
is_expanded boolean not null default false is_expanded boolean not null default false
); );
create unique index if not exists idx_folder_title on folders(title);
create table if not exists feeds ( create table if not exists feeds (
id integer primary key autoincrement, id integer primary key autoincrement,
folder_id references folders(id), folder_id references folders(id),
@ -63,12 +65,10 @@ func New() (*Storage, error) {
if err != nil { if err != nil {
return nil, err return nil, err
} }
/*
_, err = db.Exec(initQuery) _, err = db.Exec(initQuery)
if err != nil { if err != nil {
return nil, err return nil, err
} }
*/
logger := log.New(os.Stdout, "storage: ", log.Ldate | log.Ltime | log.Lshortfile) logger := log.New(os.Stdout, "storage: ", log.Ldate | log.Ltime | log.Lshortfile)
return &Storage{db: db, log: logger}, nil return &Storage{db: db, log: logger}, nil
} }

View File

@ -241,10 +241,12 @@ var vm = new Vue({
api.items.update(item.id, {status: item.status}) api.items.update(item.id, {status: item.status})
}, },
importOPML: function(event) { importOPML: function(event) {
var vm = this
var input = event.target var input = event.target
var form = document.querySelector('#opml-import-form') var form = document.querySelector('#opml-import-form')
api.upload_opml(form).then(function() { api.upload_opml(form).then(function() {
input.value = '' input.value = ''
vm.refreshFeeds()
}) })
}, },
} }