Compare commits
	
		
			2 Commits 
		
	
	
		
			main
			...
			drew/sql-i
		
	
	| Author | SHA1 | Date | 
|---|---|---|
| 
							
							
								
									
								
								 | 
						a03cf82681 | 1 year ago | 
| 
							
							
								
									
								
								 | 
						fc397d83c2 | 1 year ago | 
@ -0,0 +1,3 @@
 | 
				
			|||||||
 | 
					FLUX_DATABASE_DRIVER=sqlite3
 | 
				
			||||||
 | 
					FLUX_DATABASE_DSN=./flux-local.db
 | 
				
			||||||
 | 
					FLUX_MIGRATIONS_PATH=./migrations
 | 
				
			||||||
@ -0,0 +1,40 @@
 | 
				
			|||||||
 | 
					// Entry point for applying database migrations for flux-feed application
 | 
				
			||||||
 | 
					package main
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"database/sql"
 | 
				
			||||||
 | 
						"log"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"git.runcible.io/androiddrew/flux-feed/config"
 | 
				
			||||||
 | 
						"github.com/golang-migrate/migrate/v4"
 | 
				
			||||||
 | 
						"github.com/golang-migrate/migrate/v4/database/sqlite3"
 | 
				
			||||||
 | 
						_ "github.com/golang-migrate/migrate/v4/source/file"
 | 
				
			||||||
 | 
						_ "github.com/mattn/go-sqlite3"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func main() {
 | 
				
			||||||
 | 
						cfg := config.New()
 | 
				
			||||||
 | 
						db, err := sql.Open(cfg.DatabaseDriver, cfg.DatabaseDSN)
 | 
				
			||||||
 | 
						defer db.Close()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						err = db.Ping()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							log.Fatal(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						log.Printf("Connected to Database: %s", cfg.DatabaseDSN)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						driver, err := sqlite3.WithInstance(db, &sqlite3.Config{})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						log.Printf("Using migrations path: %s", cfg.MigrationsPath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						m, err := migrate.NewWithDatabaseInstance(
 | 
				
			||||||
 | 
							cfg.MigrationsPath, cfg.DatabaseDriver, driver)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							log.Fatal(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						log.Printf("Migrating: %s", cfg.DatabaseDSN)
 | 
				
			||||||
 | 
						err = m.Up()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							log.Fatal(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -0,0 +1,26 @@
 | 
				
			|||||||
 | 
					// Package config provides configuration structures and methods to load
 | 
				
			||||||
 | 
					// environment variables for the application.
 | 
				
			||||||
 | 
					package config
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"log"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/joho/godotenv"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Config holds the configuration for the application, including the database settings. It contains multiple embedded configuration structs.
 | 
				
			||||||
 | 
					type Config struct {
 | 
				
			||||||
 | 
						Database
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// New loads environment variables and returns a new configured Config instance.
 | 
				
			||||||
 | 
					func New() *Config {
 | 
				
			||||||
 | 
						err := godotenv.Load()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							log.Println(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return &Config{
 | 
				
			||||||
 | 
							Database: DataStore(),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -0,0 +1,18 @@
 | 
				
			|||||||
 | 
					package config
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import "github.com/kelseyhightower/envconfig"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Database holds the configuration for the database connection.
 | 
				
			||||||
 | 
					type Database struct {
 | 
				
			||||||
 | 
						DatabaseDriver string `split_words:"true"`
 | 
				
			||||||
 | 
						DatabaseDSN    string `split_words:"true"`
 | 
				
			||||||
 | 
						MigrationsPath string `default:"head" split_words:"true"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DataStore processes environment variables and returns a configured Database configuration struct.
 | 
				
			||||||
 | 
					func DataStore() Database {
 | 
				
			||||||
 | 
						var db Database
 | 
				
			||||||
 | 
						envconfig.MustProcess("flux", &db)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return db
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -0,0 +1,33 @@
 | 
				
			|||||||
 | 
					// Package database provides utility functions for connecting to and
 | 
				
			||||||
 | 
					// interacting with various database backends.
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// This package supports multiple database drivers including:
 | 
				
			||||||
 | 
					// - sqlite3
 | 
				
			||||||
 | 
					// - others coming soon...maybe
 | 
				
			||||||
 | 
					package database
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"log"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"git.runcible.io/androiddrew/flux-feed/config"
 | 
				
			||||||
 | 
						"github.com/jmoiron/sqlx"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewSqlx configures a new sqlx.DB from application config
 | 
				
			||||||
 | 
					func NewSqlx(cfg config.Database) *sqlx.DB {
 | 
				
			||||||
 | 
						var dsn string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// TODO add additional database driver support
 | 
				
			||||||
 | 
						switch cfg.DatabaseDriver {
 | 
				
			||||||
 | 
						case "sqlite3":
 | 
				
			||||||
 | 
							dsn = cfg.DatabaseDSN
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							log.Fatal("Must choose a database driver")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						db := sqlx.MustConnect(cfg.DatabaseDriver, dsn)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return db
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -1,3 +1,17 @@
 | 
				
			|||||||
module git.runcible.io/androiddrew/fluxfeed
 | 
					module git.runcible.io/androiddrew/flux-feed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
go 1.22.5
 | 
					go 1.22.5
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					require (
 | 
				
			||||||
 | 
						github.com/golang-migrate/migrate/v4 v4.17.1
 | 
				
			||||||
 | 
						github.com/jmoiron/sqlx v1.4.0
 | 
				
			||||||
 | 
						github.com/joho/godotenv v1.5.1
 | 
				
			||||||
 | 
						github.com/kelseyhightower/envconfig v1.4.0
 | 
				
			||||||
 | 
						github.com/mattn/go-sqlite3 v1.14.22
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					require (
 | 
				
			||||||
 | 
						github.com/hashicorp/errwrap v1.1.0 // indirect
 | 
				
			||||||
 | 
						github.com/hashicorp/go-multierror v1.1.1 // indirect
 | 
				
			||||||
 | 
						go.uber.org/atomic v1.7.0 // indirect
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
				
			|||||||
@ -0,0 +1,34 @@
 | 
				
			|||||||
 | 
					filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA=
 | 
				
			||||||
 | 
					filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4=
 | 
				
			||||||
 | 
					github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
				
			||||||
 | 
					github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
 | 
				
			||||||
 | 
					github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
				
			||||||
 | 
					github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y=
 | 
				
			||||||
 | 
					github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg=
 | 
				
			||||||
 | 
					github.com/golang-migrate/migrate/v4 v4.17.1 h1:4zQ6iqL6t6AiItphxJctQb3cFqWiSpMnX7wLTPnnYO4=
 | 
				
			||||||
 | 
					github.com/golang-migrate/migrate/v4 v4.17.1/go.mod h1:m8hinFyWBn0SA4QKHuKh175Pm9wjmxj3S2Mia7dbXzM=
 | 
				
			||||||
 | 
					github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
 | 
				
			||||||
 | 
					github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I=
 | 
				
			||||||
 | 
					github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
 | 
				
			||||||
 | 
					github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo=
 | 
				
			||||||
 | 
					github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM=
 | 
				
			||||||
 | 
					github.com/jmoiron/sqlx v1.4.0 h1:1PLqN7S1UYp5t4SrVVnt4nUVNemrDAtxlulVe+Qgm3o=
 | 
				
			||||||
 | 
					github.com/jmoiron/sqlx v1.4.0/go.mod h1:ZrZ7UsYB/weZdl2Bxg6jCRO9c3YHl8r3ahlKmRT4JLY=
 | 
				
			||||||
 | 
					github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
 | 
				
			||||||
 | 
					github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
 | 
				
			||||||
 | 
					github.com/kelseyhightower/envconfig v1.4.0 h1:Im6hONhd3pLkfDFsbRgu68RDNkGF1r3dvMUtDTo2cv8=
 | 
				
			||||||
 | 
					github.com/kelseyhightower/envconfig v1.4.0/go.mod h1:cccZRl6mQpaq41TPp5QxidR+Sa3axMbJDNb//FQX6Gg=
 | 
				
			||||||
 | 
					github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
 | 
				
			||||||
 | 
					github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
 | 
				
			||||||
 | 
					github.com/mattn/go-sqlite3 v1.14.22 h1:2gZY6PC6kBnID23Tichd1K+Z0oS6nE/XwU+Vz/5o4kU=
 | 
				
			||||||
 | 
					github.com/mattn/go-sqlite3 v1.14.22/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
 | 
				
			||||||
 | 
					github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
 | 
				
			||||||
 | 
					github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
				
			||||||
 | 
					github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
 | 
				
			||||||
 | 
					github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
 | 
				
			||||||
 | 
					github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY=
 | 
				
			||||||
 | 
					github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
 | 
				
			||||||
 | 
					go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
 | 
				
			||||||
 | 
					go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
 | 
				
			||||||
 | 
					gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
 | 
				
			||||||
 | 
					gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
				
			||||||
@ -1,17 +0,0 @@
 | 
				
			|||||||
package main
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import (
 | 
					 | 
				
			||||||
	"fmt"
 | 
					 | 
				
			||||||
	"io"
 | 
					 | 
				
			||||||
	"os"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const FluxGreeting string = "Welcome to Flux Feed\n"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
func HelloFluxFeed(out io.Writer) {
 | 
					 | 
				
			||||||
	fmt.Fprint(out, FluxGreeting)
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
func main() {
 | 
					 | 
				
			||||||
	HelloFluxFeed(os.Stdout)
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,18 +0,0 @@
 | 
				
			|||||||
package main
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import (
 | 
					 | 
				
			||||||
	"bytes"
 | 
					 | 
				
			||||||
	"testing"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
func TestCliGreeting(t *testing.T) {
 | 
					 | 
				
			||||||
	buffer := &bytes.Buffer{}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	HelloFluxFeed(buffer)
 | 
					 | 
				
			||||||
	got := buffer.String()
 | 
					 | 
				
			||||||
	want := FluxGreeting
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if got != want {
 | 
					 | 
				
			||||||
		t.Errorf("Got %s but wanted %s", got, want)
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -0,0 +1,3 @@
 | 
				
			|||||||
 | 
					DROP TABLE IF EXISTS user;
 | 
				
			||||||
 | 
					DROP TABLE IF EXISTS feed;
 | 
				
			||||||
 | 
					DROP TABLE IF EXISTS entry;
 | 
				
			||||||
@ -0,0 +1,72 @@
 | 
				
			|||||||
 | 
					PRAGMA foreign_keys=1;
 | 
				
			||||||
 | 
					CREATE TABLE user (
 | 
				
			||||||
 | 
					    id INTEGER PRIMARY KEY AUTOINCREMENT, 
 | 
				
			||||||
 | 
					    email TEXT UNIQUE NOT NULL, 
 | 
				
			||||||
 | 
					    password TEXT NOT NULL, 
 | 
				
			||||||
 | 
					    kindle_email TEXT
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CREATE TABLE feed (
 | 
				
			||||||
 | 
					    id INTEGER PRIMARY KEY AUTOINCREMENT, 
 | 
				
			||||||
 | 
					    user_id INTEGER UNIQUE NOT NULL, 
 | 
				
			||||||
 | 
					    url TEXT, 
 | 
				
			||||||
 | 
					    type TEXT NOT NULL, 
 | 
				
			||||||
 | 
					    name TEXT UNIQUE, 
 | 
				
			||||||
 | 
					    icon_url TEXT, 
 | 
				
			||||||
 | 
					    created TIMESTAMP NOT NULL, 
 | 
				
			||||||
 | 
					    updated TIMESTAMP NOT NULL, 
 | 
				
			||||||
 | 
					    last_fetch TIMESTAMP, 
 | 
				
			||||||
 | 
					    raw_data TEXT, 
 | 
				
			||||||
 | 
					    folder TEXT, 
 | 
				
			||||||
 | 
					    etag TEXT, 
 | 
				
			||||||
 | 
					    modified_header TEXT, 
 | 
				
			||||||
 | 
					    filters TEXT, 
 | 
				
			||||||
 | 
					    FOREIGN KEY(user_id) REFERENCES user(id)
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CREATE TABLE entry (
 | 
				
			||||||
 | 
					    id INTEGER PRIMARY KEY AUTOINCREMENT, 
 | 
				
			||||||
 | 
					    feed_id INTEGER UNIQUE, 
 | 
				
			||||||
 | 
					    user_id INTEGER NOT NULL, 
 | 
				
			||||||
 | 
					    remote_id TEXT UNIQUE NOT NULL, 
 | 
				
			||||||
 | 
					    title TEXT, 
 | 
				
			||||||
 | 
					    username TEXT, 
 | 
				
			||||||
 | 
					    user_url TEXT, 
 | 
				
			||||||
 | 
					    display_name TEXT, 
 | 
				
			||||||
 | 
					    avatar_url TEXT, 
 | 
				
			||||||
 | 
					    content_short TEXT, 
 | 
				
			||||||
 | 
					    content_full TEXT, 
 | 
				
			||||||
 | 
					    target_url TEXT, 
 | 
				
			||||||
 | 
					    content_url TEXT, 
 | 
				
			||||||
 | 
					    comments_url TEXT, 
 | 
				
			||||||
 | 
					    media_url TEXT, 
 | 
				
			||||||
 | 
					    created TIMESTAMP NOT NULL, 
 | 
				
			||||||
 | 
					    updated TIMESTAMP NOT NULL, 
 | 
				
			||||||
 | 
					    display_date TIMESTAMP NOT NULL, 
 | 
				
			||||||
 | 
					    sort_date TIMESTAMP NOT NULL, 
 | 
				
			||||||
 | 
					    viewed TIMESTAMP, 
 | 
				
			||||||
 | 
					    favorited TIMESTAMP, 
 | 
				
			||||||
 | 
					    pinned TIMESTAMP, 
 | 
				
			||||||
 | 
					    sent_to_kindle TIMESTAMP, 
 | 
				
			||||||
 | 
					    raw_data TEXT, 
 | 
				
			||||||
 | 
					    header TEXT, 
 | 
				
			||||||
 | 
					    icon_url TEXT, 
 | 
				
			||||||
 | 
					    FOREIGN KEY(feed_id) REFERENCES feed(id), 
 | 
				
			||||||
 | 
					    FOREIGN KEY(user_id) REFERENCES user(id)
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CREATE INDEX ix_feed_folder ON feed(folder);
 | 
				
			||||||
 | 
					CREATE INDEX ix_feed_user_id ON feed(user_id);
 | 
				
			||||||
 | 
					CREATE INDEX ix_name_user ON feed(user_id, name);
 | 
				
			||||||
 | 
					CREATE INDEX ix_feed_created ON feed(created);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_favorited ON entry(favorited);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_created ON entry(created);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_sort_date ON entry(sort_date);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_sent_to_kindle ON entry(sent_to_kindle);
 | 
				
			||||||
 | 
					CREATE INDEX entry_sort_ts ON entry(sort_date DESC);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_viewed ON entry(viewed);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_pinned ON entry(pinned);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_user_id ON entry(user_id);
 | 
				
			||||||
 | 
					CREATE INDEX ix_entry_username ON entry(username);
 | 
				
			||||||
					Loading…
					
					
				
		Reference in New Issue