mirror of
https://github.com/nkanaev/yarr.git
synced 2025-05-24 00:33:14 +00:00
import opml
This commit is contained in:
parent
9583b4b8d7
commit
ea4775839f
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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()
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user