Compare commits

...

3 Commits

Author SHA1 Message Date
bd5737ad7a Web edit sources 2025-02-12 10:04:58 -08:00
7361fd4600 Combine action add/edit with on conflict replace 2025-02-12 09:49:27 -08:00
7c8b4ee3a3 Disable welcome item since it broke some tests
This should move to a function that adds the item through the normal methods and runs on db init
2025-02-12 09:38:10 -08:00
19 changed files with 312 additions and 38 deletions

View File

@ -110,11 +110,11 @@ Parity features
* [x] crontab integration * [x] crontab integration
* [ ] source batching * [ ] source batching
* [x] add item from web * [x] add item from web
* [ ] web edit channels * [x] web edit channels
* web edit sources * web edit sources
* [ ] edit action argv * [x] edit action argv
* [ ] edit source envs * [x] edit source envs
* [ ] edit source crontab * [x] edit source crontab
* [x] Nix build * [x] Nix build
* [ ] NixOS module * [ ] NixOS module
* [ ] NixOS vm demo * [ ] NixOS vm demo

View File

@ -27,6 +27,8 @@ func init() {
actionAddCmd.MarkFlagRequired("action") actionAddCmd.MarkFlagRequired("action")
} }
// TODO: This is a duplicate of `action edit`, the action CLI should be simplified
func actionAdd(source string, action string, argv []string) { func actionAdd(source string, action string, argv []string) {
if source == "" { if source == "" {
log.Fatal("error: --source is empty") log.Fatal("error: --source is empty")
@ -40,7 +42,7 @@ func actionAdd(source string, action string, argv []string) {
db := openAndMigrateDb() db := openAndMigrateDb()
err := core.AddAction(db, source, action, argv) err := core.SetAction(db, source, action, argv)
if err != nil { if err != nil {
log.Fatalf("error: failed to add action: %v", err) log.Fatalf("error: failed to add action: %v", err)
} }

View File

@ -40,7 +40,7 @@ func actionEdit(source string, action string, argv []string) {
db := openAndMigrateDb() db := openAndMigrateDb()
err := core.UpdateAction(db, source, action, argv) err := core.SetAction(db, source, action, argv)
if err != nil { if err != nil {
log.Fatalf("error: failed to update action: %v", err) log.Fatalf("error: failed to update action: %v", err)
} }

View File

@ -16,7 +16,7 @@ func (a *argList) Scan(value interface{}) error {
return json.Unmarshal([]byte(value.(string)), a) return json.Unmarshal([]byte(value.(string)), a)
} }
func AddAction(db DB, source string, name string, argv []string) error { func SetAction(db DB, source string, name string, argv []string) error {
_, err := db.Exec(` _, err := db.Exec(`
insert into actions (source, name, argv) insert into actions (source, name, argv)
values (?, ?, jsonb(?)) values (?, ?, jsonb(?))
@ -24,15 +24,6 @@ func AddAction(db DB, source string, name string, argv []string) error {
return err return err
} }
func UpdateAction(db DB, source string, name string, argv []string) error {
_, err := db.Exec(`
update actions
set argv = jsonb(?)
where source = ? and name = ?
`, argList(argv), source, name)
return err
}
func GetActionsForSource(db DB, source string) ([]string, error) { func GetActionsForSource(db DB, source string) ([]string, error) {
rows, err := db.Query(` rows, err := db.Query(`
select name select name

View File

@ -7,7 +7,7 @@ import (
func TestActionCreate(t *testing.T) { func TestActionCreate(t *testing.T) {
db := EphemeralDb(t) db := EphemeralDb(t)
if err := AddAction(db, "test", "hello", []string{"echo", "hello"}); err == nil { if err := SetAction(db, "test", "hello", []string{"echo", "hello"}); err == nil {
t.Fatal("Action created for nonexistent source") t.Fatal("Action created for nonexistent source")
} }
@ -15,13 +15,13 @@ func TestActionCreate(t *testing.T) {
t.Fatal(err) t.Fatal(err)
} }
if err := AddAction(db, "test", "hello", []string{"echo", "hello"}); err != nil { if err := SetAction(db, "test", "hello", []string{"echo", "hello"}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := AddAction(db, "test", "goodbye", []string{"exit", "1"}); err != nil { if err := SetAction(db, "test", "goodbye", []string{"exit", "1"}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := UpdateAction(db, "test", "goodbye", []string{"echo", "goodbye"}); err != nil { if err := SetAction(db, "test", "goodbye", []string{"echo", "goodbye"}); err != nil {
t.Fatal(err) t.Fatal(err)
} }

View File

@ -36,7 +36,7 @@ func TestChannel(t *testing.T) {
if err != nil { if err != nil {
t.Fatalf("failed to get sources in channel: %v", err) t.Fatalf("failed to get sources in channel: %v", err)
} }
if len(sources) != 1 || sources["channel"][0] != "two" { if len(sources["channel"]) != 1 || sources["channel"][0] != "two" {
t.Fatalf("unexpected sources in channel after deletion: %v", sources) t.Fatalf("unexpected sources in channel after deletion: %v", sources)
} }
if err := AddSourceToChannel(db, "channel", "one"); err != nil { if err := AddSourceToChannel(db, "channel", "one"); err != nil {

View File

@ -104,7 +104,7 @@ func TestDeleteSourceCascade(t *testing.T) {
t.Fatalf("failed to get active items: %v", err) t.Fatalf("failed to get active items: %v", err)
} }
if len(items) != 2 { if len(items) != 2 {
t.Fatal("Expected 2 items") t.Fatalf("Expected 2 items, got %d", len(items))
} }
if err := DeleteSource(db, "source1"); err != nil { if err := DeleteSource(db, "source1"); err != nil {

View File

@ -5,7 +5,7 @@ import (
"strings" "strings"
) )
func GetEnvs(db DB, source string) ([]string, error) { func GetEnvs(db DB, source string) (envs []string, err error) {
rows, err := db.Query(` rows, err := db.Query(`
select name, value select name, value
from envs from envs
@ -15,19 +15,43 @@ func GetEnvs(db DB, source string) ([]string, error) {
return nil, err return nil, err
} }
defer rows.Close() defer rows.Close()
var envs []string
for rows.Next() { for rows.Next() {
var name string var name string
var value string var value string
if err := rows.Scan(&name, &value); err != nil { if err = rows.Scan(&name, &value); err != nil {
return nil, err return nil, err
} }
envs = append(envs, fmt.Sprintf("%s=%s", name, value)) envs = append(envs, fmt.Sprintf("%s=%s", name, value))
} }
if err := rows.Err(); err != nil { if err = rows.Err(); err != nil {
return nil, err return nil, err
} }
return envs, nil return
}
func GetEnvsMap(db DB, source string) (envs map[string]string, err error) {
rows, err := db.Query(`
select name, value
from envs
where source = ?
`, source)
if err != nil {
return nil, err
}
defer rows.Close()
envs = make(map[string]string)
for rows.Next() {
var name string
var value string
if err = rows.Scan(&name, &value); err != nil {
return nil, err
}
envs[name] = value
}
if err = rows.Err(); err != nil {
return nil, err
}
return
} }
func SetEnvs(db DB, source string, envs []string) error { func SetEnvs(db DB, source string, envs []string) error {

View File

@ -141,7 +141,7 @@ func TestOnCreateAction(t *testing.T) {
if err := AddSource(db, "test"); err != nil { if err := AddSource(db, "test"); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := AddAction(db, "test", "on_create", []string{"true"}); err != nil { if err := SetAction(db, "test", "on_create", []string{"true"}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -164,7 +164,7 @@ func TestOnCreateAction(t *testing.T) {
onCreate := func(argv []string) { onCreate := func(argv []string) {
t.Helper() t.Helper()
if err := UpdateAction(db, "test", "on_create", argv); err != nil { if err := SetAction(db, "test", "on_create", argv); err != nil {
t.Fatal(err) t.Fatal(err)
} }
} }

View File

@ -7,7 +7,7 @@ create table actions(
source text not null, source text not null,
name text not null, name text not null,
argv blob not null, argv blob not null,
primary key (source, name), unique (source, name) on conflict replace,
foreign key (source) references sources (name) on delete cascade foreign key (source) references sources (name) on delete cascade
) strict; ) strict;
create table envs( create table envs(
@ -52,10 +52,10 @@ create table sessions(
) strict; ) strict;
-- user introduction -- user introduction
insert into sources (name) values ('default'); -- insert into sources (name) values ('default');
insert into channels (name, source) values ('home', 'default'); -- insert into channels (name, source) values ('home', 'default');
insert into actions (source, name, argv) values ('default', 'fetch', jsonb('["true"]')); -- insert into actions (source, name, argv) values ('default', 'fetch', jsonb('["true"]'));
insert into items (source, id, active, title, author, body, link, time, ttl, ttd, tts, action) -- insert into items (source, id, active, title, author, body, link, time, ttl, ttd, tts, action)
values ('default', 'welcome', 1, 'Welcome to intake!', 'intake', -- values ('default', 'welcome', 1, 'Welcome to intake!', 'intake',
'<p>Click the "X" button on this item to deactivate it and hide it from the feed.</p>', -- '<p>Click the "X" button on this item to deactivate it and hide it from the feed.</p>',
'', 1, 0, 0, 0, jsonb('null')) -- '', 1, 0, 0, 0, jsonb('null'))

54
web/html/editSource.html Normal file
View File

@ -0,0 +1,54 @@
{{ define "title" }}{{ .Name }} - Intake{{ end }}
{{ define "content" -}}
<nav class="center">
<span class="feed-controls">
<a href="/">Home</a>
</span>
</nav>
<nav>
<span class="feed-controls">Edit source {{ .Name }}</span>
<p>
<form method="post">
<label for="name">Name:</label>
<input type="text" name="name" placeholder="Source name" value="{{ .Name }}" readonly>
<input type="submit" value="Update" disabled>
</form>
</p>
<p>Environment:</p>
<table id="envvars">{{ range $name, $value := .Envs }}
<tr>
<td><input type="text" value="{{ $name }}" disabled></td>
<td><input type="text" value="{{ $value }}" disabled></td>
</tr>{{ end }}
<tr>
<td><input type="text" form="env" name="envName"></td>
<td><input type="text" form="env" name="envValue"></td>
<td><form id="env"><button
type="button"
hx-post="/source/{{ .Name }}/edit"
>Update</button></form></td>
</form>
</tr>
</table>
<p>Actions:</p>
<table id="actions">{{ range $name, $argv := .Actions }}
<tr>
<td><input type="text" value="{{ $name }}" disabled></td>
<td><input type="text" value="{{ $argv }}" disabled></td>
</tr>{{ end }}
<tr>
<td><input type="text" form="action" name="actionName"></td>
<td><input type="text" form="action" name="actionArgv"></td>
<td><form id="action"><button
type="button"
hx-post="/source/{{ .Name }}/edit"
>Update</button></form></td>
</form>
</tr>
</table>
</nav>
{{- end }}

View File

@ -34,6 +34,9 @@
<button type="submit">fetch</button> <button type="submit">fetch</button>
</form> </form>
</td> </td>
<td>
<a href="/source/{{ .Name }}/edit">(edit)</a>
</td>
</td> </td>
<td><a href="/source/{{ .Name }}">{{ .Name }}</a></td> <td><a href="/source/{{ .Name }}">{{ .Name }}</a></td>
</tr> </tr>

View File

@ -182,3 +182,17 @@ func EditChannels(writer io.Writer, data EditChannelsData) {
log.Printf("error: failed to render edit channels: %v", err) log.Printf("error: failed to render edit channels: %v", err)
} }
} }
var editSource = load("editSource.html")
type EditSourceData struct {
Name string
Envs map[string]string
Actions map[string]string
}
func EditSource(writer io.Writer, data EditSourceData) {
if err := editSource.Execute(writer, data); err != nil {
log.Printf("error: failed to render edit source: %v", err)
}
}

View File

@ -98,3 +98,6 @@ article textarea {
span.error-message { span.error-message {
color: red; color: red;
} }
#envvars input {
font-family: monospace;
}

View File

@ -40,6 +40,8 @@ func RunServer(db core.DB, addr string, port string) {
handleFunc("GET /htmx.org@2.0.4.js", env.getScript, logged) handleFunc("GET /htmx.org@2.0.4.js", env.getScript, logged)
handleFunc("POST /login", env.login, logged) handleFunc("POST /login", env.login, logged)
handleFunc("GET /source/{source}", env.getSource, env.authed, logged) handleFunc("GET /source/{source}", env.getSource, env.authed, logged)
handleFunc("GET /source/{source}/edit", env.getEditSource, env.authed, logged)
handleFunc("POST /source/{source}/edit", env.editSource, env.authed, logged)
handleFunc("POST /source/{source}/fetch", env.fetchSource, env.authed, logged) handleFunc("POST /source/{source}/fetch", env.fetchSource, env.authed, logged)
handleFunc("GET /channel/", env.getChannels, env.authed, logged) handleFunc("GET /channel/", env.getChannels, env.authed, logged)
handleFunc("POST /channel/", env.editChannel, env.authed, logged) handleFunc("POST /channel/", env.editChannel, env.authed, logged)

44
web/shlex.go Normal file
View File

@ -0,0 +1,44 @@
package web
import (
"io"
"os/exec"
"strings"
)
func Shlex(cmd string) (argv []string, err error) {
xargs := exec.Command("xargs", "-n1")
stdin, err := xargs.StdinPipe()
if err != nil {
return
}
_, err = io.WriteString(stdin, cmd)
if err != nil {
return
}
err = stdin.Close()
if err != nil {
return
}
output, err := xargs.Output()
if err != nil {
return
}
argv = strings.Split(string(output), "\n")
if argv[len(argv)-1] == "" {
argv = argv[:len(argv)-1]
}
return
}
func Quote(argv []string) (cmd string, err error) {
argv = append([]string{"%q "}, argv...)
printf := exec.Command("printf", argv...)
output, err := printf.Output()
if err != nil {
return
}
cmd = string(output)
cmd = cmd[:len(cmd)-1] // strip off the last space from '%q '
return
}

60
web/shlex_test.go Normal file
View File

@ -0,0 +1,60 @@
package web
import "testing"
func TestShlex(t *testing.T) {
shlex := func(t *testing.T, cmd string, expected ...string) {
t.Helper()
actual, err := Shlex(cmd)
if err != nil {
t.Fatalf("shlex failed: %v", err)
}
if len(actual) != len(expected) {
t.Fatalf("expected %d args, got %d\nexpect: %v\nactual: %v", len(expected), len(actual), expected, actual)
}
for i := range len(expected) {
if actual[i] != expected[i] {
t.Fatalf("arg %d incorrect: expected %s got %s", i, expected[i], actual[i])
}
}
}
quote := func(t *testing.T, argvAndExpected ...string) {
t.Helper()
argv := argvAndExpected[:len(argvAndExpected)-1]
expected := argvAndExpected[len(argvAndExpected)-1]
actual, err := Quote(argv)
if err != nil {
t.Fatalf("quote failed: %v", err)
}
if actual != expected {
t.Fatalf("expected %q, got %q", expected, actual)
}
}
shlex(t,
`Lord "have mercy"`,
"Lord", "have mercy",
)
shlex(t,
`jq -cR '{id: .}'`,
"jq", "-cR", `{id: .}`,
)
shlex(t,
`jq -cR '{id: "hello"}'`,
"jq", "-cR", `{id: "hello"}`,
)
quote(t,
"Lord", "have mercy",
`Lord 'have mercy'`,
)
quote(t,
"jq", "-cR", `{id: .}`,
`jq -cR '{id: .}'`,
)
quote(t,
"jq", "-cR", `{id: "hello"}`,
`jq -cR '{id: "hello"}'`,
)
}

View File

@ -78,3 +78,80 @@ func (env *Env) fetchSource(writer http.ResponseWriter, req *http.Request) {
} }
html.Fetch(writer, data) html.Fetch(writer, data)
} }
func (env *Env) getEditSource(writer http.ResponseWriter, req *http.Request) {
source := req.PathValue("source")
if exists, err := core.SourceExists(env.db, source); !exists || err != nil {
http.NotFound(writer, req)
return
}
envs, err := core.GetEnvsMap(env.db, source)
if err != nil {
http.Error(writer, err.Error(), 500)
return
}
actions, _ := core.GetActionsForSource(env.db, source)
argvs := make(map[string]string)
for _, action := range actions {
argv, _ := core.GetArgvForAction(env.db, source, action)
args, _ := Quote(argv)
argvs[action] = args
}
data := html.EditSourceData{
Name: source,
Envs: envs,
Actions: argvs,
}
html.EditSource(writer, data)
}
func (env *Env) editSource(writer http.ResponseWriter, req *http.Request) {
source := req.PathValue("source")
if exists, err := core.SourceExists(env.db, source); !exists || err != nil {
http.NotFound(writer, req)
return
}
if err := req.ParseForm(); err != nil {
http.Error(writer, err.Error(), 400)
return
}
envName := req.PostForm.Get("envName")
envValue := req.PostForm.Get("envValue")
if envName != "" {
log.Printf("setting %s=%s", envName, envValue)
if err := core.SetEnvs(env.db, source, []string{fmt.Sprintf("%s=%s", envName, envValue)}); err != nil {
http.Error(writer, err.Error(), 500)
return
}
}
actionName := req.PostForm.Get("actionName")
actionArgv := req.PostForm.Get("actionArgv")
if actionName != "" {
log.Printf("setting %s -- %s", actionName, actionArgv)
if actionArgv == "" {
if err := core.DeleteAction(env.db, source, actionName); err != nil {
http.Error(writer, err.Error(), 500)
return
}
} else {
argv, err := Shlex(actionArgv)
if err != nil {
http.Error(writer, err.Error(), 400)
return
}
if err = core.SetAction(env.db, source, actionName, argv); err != nil {
http.Error(writer, err.Error(), 500)
return
}
}
}
writer.Header()["HX-Refresh"] = []string{"true"}
writer.WriteHeader(http.StatusNoContent)
}