diff --git a/.gitignore b/.gitignore
index 56e23a3..0f649f2 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,2 +1,5 @@
-node_modules
-.credentials
\ No newline at end of file
+/server
+/manager
+*.properties
+.DS_Store
+*.csv
\ No newline at end of file
diff --git a/Dockerfile b/Dockerfile
new file mode 100644
index 0000000..0f7b4da
--- /dev/null
+++ b/Dockerfile
@@ -0,0 +1,8 @@
+FROM golang:1.16
+WORKDIR /src
+COPY . ./
+RUN CGO_ENABLED=0 GOOS=linux go build -o server ./cmd/serve
+
+FROM scratch
+COPY --from=0 /src/server ./
+CMD ["/server"]
\ No newline at end of file
diff --git a/Makefile b/Makefile
new file mode 100644
index 0000000..178b88d
--- /dev/null
+++ b/Makefile
@@ -0,0 +1,34 @@
+.PHONY: up down run-server run-manager test
+
+GOFILES=$(shell find . -name '*.go' -o -name 'go.*')
+STATICFILES=$(shell find . -name '*.js' -o -name '*.css' -o -name '*.html')
+SQLFILES=$(shell find . -name '*.sql')
+
+ifneq (,$(wildcard ./local.properties))
+include local.properties
+export
+endif
+
+build: server manager
+
+run-server: build
+ ./server
+
+run-manager: build
+ ./manager
+
+server: $(GOFILES) $(STATICFILES)
+ go build -o server ./cmd/serve
+
+manager: $(GOFILES) $(SQLFILES)
+ go build -o manager ./cmd/manage
+
+test:
+ go test ./... -cover
+
+# dev dependencies
+up:
+ docker compose up -d
+
+down:
+ docker compose down
diff --git a/book/book.go b/book/book.go
new file mode 100644
index 0000000..4552e7a
--- /dev/null
+++ b/book/book.go
@@ -0,0 +1,21 @@
+package book
+
+type Book struct {
+ ID int `json:"-"`
+ Title string `json:"title"`
+ Authors []string `json:"authors"`
+ SortAuthor string `json:"sortAuthor"`
+ ISBN10 string `json:"isbn-10"`
+ ISBN13 string `json:"isbn-13"`
+ Format string `json:"format"`
+ Genre string `json:"genre"`
+ Publisher string `json:"publisher"`
+ Series string `json:"series"`
+ Volume string `json:"volume"`
+ Year string `json:"year"`
+ Signed bool `json:"signed"`
+ Description string `json:"description"`
+ Notes string `json:"notes"`
+ OnLoan string `json:"onLoan"`
+ CoverURL string `json:"coverURL"`
+}
diff --git a/cmd/manage/events.go b/cmd/manage/events.go
new file mode 100644
index 0000000..3b2468a
--- /dev/null
+++ b/cmd/manage/events.go
@@ -0,0 +1,113 @@
+package main
+
+import (
+ "git.yetaga.in/alazyreader/library/book"
+ "github.com/gdamore/tcell"
+)
+
+// error message
+type EventError struct {
+ tcell.EventTime
+ err error
+}
+
+func NewEventError(err error) *EventError {
+ e := &EventError{err: err}
+ e.SetEventNow()
+ return e
+}
+
+// save change to book
+type EventBookUpdate struct {
+ tcell.EventTime
+ book *book.Book
+}
+
+func NewEventBookUpdate(b *book.Book) *EventBookUpdate {
+ e := &EventBookUpdate{book: b}
+ e.SetEventNow()
+ return e
+}
+
+func (e *EventBookUpdate) Book() *book.Book {
+ return e.book
+}
+
+// open new book in display
+type EventLoadBook struct {
+ tcell.EventTime
+ ID int
+}
+
+func NewEventLoadBook(id int) *EventLoadBook {
+ e := &EventLoadBook{ID: id}
+ e.SetEventNow()
+ return e
+}
+
+// open new book in display
+type EventEnterBook struct {
+ tcell.EventTime
+}
+
+func NewEventEnterBook() *EventEnterBook {
+ e := &EventEnterBook{}
+ e.SetEventNow()
+ return e
+}
+
+// switch back to menu control
+type EventExitBook struct {
+ tcell.EventTime
+}
+
+func NewEventExitBook() *EventExitBook {
+ e := &EventExitBook{}
+ e.SetEventNow()
+ return e
+}
+
+// open import window
+type EventOpenImport struct {
+ tcell.EventTime
+}
+
+func NewEventOpenImport() *EventOpenImport {
+ e := &EventOpenImport{}
+ e.SetEventNow()
+ return e
+}
+
+// attempt to import given filename.csv
+type EventAttemptImport struct {
+ tcell.EventTime
+ filename string
+}
+
+func NewEventAttemptImport(f string) *EventAttemptImport {
+ e := &EventAttemptImport{filename: f}
+ e.SetEventNow()
+ return e
+}
+
+// close import window
+type EventCloseImport struct {
+ tcell.EventTime
+}
+
+func NewEventCloseImport() *EventCloseImport {
+ e := &EventCloseImport{}
+ e.SetEventNow()
+ return e
+}
+
+// quit
+type EventQuit struct {
+ tcell.EventTime
+}
+
+func NewEventQuit() *EventQuit {
+ e := &EventQuit{}
+ e.SetEventNow()
+ return e
+}
diff --git a/cmd/manage/main.go b/cmd/manage/main.go
new file mode 100644
index 0000000..3d105db
--- /dev/null
+++ b/cmd/manage/main.go
@@ -0,0 +1,336 @@
+package main
+
+import (
+ "context"
+ "fmt"
+ "log"
+ "os"
+ "runtime/debug"
+ "strings"
+ "sync"
+
+ "git.yetaga.in/alazyreader/library/book"
+ "git.yetaga.in/alazyreader/library/config"
+ "git.yetaga.in/alazyreader/library/database"
+ "git.yetaga.in/alazyreader/library/importer"
+ "git.yetaga.in/alazyreader/library/ui"
+ "github.com/gdamore/tcell"
+ "github.com/kelseyhightower/envconfig"
+)
+
+// State holds the UI state keys=>value map and manages access to the map with a mutex
+type State struct {
+ m sync.Mutex
+ stateMap map[string]interface{}
+}
+
+// key, present
+func (s *State) Get(key string) interface{} {
+ s.m.Lock()
+ defer s.m.Unlock()
+ if s.stateMap == nil {
+ s.stateMap = make(map[string]interface{})
+ }
+ k, ok := s.stateMap[key]
+ if !ok {
+ return nil
+ }
+ return k
+}
+
+// key, value
+func (s *State) Set(key string, value interface{}) {
+ s.m.Lock()
+ defer s.m.Unlock()
+ if s.stateMap == nil {
+ s.stateMap = make(map[string]interface{})
+ }
+ s.stateMap[key] = value
+}
+
+func max(a, b int) int {
+ if a > b {
+ return a
+ }
+ return b
+}
+
+// UI states
+const (
+ IN_MENU = iota
+ IN_BOOK
+ IN_IMPORT
+)
+
+func main() {
+ var c config.Config
+ err := envconfig.Process("library", &c)
+ if err != nil {
+ log.Fatalln(err)
+ }
+
+ // create state
+ state := State{}
+
+ // set up DB connection
+ if c.DBUser == "" || c.DBPass == "" || c.DBHost == "" || c.DBPort == "" || c.DBName == "" {
+ if c.DBPass != "" { // obscure password
+ c.DBPass = c.DBPass[0:max(3, len(c.DBPass))] + strings.Repeat("*", max(0, len(c.DBPass)-3))
+ }
+ log.Fatalf("vars: %+v", c)
+ }
+ lib, err := database.NewMySQLConnection(c.DBUser, c.DBPass, c.DBHost, c.DBPort, c.DBName)
+ if err != nil {
+ log.Fatalln(err)
+ }
+ err = lib.PrepareDatabase(context.Background())
+ if err != nil {
+ log.Fatalln(err)
+ }
+ _, _, err = lib.RunMigrations(context.Background())
+ if err != nil {
+ log.Fatalln(err)
+ }
+ books, err := lib.GetAllBooks(context.Background())
+ if err != nil {
+ log.Fatalln(err)
+ }
+ state.Set("library", books)
+
+ screen, err := tcell.NewScreen()
+ if err != nil {
+ log.Fatalln(err)
+ }
+ err = screen.Init()
+ if err != nil {
+ log.Fatalln(err)
+ }
+ // cleanup our screen and log if we panic and crash out somewhere
+ defer func() {
+ if r := recover(); r != nil {
+ if screen != nil {
+ screen.Fini()
+ }
+ fmt.Println("fatal panic;", r)
+ if c.Debug {
+ fmt.Println("stacktrace: \n" + string(debug.Stack()))
+ }
+ return
+ }
+ }()
+
+ // book list and options menu (left column)
+ l := ui.NewList(Titles(state.Get("library").([]book.Book)), 0)
+ menu := ui.NewBox(
+ "library",
+ []string{"˄˅ select", "⏎ edit", "(n)ew", "(i)mport", "(q)uit"},
+ ui.Contents{{
+ Offsets: ui.Offsets{Top: 1, Left: 2, Bottom: -2, Right: -2},
+ Container: l,
+ }},
+ ui.StyleActive,
+ false,
+ )
+ activeBookDetails := ui.NewBookDetails(&book.Book{})
+
+ // book display (right column)
+ activeBook := ui.NewBox(
+ "book",
+ []string{"˄˅ select", "⏎ edit", "(esc) close"},
+ ui.Contents{{
+ Offsets: ui.Offsets{Top: 1, Left: 2, Bottom: 0, Right: 0},
+ Container: activeBookDetails,
+ }},
+ ui.StyleInactive,
+ false,
+ )
+
+ // parent container
+ container := ui.NewContainer(
+ ui.Contents{
+ {Container: menu, Offsets: ui.Offsets{Percent: 1}},
+ {Container: activeBook, Offsets: ui.Offsets{Percent: 2}},
+ },
+ ui.LayoutHorizontalPercent,
+ )
+
+ // import pop-up
+ wd, _ := os.Getwd()
+ fileSelector := ui.NewEditableTextLine(wd)
+ fileSelector.ResetCursor(false)
+ fileSelector.SetStyle(ui.StyleActive.Underline(true))
+ popup := ui.NewBox(
+ "import csv file",
+ []string{"⏎ submit", "(esc)close"},
+ ui.Contents{
+ {Container: fileSelector, Offsets: ui.Offsets{Top: 2, Left: 2}},
+ },
+ ui.StyleActive,
+ false,
+ )
+ popup.SetVisible(false)
+
+ // error pop-up
+ errorMessage := ui.NewEditableTextLine("")
+ errorPopup := ui.NewBox(
+ "error",
+ []string{"⏎ close"},
+ ui.Contents{
+ {Container: errorMessage, Offsets: ui.Offsets{Top: 1, Left: 1}},
+ },
+ ui.StyleActive.Bold(true).Foreground(tcell.ColorRed),
+ false,
+ )
+ errorPopup.SetVisible(false)
+
+ // init
+ screen.Clear()
+ w, h := screen.Size()
+ container.SetSize(0, 0, h, w)
+ container.Draw(screen)
+ screen.Sync()
+
+ // init UI state
+ state.Set("ui_state", IN_MENU)
+ screen.PostEvent(NewEventLoadBook(l.SelectedID()))
+
+ // UI loop
+ for {
+ e := screen.PollEvent()
+ switch v := e.(type) {
+ case *tcell.EventError:
+ fmt.Fprintf(os.Stderr, "%v", v)
+ screen.Beep()
+ case *tcell.EventKey: // input handling
+ curr := state.Get("ui_state").(int)
+ if curr == IN_MENU {
+ if v.Key() == tcell.KeyUp && l.Selected() > 0 {
+ l.SetSelected(l.Selected() - 1)
+ screen.PostEvent(NewEventLoadBook(l.SelectedID()))
+ }
+ if v.Key() == tcell.KeyDown && l.Selected() < len(l.ListMembers())-1 {
+ l.SetSelected(l.Selected() + 1)
+ screen.PostEvent(NewEventLoadBook(l.SelectedID()))
+ }
+ if v.Key() == tcell.KeyEnter {
+ screen.PostEvent(NewEventEnterBook())
+ }
+ if v.Rune() == 'i' {
+ screen.PostEvent(NewEventOpenImport())
+ }
+ if v.Rune() == 'q' {
+ screen.PostEvent(NewEventQuit())
+ }
+ } else if curr == IN_BOOK {
+ if v.Key() == tcell.KeyEsc {
+ screen.PostEvent(NewEventExitBook())
+ }
+ } else if curr == IN_IMPORT {
+ if v.Key() == tcell.KeyEsc {
+ fileSelector.SetText(wd)
+ fileSelector.ResetCursor(false)
+ screen.PostEvent(NewEventCloseImport())
+ }
+ if v.Key() == tcell.KeyBackspace || v.Key() == tcell.KeyBackspace2 {
+ fileSelector.DeleteAtCursor()
+ } else if v.Key() == tcell.KeyRight {
+ fileSelector.MoveCursor(1)
+
+ } else if v.Key() == tcell.KeyLeft {
+ fileSelector.MoveCursor(-1)
+ } else if v.Key() == tcell.KeyEnter {
+ screen.PostEvent(NewEventAttemptImport(fileSelector.Text()))
+ } else if v.Rune() != 0 {
+ fileSelector.InsertAtCursor(v.Rune())
+ }
+ }
+ case *tcell.EventResize: // screen redraw
+ w, h := screen.Size()
+ container.SetSize(0, 0, h, w)
+ case *EventBookUpdate:
+ // TK
+ case *EventEnterBook:
+ activeBook.SetStyle(ui.StyleActive)
+ menu.SetStyle(ui.StyleInactive)
+ state.Set("ui_state", IN_BOOK)
+ case *EventExitBook:
+ state.Set("ui_state", IN_MENU)
+ activeBook.SetStyle(ui.StyleInactive)
+ menu.SetStyle(ui.StyleActive)
+ case *EventLoadBook:
+ activeBookDetails.SetBook(GetBookByID(v.ID, books))
+ case *EventOpenImport:
+ state.Set("ui_state", IN_IMPORT)
+ menu.SetStyle(ui.StyleInactive)
+ popup.SetVisible(true)
+ popup.SetSize(6, 3, 5, 80)
+ case *EventAttemptImport:
+ // this will block other events, but it shouldn't take too long...
+ f, err := os.Open(v.filename)
+ if err != nil {
+ screen.PostEvent(NewEventError(err))
+ continue
+ }
+ books, err := importer.CSVToBooks(f)
+ if err != nil {
+ screen.PostEvent(NewEventError(err))
+ continue
+ }
+ for b := range books {
+ err = lib.AddBook(context.Background(), &books[b])
+ if err != nil {
+ screen.PostEvent(NewEventError(err))
+ }
+ }
+ screen.PostEvent(NewEventCloseImport())
+ allbooks, err := lib.GetAllBooks(context.Background())
+ if err != nil {
+ screen.PostEvent(NewEventError(err))
+ }
+ state.Set("library", allbooks)
+ state.Set("ui_state", IN_MENU)
+ case *EventCloseImport:
+ state.Set("ui_state", IN_MENU)
+ screen.HideCursor()
+ menu.SetStyle(ui.StyleActive)
+ popup.SetVisible(false)
+ case *EventError:
+ errorMessage.SetText(v.err.Error())
+ errorPopup.SetVisible(true)
+ case *EventQuit:
+ screen.Fini()
+ fmt.Printf("Thank you for playing Wing Commander!\n\n")
+ return
+ case *tcell.EventInterrupt:
+ case *tcell.EventMouse:
+ case *tcell.EventTime:
+ default:
+ }
+ // repaint
+ l.SetMembers(Titles(state.Get("library").([]book.Book)))
+ container.Draw(screen)
+ popup.Draw(screen)
+ errorPopup.Draw(screen)
+ screen.Show()
+ }
+}
+
+func Titles(lb []book.Book) []ui.ListKeyValue {
+ r := []ui.ListKeyValue{}
+ for i := range lb {
+ r = append(r, ui.ListKeyValue{
+ Key: lb[i].ID,
+ Value: lb[i].Title,
+ })
+ }
+ return r
+}
+
+func GetBookByID(id int, lb []book.Book) *book.Book {
+ for i := range lb {
+ if lb[i].ID == id {
+ return &lb[i]
+ }
+ }
+ return &book.Book{}
+}
diff --git a/cmd/serve/main.go b/cmd/serve/main.go
new file mode 100644
index 0000000..567a0de
--- /dev/null
+++ b/cmd/serve/main.go
@@ -0,0 +1,100 @@
+package main
+
+import (
+ "context"
+ "encoding/json"
+ "io/fs"
+ "log"
+ "net/http"
+ "strings"
+
+ "git.yetaga.in/alazyreader/library/book"
+ "git.yetaga.in/alazyreader/library/config"
+ "git.yetaga.in/alazyreader/library/database"
+ "git.yetaga.in/alazyreader/library/frontend"
+ "github.com/kelseyhightower/envconfig"
+)
+
+func max(a, b int) int {
+ if a > b {
+ return a
+ }
+ return b
+}
+
+type Library interface {
+ GetAllBooks(context.Context) ([]book.Book, error)
+}
+
+type Router struct {
+ static fs.FS
+ lib Library
+}
+
+func (r *Router) ServeHTTP(w http.ResponseWriter, req *http.Request) {
+ if req.URL.Path == "/api" {
+ APIHandler(r.lib).ServeHTTP(w, req)
+ return
+ }
+ StaticHandler(r.static).ServeHTTP(w, req)
+}
+
+func APIHandler(l Library) http.Handler {
+ return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ books, err := l.GetAllBooks(r.Context())
+ if err != nil {
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ b, err := json.Marshal(books)
+ if err != nil {
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ w.Header().Set("Content-Type", "application/json")
+ w.WriteHeader(http.StatusOK)
+ w.Write(b)
+ w.Write([]byte("\n"))
+ })
+}
+
+func StaticHandler(f fs.FS) http.Handler {
+ return http.FileServer(http.FS(f))
+}
+
+func main() {
+ var c config.Config
+ err := envconfig.Process("library", &c)
+ if err != nil {
+ log.Fatalln(err)
+ }
+ f, err := frontend.Root()
+ if err != nil {
+ log.Fatalln(err)
+ }
+ if c.DBUser == "" || c.DBPass == "" || c.DBHost == "" || c.DBPort == "" || c.DBName == "" {
+ if c.DBPass != "" { // obscure password
+ c.DBPass = c.DBPass[0:max(3, len(c.DBPass))] + strings.Repeat("*", max(0, len(c.DBPass)-3))
+ }
+ log.Fatalf("vars: %+v", c)
+ }
+ lib, err := database.NewMySQLConnection(c.DBUser, c.DBPass, c.DBHost, c.DBPort, c.DBName)
+ if err != nil {
+ log.Fatalln(err)
+ }
+ err = lib.PrepareDatabase(context.Background())
+ if err != nil {
+ log.Fatalln(err)
+ }
+ latest, run, err := lib.RunMigrations(context.Background())
+ if err != nil {
+ log.Fatalln(err)
+ }
+ log.Printf("latest migration: %d; migrations run: %d", latest, run)
+ r := &Router{
+ static: f,
+ lib: lib,
+ }
+ log.Println("Listening on http://localhost:8080/")
+ log.Fatalln(http.ListenAndServe(":8080", r))
+}
diff --git a/config/config.go b/config/config.go
new file mode 100644
index 0000000..0813444
--- /dev/null
+++ b/config/config.go
@@ -0,0 +1,10 @@
+package config
+
+type Config struct {
+ DBUser string
+ DBPass string
+ DBHost string
+ DBPort string
+ DBName string
+ Debug bool
+}
diff --git a/css/reset.css b/css/reset.css
deleted file mode 100644
index 9ce89e8..0000000
--- a/css/reset.css
+++ /dev/null
@@ -1,48 +0,0 @@
-/* http://meyerweb.com/eric/tools/css/reset/
- v2.0 | 20110126
- License: none (public domain)
-*/
-
-html, body, div, span, applet, object, iframe,
-h1, h2, h3, h4, h5, h6, p, blockquote, pre,
-a, abbr, acronym, address, big, cite, code,
-del, dfn, em, img, ins, kbd, q, s, samp,
-small, strike, strong, sub, sup, tt, var,
-b, u, i, center,
-dl, dt, dd, ol, ul, li,
-fieldset, form, label, legend,
-table, caption, tbody, tfoot, thead, tr, th, td,
-article, aside, canvas, details, embed,
-figure, figcaption, footer, header, hgroup,
-menu, nav, output, ruby, section, summary,
-time, mark, audio, video {
- margin: 0;
- padding: 0;
- border: 0;
- font-size: 100%;
- font: inherit;
- vertical-align: baseline;
-}
-/* HTML5 display-role reset for older browsers */
-article, aside, details, figcaption, figure,
-footer, header, hgroup, menu, nav, section {
- display: block;
-}
-body {
- line-height: 1;
-}
-ol, ul {
- list-style: none;
-}
-blockquote, q {
- quotes: none;
-}
-blockquote:before, blockquote:after,
-q:before, q:after {
- content: '';
- content: none;
-}
-table {
- border-collapse: collapse;
- border-spacing: 0;
-}
\ No newline at end of file
diff --git a/database/memory.go b/database/memory.go
new file mode 100644
index 0000000..58c027e
--- /dev/null
+++ b/database/memory.go
@@ -0,0 +1,79 @@
+package database
+
+import (
+ "context"
+ "fmt"
+ "sync"
+
+ "git.yetaga.in/alazyreader/library/book"
+)
+
+type Memory struct {
+ lock sync.Mutex
+ shelf []book.Book
+}
+
+func (m *Memory) GetAllBooks(_ context.Context) ([]book.Book, error) {
+ m.lock.Lock()
+ defer m.lock.Unlock()
+
+ if m.shelf == nil {
+ m.shelf = []book.Book{}
+ }
+
+ return m.shelf, nil
+}
+
+func (m *Memory) AddBook(_ context.Context, b *book.Book) error {
+ m.lock.Lock()
+ defer m.lock.Unlock()
+
+ if m.shelf == nil {
+ m.shelf = []book.Book{}
+ }
+
+ m.shelf = append(m.shelf, *b)
+ return nil
+}
+
+func (m *Memory) UpdateBook(_ context.Context, old, new *book.Book) error {
+ m.lock.Lock()
+ defer m.lock.Unlock()
+
+ if m.shelf == nil {
+ m.shelf = []book.Book{}
+ return fmt.Errorf("book does not exist")
+ }
+
+ if old.ID != new.ID {
+ return fmt.Errorf("cannot change book ID")
+ }
+
+ for i := range m.shelf {
+ if m.shelf[i].ID == old.ID {
+ m.shelf[i] = *new
+ return nil
+ }
+ }
+ return fmt.Errorf("book does not exist")
+}
+
+func (m *Memory) DeleteBook(_ context.Context, b *book.Book) error {
+ m.lock.Lock()
+ defer m.lock.Unlock()
+
+ if m.shelf == nil {
+ m.shelf = []book.Book{}
+ return fmt.Errorf("book does not exist")
+ }
+
+ for i := range m.shelf {
+ if m.shelf[i].ID == b.ID {
+ // reorder slice to remove book quickly
+ m.shelf[i] = m.shelf[len(m.shelf)-1]
+ m.shelf = m.shelf[:len(m.shelf)-1]
+ return nil
+ }
+ }
+ return fmt.Errorf("book does not exist")
+}
diff --git a/database/migrations/mysql/01-init.sql b/database/migrations/mysql/01-init.sql
new file mode 100644
index 0000000..a456a86
--- /dev/null
+++ b/database/migrations/mysql/01-init.sql
@@ -0,0 +1,20 @@
+CREATE TABLE IF NOT EXISTS books(
+ id INT NOT NULL AUTO_INCREMENT,
+ title VARCHAR(1024),
+ authors VARCHAR(1024),
+ sortauthor VARCHAR(1024),
+ isbn10 VARCHAR(10),
+ isbn13 VARCHAR(13),
+ format VARCHAR(255),
+ genre VARCHAR(255),
+ publisher VARCHAR(255),
+ series VARCHAR(255),
+ volume VARCHAR(255),
+ year VARCHAR(10),
+ signed BOOLEAN,
+ description TEXT,
+ notes TEXT,
+ onloan VARCHAR(255),
+ coverurl VARCHAR(1024),
+ PRIMARY KEY (id)
+)
\ No newline at end of file
diff --git a/database/mysql.go b/database/mysql.go
new file mode 100644
index 0000000..bd0d1e0
--- /dev/null
+++ b/database/mysql.go
@@ -0,0 +1,301 @@
+package database
+
+import (
+ "context"
+ "database/sql"
+ "embed"
+ "fmt"
+ "io/fs"
+ "strconv"
+ "strings"
+ "time"
+
+ "git.yetaga.in/alazyreader/library/book"
+ _ "github.com/go-sql-driver/mysql"
+)
+
+//go:embed migrations/mysql
+var migrationsFS embed.FS
+
+type migration struct {
+ id int
+ name string
+ content []byte
+}
+
+type MySQL struct {
+ connection *sql.DB
+ tableName string
+ versionTable string
+ migrationsDirectory string
+}
+
+func NewMySQLConnection(user, pass, host, port, db string) (*MySQL, error) {
+ dsn := fmt.Sprintf("%s:%s@tcp(%s:%s)/%s", user, pass, host, port, db) // what a strange syntax
+ connection, err := sql.Open("mysql", dsn)
+ if err != nil {
+ return nil, err
+ }
+ return &MySQL{
+ connection: connection,
+ tableName: "books",
+ versionTable: "migrations",
+ migrationsDirectory: "migrations/mysql",
+ }, nil
+}
+
+func (m *MySQL) PrepareDatabase(ctx context.Context) error {
+ if m.connection == nil || m.migrationsDirectory == "" || m.versionTable == "" {
+ return fmt.Errorf("uninitialized mysql client")
+ }
+
+ tablecheck := `SELECT count(*) AS count
+ FROM information_schema.TABLES
+ WHERE TABLE_NAME = '` + m.versionTable + `'
+ AND TABLE_SCHEMA in (SELECT DATABASE());`
+ tableschema := `CREATE TABLE ` + m.versionTable + `(
+ id INT NOT NULL,
+ name VARCHAR(100) NOT NULL,
+ datetime DATE,
+ PRIMARY KEY (id))`
+
+ var versionTableExists int
+ m.connection.QueryRowContext(ctx, tablecheck).Scan(&versionTableExists)
+ if versionTableExists != 0 {
+ return nil
+ }
+ _, err := m.connection.ExecContext(ctx, tableschema)
+ return err
+}
+
+func (m *MySQL) GetLatestMigration(ctx context.Context) (int, error) {
+ if m.connection == nil || m.migrationsDirectory == "" || m.versionTable == "" {
+ return 0, fmt.Errorf("uninitialized mysql client")
+ }
+
+ var latestMigration int
+ err := m.connection.QueryRowContext(ctx, "SELECT COALESCE(MAX(id), 0) FROM "+m.versionTable).Scan(&latestMigration)
+ return latestMigration, err
+}
+
+func (m *MySQL) RunMigrations(ctx context.Context) (int, int, error) {
+ if m.connection == nil || m.migrationsDirectory == "" || m.versionTable == "" {
+ return 0, 0, fmt.Errorf("uninitialized mysql client")
+ }
+
+ migrations := map[int]migration{}
+ dir, err := migrationsFS.ReadDir(m.migrationsDirectory)
+ if err != nil {
+ return 0, 0, err
+ }
+ for f := range dir {
+ if dir[f].Type().IsRegular() {
+ mig := migration{}
+ id, name, err := parseMigrationFileName(dir[f].Name())
+ if err != nil {
+ return 0, 0, err
+ }
+ mig.id, mig.name = id, name
+ mig.content, err = fs.ReadFile(migrationsFS, m.migrationsDirectory+"/"+dir[f].Name())
+ migrations[mig.id] = mig
+ }
+ }
+
+ latestMigrationRan, err := m.GetLatestMigration(ctx)
+ if err != nil {
+ return 0, 0, err
+ }
+
+ // exit if nothing to do (that is, there's no greater migration ID)
+ if _, ok := migrations[latestMigrationRan+1]; !ok {
+ return latestMigrationRan, 0, nil
+ }
+
+ // loop over and apply migrations if required
+ tx, err := m.connection.BeginTx(ctx, nil)
+ if err != nil {
+ return latestMigrationRan, 0, err
+ }
+ migrationsRun := 0
+ for migrationsToRun := true; migrationsToRun; _, migrationsToRun = migrations[latestMigrationRan+1] {
+ mig := migrations[latestMigrationRan+1]
+ _, err := tx.ExecContext(ctx, string(mig.content))
+ if err != nil {
+ nestederr := tx.Rollback()
+ if nestederr != nil {
+ return latestMigrationRan, migrationsRun, nestederr
+ }
+ return latestMigrationRan, migrationsRun, err
+ }
+ _, err = tx.ExecContext(ctx, "INSERT INTO "+m.versionTable+" (id, name, datetime) VALUES (?, ?, ?)", mig.id, mig.name, time.Now())
+ if err != nil {
+ nestederr := tx.Rollback()
+ if nestederr != nil {
+ return latestMigrationRan, migrationsRun, nestederr
+ }
+ return latestMigrationRan, migrationsRun, err
+ }
+ latestMigrationRan = latestMigrationRan + 1
+ migrationsRun = migrationsRun + 1
+ }
+ err = tx.Commit()
+ return latestMigrationRan, migrationsRun, err
+}
+
+func (m *MySQL) GetAllBooks(ctx context.Context) ([]book.Book, error) {
+ if m.connection == nil {
+ return nil, fmt.Errorf("uninitialized mysql client")
+ }
+
+ books := []book.Book{}
+ rows, err := m.connection.QueryContext(ctx, `
+ SELECT id,
+ title,
+ authors,
+ sortauthor,
+ isbn10,
+ isbn13,
+ format,
+ genre,
+ publisher,
+ series,
+ volume,
+ year,
+ signed,
+ description,
+ notes,
+ onloan,
+ coverurl
+ FROM `+m.tableName)
+ if err != nil {
+ return nil, err
+ }
+ defer rows.Close()
+
+ for rows.Next() {
+ b := book.Book{}
+ var authors string
+ err := rows.Scan(
+ &b.ID, &b.Title, &authors,
+ &b.SortAuthor, &b.ISBN10, &b.ISBN13,
+ &b.Format, &b.Genre, &b.Publisher,
+ &b.Series, &b.Volume, &b.Year,
+ &b.Signed, &b.Description, &b.Notes,
+ &b.OnLoan, &b.CoverURL)
+ if err != nil {
+ return nil, err
+ }
+ b.Authors = strings.Split(authors, ";")
+ books = append(books, b)
+ }
+
+ return books, nil
+}
+
+func (m *MySQL) AddBook(ctx context.Context, b *book.Book) error {
+ if m.connection == nil {
+ return fmt.Errorf("uninitialized mysql client")
+ }
+
+ res, err := m.connection.ExecContext(ctx, `
+ INSERT INTO `+m.tableName+`
+ (title, authors, sortauthor, isbn10, isbn13, format, genre, publisher, series, volume, year, signed, description, notes, onloan, coverurl)
+ VALUES
+ (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
+ b.Title,
+ strings.Join(b.Authors, ";"),
+ b.SortAuthor,
+ b.ISBN10,
+ b.ISBN13,
+ b.Format,
+ b.Genre,
+ b.Publisher,
+ b.Series,
+ b.Volume,
+ b.Year,
+ b.Signed,
+ b.Description,
+ b.Notes,
+ b.OnLoan,
+ b.CoverURL,
+ )
+ if err != nil {
+ return err
+ }
+ i, err := res.RowsAffected()
+ if err != nil {
+ return err
+ }
+ if i != 1 {
+ return fmt.Errorf("unexpectedly updated more than one row: %d", i)
+ }
+ return nil
+}
+
+func (m *MySQL) UpdateBook(ctx context.Context, old, new *book.Book) error {
+ if m.connection == nil {
+ return fmt.Errorf("uninitialized mysql client")
+ }
+ if old.ID != new.ID {
+ return fmt.Errorf("cannot change book ID")
+ }
+
+ res, err := m.connection.ExecContext(ctx, `
+ UPDATE `+m.tableName+`
+ SET id=?
+ title=?
+ authors=?
+ sortauthor=?
+ isbn10=?
+ isbn13=?
+ format=?
+ genre=?
+ publisher=?
+ series=?
+ volume=?
+ year=?
+ signed=?
+ description=?
+ notes=?
+ onloan=?
+ coverurl=?
+ WHERE id=?`,
+ new.Title,
+ strings.Join(new.Authors, ";"),
+ new.SortAuthor,
+ new.ISBN10,
+ new.ISBN13,
+ new.Format,
+ new.Genre,
+ new.Publisher,
+ new.Series,
+ new.Volume,
+ new.Year,
+ new.Signed,
+ new.Description,
+ new.Notes,
+ new.OnLoan,
+ new.CoverURL,
+ old.ID)
+ if err != nil {
+ return err
+ }
+ i, err := res.RowsAffected()
+ if err != nil {
+ return err
+ }
+ if i != 1 {
+ return fmt.Errorf("unexpectedly updated more than one row: %d", i)
+ }
+ return nil
+}
+
+func parseMigrationFileName(filename string) (int, string, error) {
+ sp := strings.SplitN(filename, "-", 2)
+ i, err := strconv.Atoi(sp[0])
+ if err != nil {
+ return 0, "", err
+ }
+ tr := strings.TrimSuffix(sp[1], ".sql")
+ return i, tr, nil
+}
diff --git a/docker-compose.yml b/docker-compose.yml
new file mode 100644
index 0000000..362dc18
--- /dev/null
+++ b/docker-compose.yml
@@ -0,0 +1,8 @@
+version: "3.8"
+services:
+ mysql:
+ image: mysql:5.7
+ ports:
+ - 3306:3306
+ environment:
+ - MYSQL_ROOT_PASSWORD=KigYBNCT9IU5XyB3ehzMLFWyI # for dev testing only, obviously.
diff --git a/favicon.ico b/favicon.ico
deleted file mode 100644
index 1824d53..0000000
Binary files a/favicon.ico and /dev/null differ
diff --git a/frontend/files/favicon.ico b/frontend/files/favicon.ico
new file mode 100644
index 0000000..668a4f1
Binary files /dev/null and b/frontend/files/favicon.ico differ
diff --git a/frontend/files/favicon.png b/frontend/files/favicon.png
new file mode 100644
index 0000000..ce2be3f
Binary files /dev/null and b/frontend/files/favicon.png differ
diff --git a/frontend/files/index.html b/frontend/files/index.html
new file mode 100644
index 0000000..7d2f96d
--- /dev/null
+++ b/frontend/files/index.html
@@ -0,0 +1,243 @@
+
+
+
+ Library
+
+
+
+
+
+
+
+
+
+
No Book Selected
+
+
+
+
+
diff --git a/css/style.css b/frontend/files/style.css
similarity index 61%
rename from css/style.css
rename to frontend/files/style.css
index ecf2083..7e06c43 100644
--- a/css/style.css
+++ b/frontend/files/style.css
@@ -1,3 +1,134 @@
+/* http://meyerweb.com/eric/tools/css/reset/
+ v2.0 | 20110126
+ License: none (public domain)
+*/
+
+html,
+body,
+div,
+span,
+applet,
+object,
+iframe,
+h1,
+h2,
+h3,
+h4,
+h5,
+h6,
+p,
+blockquote,
+pre,
+a,
+abbr,
+acronym,
+address,
+big,
+cite,
+code,
+del,
+dfn,
+em,
+img,
+ins,
+kbd,
+q,
+s,
+samp,
+small,
+strike,
+strong,
+sub,
+sup,
+tt,
+var,
+b,
+u,
+i,
+center,
+dl,
+dt,
+dd,
+ol,
+ul,
+li,
+fieldset,
+form,
+label,
+legend,
+table,
+caption,
+tbody,
+tfoot,
+thead,
+tr,
+th,
+td,
+article,
+aside,
+canvas,
+details,
+embed,
+figure,
+figcaption,
+footer,
+header,
+hgroup,
+menu,
+nav,
+output,
+ruby,
+section,
+summary,
+time,
+mark,
+audio,
+video {
+ margin: 0;
+ padding: 0;
+ border: 0;
+ font-size: 100%;
+ font: inherit;
+ vertical-align: baseline;
+}
+article,
+aside,
+details,
+figcaption,
+figure,
+footer,
+header,
+hgroup,
+menu,
+nav,
+section {
+ display: block;
+}
+body {
+ line-height: 1;
+}
+ol,
+ul {
+ list-style: none;
+}
+blockquote,
+q {
+ quotes: none;
+}
+blockquote:before,
+blockquote:after,
+q:before,
+q:after {
+ content: "";
+ content: none;
+}
+table {
+ border-collapse: collapse;
+ border-spacing: 0;
+}
+
+/* site CSS starts here */
+
body {
overflow: hidden;
}
@@ -8,10 +139,10 @@ body {
padding: 4px 10px;
background-color: #f7f3dc;
border-bottom: 2px solid #d8d0a0;
- font-family: 'Libre Baskerville', sans-serif;
+ font-family: "Libre Baskerville", sans-serif;
}
-#header h1{
+#header h1 {
font-size: xx-large;
display: inline;
}
@@ -31,7 +162,7 @@ body {
padding: 2px 5px;
border: none;
border-bottom: 2px solid #d8d0a0;
- font-family: 'Libre Baskerville', sans-serif;
+ font-family: "Libre Baskerville", sans-serif;
}
#searchBox input:focus {
@@ -39,7 +170,7 @@ body {
}
#searchBox input::placeholder {
- font-family: 'Libre Baskerville', sans-serif;
+ font-family: "Libre Baskerville", sans-serif;
color: #d8d0a0;
}
@@ -63,7 +194,7 @@ body {
.bookTable th {
font-weight: bold;
text-align: left;
- font-family: 'Libre Baskerville', sans-serif;
+ font-family: "Libre Baskerville", sans-serif;
}
.bookTable th[data-sort-by] {
@@ -97,7 +228,8 @@ body {
bottom: 2px;
}
-.bookTable td, .bookTable th {
+.bookTable td,
+.bookTable th {
padding: 5px;
min-width: 50px;
}
diff --git a/frontend/frontend.go b/frontend/frontend.go
new file mode 100644
index 0000000..49c6b61
--- /dev/null
+++ b/frontend/frontend.go
@@ -0,0 +1,13 @@
+package frontend
+
+import (
+ "embed"
+ "io/fs"
+)
+
+//go:embed files
+var static embed.FS
+
+func Root() (fs.FS, error) {
+ return fs.Sub(static, "files")
+}
diff --git a/go.mod b/go.mod
new file mode 100644
index 0000000..a6be73c
--- /dev/null
+++ b/go.mod
@@ -0,0 +1,9 @@
+module git.yetaga.in/alazyreader/library
+
+go 1.16
+
+require (
+ github.com/gdamore/tcell v1.4.0
+ github.com/go-sql-driver/mysql v1.6.0
+ github.com/kelseyhightower/envconfig v1.4.0
+)
diff --git a/go.sum b/go.sum
new file mode 100644
index 0000000..4780059
--- /dev/null
+++ b/go.sum
@@ -0,0 +1,16 @@
+github.com/gdamore/encoding v1.0.0 h1:+7OoQ1Bc6eTm5niUzBa0Ctsh6JbMW6Ra+YNuAtDBdko=
+github.com/gdamore/encoding v1.0.0/go.mod h1:alR0ol34c49FCSBLjhosxzcPHQbf2trDkoo5dl+VrEg=
+github.com/gdamore/tcell v1.4.0 h1:vUnHwJRvcPQa3tzi+0QI4U9JINXYJlOz9yiaiPQ2wMU=
+github.com/gdamore/tcell v1.4.0/go.mod h1:vxEiSDZdW3L+Uhjii9c3375IlDmR05bzxY404ZVSMo0=
+github.com/go-sql-driver/mysql v1.6.0 h1:BCTh4TKNUYmOmMUcQ3IipzF5prigylS7XXjEkfCHuOE=
+github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg=
+github.com/kelseyhightower/envconfig v1.4.0 h1:Im6hONhd3pLkfDFsbRgu68RDNkGF1r3dvMUtDTo2cv8=
+github.com/kelseyhightower/envconfig v1.4.0/go.mod h1:cccZRl6mQpaq41TPp5QxidR+Sa3axMbJDNb//FQX6Gg=
+github.com/lucasb-eyer/go-colorful v1.0.3 h1:QIbQXiugsb+q10B+MI+7DI1oQLdmnep86tWFlaaUAac=
+github.com/lucasb-eyer/go-colorful v1.0.3/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
+github.com/mattn/go-runewidth v0.0.7 h1:Ei8KR0497xHyKJPAv59M1dkC+rOZCMBJ+t3fZ+twI54=
+github.com/mattn/go-runewidth v0.0.7/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI=
+golang.org/x/sys v0.0.0-20190626150813-e07cf5db2756 h1:9nuHUbU8dRnRRfj9KjWUVrJeoexdbeMjttk6Oh1rD10=
+golang.org/x/sys v0.0.0-20190626150813-e07cf5db2756/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg=
+golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
diff --git a/importer/csv.go b/importer/csv.go
new file mode 100644
index 0000000..e458577
--- /dev/null
+++ b/importer/csv.go
@@ -0,0 +1,65 @@
+package importer
+
+import (
+ "encoding/csv"
+ "io"
+ "strings"
+
+ "git.yetaga.in/alazyreader/library/book"
+)
+
+func CSVToBooks(r io.Reader) ([]book.Book, error) {
+ reader := csv.NewReader(r)
+ header, err := reader.Read()
+ if err != nil {
+ return nil, err
+ }
+ hmap := parseHeader(header)
+ books := []book.Book{}
+
+ for {
+ row, err := reader.Read()
+ if err == io.EOF {
+ break
+ }
+ if err != nil {
+ return books, err
+ }
+ b := book.Book{
+ Title: row[hmap["title"]],
+ Authors: parseAuthors(row[hmap["author"]]),
+ SortAuthor: row[hmap["authorlast"]],
+ ISBN10: row[hmap["isbn-10"]],
+ ISBN13: row[hmap["isbn-13"]],
+ Format: row[hmap["format"]],
+ Genre: row[hmap["genre"]],
+ Publisher: row[hmap["publisher"]],
+ Series: row[hmap["series"]],
+ Volume: row[hmap["volume"]],
+ Year: row[hmap["year"]],
+ Signed: row[hmap["signed"]] == "yes", // convert from known string to bool
+ Description: row[hmap["description"]],
+ Notes: row[hmap["notes"]],
+ OnLoan: row[hmap["onloan"]],
+ CoverURL: row[hmap["coverurl"]],
+ }
+ books = append(books, b)
+ }
+ return books, nil
+}
+
+func parseHeader(header []string) map[string]int {
+ m := make(map[string]int, len(header)-1)
+ for i := range header {
+ m[strings.TrimSpace(strings.ToLower(header[i]))] = i
+ }
+ return m
+}
+
+func parseAuthors(a string) []string {
+ as := strings.Split(a, ";")
+ for i := range as {
+ as[i] = strings.TrimSpace(as[i])
+ }
+ return as
+}
diff --git a/index.html b/index.html
deleted file mode 100644
index 3c20e80..0000000
--- a/index.html
+++ /dev/null
@@ -1,188 +0,0 @@
-
-
-
- Library
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/index.js b/index.js
deleted file mode 100644
index 818576f..0000000
--- a/index.js
+++ /dev/null
@@ -1,174 +0,0 @@
-var fs = require('fs');
-var readline = require('readline');
-var google = require('googleapis');
-var googleAuth = require('google-auth-library');
-var books = require('google-books-search');
-var _ = require('lodash');
-
-var SCOPES = ['https://www.googleapis.com/auth/spreadsheets'];
-var TOKEN_DIR = '.credentials/';
-var TOKEN_PATH = TOKEN_DIR + 'sheets.googleapis.com-my-library.json';
-var SHEET_ID = '1w5Dc57wV0_rrKFsG7KM-qdPWEpqYk6lFu3JzAA0cSv0';
-var SHEET_NAME = 'Sheet1';
-
-// Load client secrets from a local file.
-fs.readFile(TOKEN_DIR + 'client_secret.json', function processClientSecrets(err, content) {
- if (err) {
- console.log('Error loading client secret file: ' + err);
- return;
- }
- // Authorize a client with the loaded credentials, then call the
- // Google Sheets API.
- authorize(JSON.parse(content), inputLoop);
-});
-
-/**
- * Create an OAuth2 client with the given credentials, and then execute the
- * given callback function.
- *
- * @param {Object} credentials The authorization client credentials.
- * @param {function} callback The callback to call with the authorized client.
- */
-function authorize(credentials, callback) {
- var clientSecret = credentials.installed.client_secret;
- var clientId = credentials.installed.client_id;
- var redirectUrl = credentials.installed.redirect_uris[0];
- var auth = new googleAuth();
- var oauth2Client = new auth.OAuth2(clientId, clientSecret, redirectUrl);
-
- // Check if we have previously stored a token.
- fs.readFile(TOKEN_PATH, function(err, token) {
- if (err) {
- getNewToken(oauth2Client, callback);
- } else {
- oauth2Client.credentials = JSON.parse(token);
- callback(oauth2Client);
- }
- });
-}
-
-/**
- * Get and store new token after prompting for user authorization, and then
- * execute the given callback with the authorized OAuth2 client.
- *
- * @param {google.auth.OAuth2} oauth2Client The OAuth2 client to get token for.
- * @param {getEventsCallback} callback The callback to call with the authorized
- * client.
- */
-function getNewToken(oauth2Client, callback) {
- var authUrl = oauth2Client.generateAuthUrl({
- access_type: 'offline',
- scope: SCOPES
- });
- console.log('Authorize this app by visiting this url: ', authUrl);
- var rl = readline.createInterface({
- input: process.stdin,
- output: process.stdout
- });
- rl.question('Enter the code from that page here: ', function(code) {
- rl.close();
- oauth2Client.getToken(code, function(err, token) {
- if (err) {
- console.log('Error while trying to retrieve access token', err);
- return;
- }
- oauth2Client.credentials = token;
- storeToken(token);
- callback(oauth2Client);
- });
- });
-}
-
-/**
- * Store token to disk be used in later program executions.
- *
- * @param {Object} token The token to store to disk.
- */
-function storeToken(token) {
- try {
- fs.mkdirSync(TOKEN_DIR);
- } catch (err) {
- if (err.code != 'EEXIST') {
- throw err;
- }
- }
- fs.writeFile(TOKEN_PATH, JSON.stringify(token));
- console.log('Token stored to ' + TOKEN_PATH);
-}
-
-function inputLoop(auth) {
- insertRow(auth);
-}
-
-function insertRow(auth) {
- var rl = readline.createInterface({
- input: process.stdin,
- output: process.stdout
- });
- rl.question('ISBN: ', function(isbn) {
- rl.close();
- books.search(isbn, {
- field: 'isbn',
- lang: 'en'
- }, function(error, results) {
- if ( ! error && results.length > 0 ) {
- console.log(results[0]);
- var book = normalizeGoogleData(results[0]);
- console.log(book);
- appendToSheet(auth, _.values(book));
- } else if ( results.length == 0 ) {
- console.log("no book found");
- inputLoop(auth);
- } else {
- console.log(error);
- }
- });
- });
-}
-
-function normalizeGoogleData(book) {
- return {
- title: book.subtitle ? book.title + ': ' + book.subtitle : book.title,
- author: _.join(book.authors, ', '),
- authorLast: book.authors ? _.lowerCase(_.head(_.reverse(_.split(book.authors[0], ' ')))) : '',
- "isbn-10": _.find(book.industryIdentifiers, { type: "ISBN_10" })
- ? _.find(book.industryIdentifiers, { type: "ISBN_10" }).identifier
- : '',
- "isbn-13": _.find(book.industryIdentifiers, { type: "ISBN_13" })
- ? _.find(book.industryIdentifiers, { type: "ISBN_13" }).identifier
- : '',
- format: '',
- genre: book.categories ? book.categories[0] : '',
- publisher: _.trim(book.publisher, '"'),
- series: '',
- volume: '',
- publishedDate: book.publishedDate ? book.publishedDate.substring(0, 4) : '',
- coverurl: '',
- description: book.description,
- notes: '',
- signed: ''
- };
-}
-
-function appendToSheet(auth, book) {
- var sheets = google.sheets('v4');
- sheets.spreadsheets.values.append({
- auth: auth,
- spreadsheetId: SHEET_ID,
- range: SHEET_NAME,
- valueInputOption: "RAW",
- insertDataOption: "INSERT_ROWS",
- resource: {
- "range": SHEET_NAME,
- "majorDimension": "ROWS",
- "values": [
- book
- ]
- }
- }, function(err, response) {
- if (err) {
- console.log('The API returned an error: ' + err);
- }
- inputLoop(auth);
- });
-}
\ No newline at end of file
diff --git a/js/jquery.js b/js/jquery.js
deleted file mode 100644
index 2ec0d1d..0000000
--- a/js/jquery.js
+++ /dev/null
@@ -1,4 +0,0 @@
-/*! jQuery v3.2.0 | (c) JS Foundation and other contributors | jquery.org/license */
-!function(a,b){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){"use strict";var c=[],d=a.document,e=Object.getPrototypeOf,f=c.slice,g=c.concat,h=c.push,i=c.indexOf,j={},k=j.toString,l=j.hasOwnProperty,m=l.toString,n=m.call(Object),o={};function p(a,b){b=b||d;var c=b.createElement("script");c.text=a,b.head.appendChild(c).parentNode.removeChild(c)}var q="3.2.0",r=function(a,b){return new r.fn.init(a,b)},s=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,t=/^-ms-/,u=/-([a-z])/g,v=function(a,b){return b.toUpperCase()};r.fn=r.prototype={jquery:q,constructor:r,length:0,toArray:function(){return f.call(this)},get:function(a){return null==a?f.call(this):a<0?this[a+this.length]:this[a]},pushStack:function(a){var b=r.merge(this.constructor(),a);return b.prevObject=this,b},each:function(a){return r.each(this,a)},map:function(a){return this.pushStack(r.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(f.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(a<0?b:0);return this.pushStack(c>=0&&c0&&b-1 in a)}var x=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ha(),z=ha(),A=ha(),B=function(a,b){return a===b&&(l=!0),0},C={}.hasOwnProperty,D=[],E=D.pop,F=D.push,G=D.push,H=D.slice,I=function(a,b){for(var c=0,d=a.length;c+~]|"+K+")"+K+"*"),S=new RegExp("="+K+"*([^\\]'\"]*?)"+K+"*\\]","g"),T=new RegExp(N),U=new RegExp("^"+L+"$"),V={ID:new RegExp("^#("+L+")"),CLASS:new RegExp("^\\.("+L+")"),TAG:new RegExp("^("+L+"|[*])"),ATTR:new RegExp("^"+M),PSEUDO:new RegExp("^"+N),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+K+"*(even|odd|(([+-]|)(\\d*)n|)"+K+"*(?:([+-]|)"+K+"*(\\d+)|))"+K+"*\\)|)","i"),bool:new RegExp("^(?:"+J+")$","i"),needsContext:new RegExp("^"+K+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+K+"*((?:-\\d)?\\d*)"+K+"*\\)|)(?=[^-]|$)","i")},W=/^(?:input|select|textarea|button)$/i,X=/^h\d$/i,Y=/^[^{]+\{\s*\[native \w/,Z=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,$=/[+~]/,_=new RegExp("\\\\([\\da-f]{1,6}"+K+"?|("+K+")|.)","ig"),aa=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:d<0?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},ba=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,ca=function(a,b){return b?"\0"===a?"\ufffd":a.slice(0,-1)+"\\"+a.charCodeAt(a.length-1).toString(16)+" ":"\\"+a},da=function(){m()},ea=ta(function(a){return a.disabled===!0&&("form"in a||"label"in a)},{dir:"parentNode",next:"legend"});try{G.apply(D=H.call(v.childNodes),v.childNodes),D[v.childNodes.length].nodeType}catch(fa){G={apply:D.length?function(a,b){F.apply(a,H.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function ga(a,b,d,e){var f,h,j,k,l,o,r,s=b&&b.ownerDocument,w=b?b.nodeType:9;if(d=d||[],"string"!=typeof a||!a||1!==w&&9!==w&&11!==w)return d;if(!e&&((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,p)){if(11!==w&&(l=Z.exec(a)))if(f=l[1]){if(9===w){if(!(j=b.getElementById(f)))return d;if(j.id===f)return d.push(j),d}else if(s&&(j=s.getElementById(f))&&t(b,j)&&j.id===f)return d.push(j),d}else{if(l[2])return G.apply(d,b.getElementsByTagName(a)),d;if((f=l[3])&&c.getElementsByClassName&&b.getElementsByClassName)return G.apply(d,b.getElementsByClassName(f)),d}if(c.qsa&&!A[a+" "]&&(!q||!q.test(a))){if(1!==w)s=b,r=a;else if("object"!==b.nodeName.toLowerCase()){(k=b.getAttribute("id"))?k=k.replace(ba,ca):b.setAttribute("id",k=u),o=g(a),h=o.length;while(h--)o[h]="#"+k+" "+sa(o[h]);r=o.join(","),s=$.test(a)&&qa(b.parentNode)||b}if(r)try{return G.apply(d,s.querySelectorAll(r)),d}catch(x){}finally{k===u&&b.removeAttribute("id")}}}return i(a.replace(P,"$1"),b,d,e)}function ha(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ia(a){return a[u]=!0,a}function ja(a){var b=n.createElement("fieldset");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ka(a,b){var c=a.split("|"),e=c.length;while(e--)d.attrHandle[c[e]]=b}function la(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&a.sourceIndex-b.sourceIndex;if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function na(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function oa(a){return function(b){return"form"in b?b.parentNode&&b.disabled===!1?"label"in b?"label"in b.parentNode?b.parentNode.disabled===a:b.disabled===a:b.isDisabled===a||b.isDisabled!==!a&&ea(b)===a:b.disabled===a:"label"in b&&b.disabled===a}}function pa(a){return ia(function(b){return b=+b,ia(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function qa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=ga.support={},f=ga.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return!!b&&"HTML"!==b.nodeName},m=ga.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=n.documentElement,p=!f(n),v!==n&&(e=n.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",da,!1):e.attachEvent&&e.attachEvent("onunload",da)),c.attributes=ja(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ja(function(a){return a.appendChild(n.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=Y.test(n.getElementsByClassName),c.getById=ja(function(a){return o.appendChild(a).id=u,!n.getElementsByName||!n.getElementsByName(u).length}),c.getById?(d.filter.ID=function(a){var b=a.replace(_,aa);return function(a){return a.getAttribute("id")===b}},d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c?[c]:[]}}):(d.filter.ID=function(a){var b=a.replace(_,aa);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}},d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c,d,e,f=b.getElementById(a);if(f){if(c=f.getAttributeNode("id"),c&&c.value===a)return[f];e=b.getElementsByName(a),d=0;while(f=e[d++])if(c=f.getAttributeNode("id"),c&&c.value===a)return[f]}return[]}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){if("undefined"!=typeof b.getElementsByClassName&&p)return b.getElementsByClassName(a)},r=[],q=[],(c.qsa=Y.test(n.querySelectorAll))&&(ja(function(a){o.appendChild(a).innerHTML="",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+K+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+K+"*(?:value|"+J+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ja(function(a){a.innerHTML="";var b=n.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+K+"*[*^$|!~]?="),2!==a.querySelectorAll(":enabled").length&&q.push(":enabled",":disabled"),o.appendChild(a).disabled=!0,2!==a.querySelectorAll(":disabled").length&&q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=Y.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ja(function(a){c.disconnectedMatch=s.call(a,"*"),s.call(a,"[s!='']:x"),r.push("!=",N)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=Y.test(o.compareDocumentPosition),t=b||Y.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===n||a.ownerDocument===v&&t(v,a)?-1:b===n||b.ownerDocument===v&&t(v,b)?1:k?I(k,a)-I(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,g=[a],h=[b];if(!e||!f)return a===n?-1:b===n?1:e?-1:f?1:k?I(k,a)-I(k,b):0;if(e===f)return la(a,b);c=a;while(c=c.parentNode)g.unshift(c);c=b;while(c=c.parentNode)h.unshift(c);while(g[d]===h[d])d++;return d?la(g[d],h[d]):g[d]===v?-1:h[d]===v?1:0},n):n},ga.matches=function(a,b){return ga(a,null,null,b)},ga.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(S,"='$1']"),c.matchesSelector&&p&&!A[b+" "]&&(!r||!r.test(b))&&(!q||!q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return ga(b,n,null,[a]).length>0},ga.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},ga.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&C.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},ga.escape=function(a){return(a+"").replace(ba,ca)},ga.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},ga.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=ga.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=ga.selectors={cacheLength:50,createPseudo:ia,match:V,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(_,aa),a[3]=(a[3]||a[4]||a[5]||"").replace(_,aa),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||ga.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&ga.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return V.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&T.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(_,aa).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+K+")"+a+"("+K+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=ga.attr(d,a);return null==e?"!="===b:!b||(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(O," ")+" ").indexOf(c)>-1:"|="===b&&(e===c||e.slice(0,c.length+1)===c+"-"))}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h,t=!1;if(q){if(f){while(p){m=b;while(m=m[p])if(h?m.nodeName.toLowerCase()===r:1===m.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){m=q,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n&&j[2],m=n&&q.childNodes[n];while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if(1===m.nodeType&&++t&&m===b){k[a]=[w,n,t];break}}else if(s&&(m=b,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n),t===!1)while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if((h?m.nodeName.toLowerCase()===r:1===m.nodeType)&&++t&&(s&&(l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),k[a]=[w,t]),m===b))break;return t-=e,t===d||t%d===0&&t/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||ga.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ia(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=I(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ia(function(a){var b=[],c=[],d=h(a.replace(P,"$1"));return d[u]?ia(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ia(function(a){return function(b){return ga(a,b).length>0}}),contains:ia(function(a){return a=a.replace(_,aa),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ia(function(a){return U.test(a||"")||ga.error("unsupported lang: "+a),a=a.replace(_,aa).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:oa(!1),disabled:oa(!0),checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return X.test(a.nodeName)},input:function(a){return W.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:pa(function(){return[0]}),last:pa(function(a,b){return[b-1]}),eq:pa(function(a,b,c){return[c<0?c+b:c]}),even:pa(function(a,b){for(var c=0;c=0;)a.push(d);return a}),gt:pa(function(a,b,c){for(var d=c<0?c+b:c;++d1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function va(a,b,c){for(var d=0,e=b.length;d-1&&(f[j]=!(g[j]=l))}}else r=wa(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):G.apply(g,r)})}function ya(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=ta(function(a){return a===b},h,!0),l=ta(function(a){return I(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];i1&&ua(m),i>1&&sa(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(P,"$1"),c,i0,e=a.length>0,f=function(f,g,h,i,k){var l,o,q,r=0,s="0",t=f&&[],u=[],v=j,x=f||e&&d.find.TAG("*",k),y=w+=null==v?1:Math.random()||.1,z=x.length;for(k&&(j=g===n||g||k);s!==z&&null!=(l=x[s]);s++){if(e&&l){o=0,g||l.ownerDocument===n||(m(l),h=!p);while(q=a[o++])if(q(l,g||n,h)){i.push(l);break}k&&(w=y)}c&&((l=!q&&l)&&r--,f&&t.push(l))}if(r+=s,c&&s!==r){o=0;while(q=b[o++])q(t,u,g,h);if(f){if(r>0)while(s--)t[s]||u[s]||(u[s]=E.call(i));u=wa(u)}G.apply(i,u),k&&!f&&u.length>0&&r+b.length>1&&ga.uniqueSort(i)}return k&&(w=y,j=v),t};return c?ia(f):f}return h=ga.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=ya(b[c]),f[u]?d.push(f):e.push(f);f=A(a,za(e,d)),f.selector=a}return f},i=ga.select=function(a,b,c,e){var f,i,j,k,l,m="function"==typeof a&&a,n=!e&&g(a=m.selector||a);if(c=c||[],1===n.length){if(i=n[0]=n[0].slice(0),i.length>2&&"ID"===(j=i[0]).type&&9===b.nodeType&&p&&d.relative[i[1].type]){if(b=(d.find.ID(j.matches[0].replace(_,aa),b)||[])[0],!b)return c;m&&(b=b.parentNode),a=a.slice(i.shift().value.length)}f=V.needsContext.test(a)?0:i.length;while(f--){if(j=i[f],d.relative[k=j.type])break;if((l=d.find[k])&&(e=l(j.matches[0].replace(_,aa),$.test(i[0].type)&&qa(b.parentNode)||b))){if(i.splice(f,1),a=e.length&&sa(i),!a)return G.apply(c,e),c;break}}}return(m||h(a,n))(e,b,!p,c,!b||$.test(a)&&qa(b.parentNode)||b),c},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ja(function(a){return 1&a.compareDocumentPosition(n.createElement("fieldset"))}),ja(function(a){return a.innerHTML="","#"===a.firstChild.getAttribute("href")})||ka("type|href|height|width",function(a,b,c){if(!c)return a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ja(function(a){return a.innerHTML="",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ka("value",function(a,b,c){if(!c&&"input"===a.nodeName.toLowerCase())return a.defaultValue}),ja(function(a){return null==a.getAttribute("disabled")})||ka(J,function(a,b,c){var d;if(!c)return a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),ga}(a);r.find=x,r.expr=x.selectors,r.expr[":"]=r.expr.pseudos,r.uniqueSort=r.unique=x.uniqueSort,r.text=x.getText,r.isXMLDoc=x.isXML,r.contains=x.contains,r.escapeSelector=x.escape;var y=function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&r(a).is(c))break;d.push(a)}return d},z=function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c},A=r.expr.match.needsContext;function B(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()}var C=/^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i,D=/^.[^:#\[\.,]*$/;function E(a,b,c){return r.isFunction(b)?r.grep(a,function(a,d){return!!b.call(a,d,a)!==c}):b.nodeType?r.grep(a,function(a){return a===b!==c}):"string"!=typeof b?r.grep(a,function(a){return i.call(b,a)>-1!==c}):D.test(b)?r.filter(b,a,c):(b=r.filter(b,a),r.grep(a,function(a){return i.call(b,a)>-1!==c&&1===a.nodeType}))}r.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?r.find.matchesSelector(d,a)?[d]:[]:r.find.matches(a,r.grep(b,function(a){return 1===a.nodeType}))},r.fn.extend({find:function(a){var b,c,d=this.length,e=this;if("string"!=typeof a)return this.pushStack(r(a).filter(function(){for(b=0;b1?r.uniqueSort(c):c},filter:function(a){return this.pushStack(E(this,a||[],!1))},not:function(a){return this.pushStack(E(this,a||[],!0))},is:function(a){return!!E(this,"string"==typeof a&&A.test(a)?r(a):a||[],!1).length}});var F,G=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/,H=r.fn.init=function(a,b,c){var e,f;if(!a)return this;if(c=c||F,"string"==typeof a){if(e="<"===a[0]&&">"===a[a.length-1]&&a.length>=3?[null,a,null]:G.exec(a),!e||!e[1]&&b)return!b||b.jquery?(b||c).find(a):this.constructor(b).find(a);if(e[1]){if(b=b instanceof r?b[0]:b,r.merge(this,r.parseHTML(e[1],b&&b.nodeType?b.ownerDocument||b:d,!0)),C.test(e[1])&&r.isPlainObject(b))for(e in b)r.isFunction(this[e])?this[e](b[e]):this.attr(e,b[e]);return this}return f=d.getElementById(e[2]),f&&(this[0]=f,this.length=1),this}return a.nodeType?(this[0]=a,this.length=1,this):r.isFunction(a)?void 0!==c.ready?c.ready(a):a(r):r.makeArray(a,this)};H.prototype=r.fn,F=r(d);var I=/^(?:parents|prev(?:Until|All))/,J={children:!0,contents:!0,next:!0,prev:!0};r.fn.extend({has:function(a){var b=r(a,this),c=b.length;return this.filter(function(){for(var a=0;a-1:1===c.nodeType&&r.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?r.uniqueSort(f):f)},index:function(a){return a?"string"==typeof a?i.call(r(a),this[0]):i.call(this,a.jquery?a[0]:a):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(r.uniqueSort(r.merge(this.get(),r(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function K(a,b){while((a=a[b])&&1!==a.nodeType);return a}r.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return y(a,"parentNode")},parentsUntil:function(a,b,c){return y(a,"parentNode",c)},next:function(a){return K(a,"nextSibling")},prev:function(a){return K(a,"previousSibling")},nextAll:function(a){return y(a,"nextSibling")},prevAll:function(a){return y(a,"previousSibling")},nextUntil:function(a,b,c){return y(a,"nextSibling",c)},prevUntil:function(a,b,c){return y(a,"previousSibling",c)},siblings:function(a){return z((a.parentNode||{}).firstChild,a)},children:function(a){return z(a.firstChild)},contents:function(a){return B(a,"iframe")?a.contentDocument:(B(a,"template")&&(a=a.content||a),r.merge([],a.childNodes))}},function(a,b){r.fn[a]=function(c,d){var e=r.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=r.filter(d,e)),this.length>1&&(J[a]||r.uniqueSort(e),I.test(a)&&e.reverse()),this.pushStack(e)}});var L=/[^\x20\t\r\n\f]+/g;function M(a){var b={};return r.each(a.match(L)||[],function(a,c){b[c]=!0}),b}r.Callbacks=function(a){a="string"==typeof a?M(a):r.extend({},a);var b,c,d,e,f=[],g=[],h=-1,i=function(){for(e=e||a.once,d=b=!0;g.length;h=-1){c=g.shift();while(++h-1)f.splice(c,1),c<=h&&h--}),this},has:function(a){return a?r.inArray(a,f)>-1:f.length>0},empty:function(){return f&&(f=[]),this},disable:function(){return e=g=[],f=c="",this},disabled:function(){return!f},lock:function(){return e=g=[],c||b||(f=c=""),this},locked:function(){return!!e},fireWith:function(a,c){return e||(c=c||[],c=[a,c.slice?c.slice():c],g.push(c),b||i()),this},fire:function(){return j.fireWith(this,arguments),this},fired:function(){return!!d}};return j};function N(a){return a}function O(a){throw a}function P(a,b,c,d){var e;try{a&&r.isFunction(e=a.promise)?e.call(a).done(b).fail(c):a&&r.isFunction(e=a.then)?e.call(a,b,c):b.apply(void 0,[a].slice(d))}catch(a){c.apply(void 0,[a])}}r.extend({Deferred:function(b){var c=[["notify","progress",r.Callbacks("memory"),r.Callbacks("memory"),2],["resolve","done",r.Callbacks("once memory"),r.Callbacks("once memory"),0,"resolved"],["reject","fail",r.Callbacks("once memory"),r.Callbacks("once memory"),1,"rejected"]],d="pending",e={state:function(){return d},always:function(){return f.done(arguments).fail(arguments),this},"catch":function(a){return e.then(null,a)},pipe:function(){var a=arguments;return r.Deferred(function(b){r.each(c,function(c,d){var e=r.isFunction(a[d[4]])&&a[d[4]];f[d[1]](function(){var a=e&&e.apply(this,arguments);a&&r.isFunction(a.promise)?a.promise().progress(b.notify).done(b.resolve).fail(b.reject):b[d[0]+"With"](this,e?[a]:arguments)})}),a=null}).promise()},then:function(b,d,e){var f=0;function g(b,c,d,e){return function(){var h=this,i=arguments,j=function(){var a,j;if(!(b=f&&(d!==O&&(h=void 0,i=[a]),c.rejectWith(h,i))}};b?k():(r.Deferred.getStackHook&&(k.stackTrace=r.Deferred.getStackHook()),a.setTimeout(k))}}return r.Deferred(function(a){c[0][3].add(g(0,a,r.isFunction(e)?e:N,a.notifyWith)),c[1][3].add(g(0,a,r.isFunction(b)?b:N)),c[2][3].add(g(0,a,r.isFunction(d)?d:O))}).promise()},promise:function(a){return null!=a?r.extend(a,e):e}},f={};return r.each(c,function(a,b){var g=b[2],h=b[5];e[b[1]]=g.add,h&&g.add(function(){d=h},c[3-a][2].disable,c[0][2].lock),g.add(b[3].fire),f[b[0]]=function(){return f[b[0]+"With"](this===f?void 0:this,arguments),this},f[b[0]+"With"]=g.fireWith}),e.promise(f),b&&b.call(f,f),f},when:function(a){var b=arguments.length,c=b,d=Array(c),e=f.call(arguments),g=r.Deferred(),h=function(a){return function(c){d[a]=this,e[a]=arguments.length>1?f.call(arguments):c,--b||g.resolveWith(d,e)}};if(b<=1&&(P(a,g.done(h(c)).resolve,g.reject,!b),"pending"===g.state()||r.isFunction(e[c]&&e[c].then)))return g.then();while(c--)P(e[c],h(c),g.reject);return g.promise()}});var Q=/^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/;r.Deferred.exceptionHook=function(b,c){a.console&&a.console.warn&&b&&Q.test(b.name)&&a.console.warn("jQuery.Deferred exception: "+b.message,b.stack,c)},r.readyException=function(b){a.setTimeout(function(){throw b})};var R=r.Deferred();r.fn.ready=function(a){return R.then(a)["catch"](function(a){r.readyException(a)}),this},r.extend({isReady:!1,readyWait:1,ready:function(a){(a===!0?--r.readyWait:r.isReady)||(r.isReady=!0,a!==!0&&--r.readyWait>0||R.resolveWith(d,[r]))}}),r.ready.then=R.then;function S(){d.removeEventListener("DOMContentLoaded",S),
-a.removeEventListener("load",S),r.ready()}"complete"===d.readyState||"loading"!==d.readyState&&!d.documentElement.doScroll?a.setTimeout(r.ready):(d.addEventListener("DOMContentLoaded",S),a.addEventListener("load",S));var T=function(a,b,c,d,e,f,g){var h=0,i=a.length,j=null==c;if("object"===r.type(c)){e=!0;for(h in c)T(a,b,h,c[h],!0,f,g)}else if(void 0!==d&&(e=!0,r.isFunction(d)||(g=!0),j&&(g?(b.call(a,d),b=null):(j=b,b=function(a,b,c){return j.call(r(a),c)})),b))for(;h1,null,!0)},removeData:function(a){return this.each(function(){X.remove(this,a)})}}),r.extend({queue:function(a,b,c){var d;if(a)return b=(b||"fx")+"queue",d=W.get(a,b),c&&(!d||Array.isArray(c)?d=W.access(a,b,r.makeArray(c)):d.push(c)),d||[]},dequeue:function(a,b){b=b||"fx";var c=r.queue(a,b),d=c.length,e=c.shift(),f=r._queueHooks(a,b),g=function(){r.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return W.get(a,c)||W.access(a,c,{empty:r.Callbacks("once memory").add(function(){W.remove(a,[b+"queue",c])})})}}),r.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.length\x20\t\r\n\f]+)/i,la=/^$|\/(?:java|ecma)script/i,ma={option:[1,""],thead:[1,""],col:[2,""],tr:[2,""],td:[3,""],_default:[0,"",""]};ma.optgroup=ma.option,ma.tbody=ma.tfoot=ma.colgroup=ma.caption=ma.thead,ma.th=ma.td;function na(a,b){var c;return c="undefined"!=typeof a.getElementsByTagName?a.getElementsByTagName(b||"*"):"undefined"!=typeof a.querySelectorAll?a.querySelectorAll(b||"*"):[],void 0===b||b&&B(a,b)?r.merge([a],c):c}function oa(a,b){for(var c=0,d=a.length;c-1)e&&e.push(f);else if(j=r.contains(f.ownerDocument,f),g=na(l.appendChild(f),"script"),j&&oa(g),c){k=0;while(f=g[k++])la.test(f.type||"")&&c.push(f)}return l}!function(){var a=d.createDocumentFragment(),b=a.appendChild(d.createElement("div")),c=d.createElement("input");c.setAttribute("type","radio"),c.setAttribute("checked","checked"),c.setAttribute("name","t"),b.appendChild(c),o.checkClone=b.cloneNode(!0).cloneNode(!0).lastChild.checked,b.innerHTML="",o.noCloneChecked=!!b.cloneNode(!0).lastChild.defaultValue}();var ra=d.documentElement,sa=/^key/,ta=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,ua=/^([^.]*)(?:\.(.+)|)/;function va(){return!0}function wa(){return!1}function xa(){try{return d.activeElement}catch(a){}}function ya(a,b,c,d,e,f){var g,h;if("object"==typeof b){"string"!=typeof c&&(d=d||c,c=void 0);for(h in b)ya(a,h,c,d,b[h],f);return a}if(null==d&&null==e?(e=c,d=c=void 0):null==e&&("string"==typeof c?(e=d,d=void 0):(e=d,d=c,c=void 0)),e===!1)e=wa;else if(!e)return a;return 1===f&&(g=e,e=function(a){return r().off(a),g.apply(this,arguments)},e.guid=g.guid||(g.guid=r.guid++)),a.each(function(){r.event.add(this,b,e,d,c)})}r.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,n,o,p,q=W.get(a);if(q){c.handler&&(f=c,c=f.handler,e=f.selector),e&&r.find.matchesSelector(ra,e),c.guid||(c.guid=r.guid++),(i=q.events)||(i=q.events={}),(g=q.handle)||(g=q.handle=function(b){return"undefined"!=typeof r&&r.event.triggered!==b.type?r.event.dispatch.apply(a,arguments):void 0}),b=(b||"").match(L)||[""],j=b.length;while(j--)h=ua.exec(b[j])||[],n=p=h[1],o=(h[2]||"").split(".").sort(),n&&(l=r.event.special[n]||{},n=(e?l.delegateType:l.bindType)||n,l=r.event.special[n]||{},k=r.extend({type:n,origType:p,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&r.expr.match.needsContext.test(e),namespace:o.join(".")},f),(m=i[n])||(m=i[n]=[],m.delegateCount=0,l.setup&&l.setup.call(a,d,o,g)!==!1||a.addEventListener&&a.addEventListener(n,g)),l.add&&(l.add.call(a,k),k.handler.guid||(k.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,k):m.push(k),r.event.global[n]=!0)}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,n,o,p,q=W.hasData(a)&&W.get(a);if(q&&(i=q.events)){b=(b||"").match(L)||[""],j=b.length;while(j--)if(h=ua.exec(b[j])||[],n=p=h[1],o=(h[2]||"").split(".").sort(),n){l=r.event.special[n]||{},n=(d?l.delegateType:l.bindType)||n,m=i[n]||[],h=h[2]&&new RegExp("(^|\\.)"+o.join("\\.(?:.*\\.|)")+"(\\.|$)"),g=f=m.length;while(f--)k=m[f],!e&&p!==k.origType||c&&c.guid!==k.guid||h&&!h.test(k.namespace)||d&&d!==k.selector&&("**"!==d||!k.selector)||(m.splice(f,1),k.selector&&m.delegateCount--,l.remove&&l.remove.call(a,k));g&&!m.length&&(l.teardown&&l.teardown.call(a,o,q.handle)!==!1||r.removeEvent(a,n,q.handle),delete i[n])}else for(n in i)r.event.remove(a,n+b[j],c,d,!0);r.isEmptyObject(i)&&W.remove(a,"handle events")}},dispatch:function(a){var b=r.event.fix(a),c,d,e,f,g,h,i=new Array(arguments.length),j=(W.get(this,"events")||{})[b.type]||[],k=r.event.special[b.type]||{};for(i[0]=b,c=1;c=1))for(;j!==this;j=j.parentNode||this)if(1===j.nodeType&&("click"!==a.type||j.disabled!==!0)){for(f=[],g={},c=0;c-1:r.find(e,this,null,[j]).length),g[e]&&f.push(d);f.length&&h.push({elem:j,handlers:f})}return j=this,i\x20\t\r\n\f]*)[^>]*)\/>/gi,Aa=/