Compare commits
5 Commits
74f7230c65
...
879238e61e
Author | SHA1 | Date | |
---|---|---|---|
879238e61e | |||
3a39b84528 | |||
227c05c365 | |||
9703db8478 | |||
b12a411fd6 |
17
README.md
17
README.md
@ -55,6 +55,10 @@ A minimally functional source requires a `fetch` action that returns items.
|
|||||||
TTL, TTD, and TTS can be configured at the source level by setting the environment variables `INTAKE_TTL`, `INTAKE_TTS`, or `INTAKE_TTS` to an integer value.
|
TTL, TTD, and TTS can be configured at the source level by setting the environment variables `INTAKE_TTL`, `INTAKE_TTS`, or `INTAKE_TTS` to an integer value.
|
||||||
These values override any `ttl`, `ttd`, or `tts` value returned by a fetch or action.
|
These values override any `ttl`, `ttd`, or `tts` value returned by a fetch or action.
|
||||||
|
|
||||||
|
Intake provides integration with `cron`.
|
||||||
|
To create a cron job for a source, set the `INTAKE_CRON` environment variable to a five-element crontab spec (e.g. `0 0 * * *`).
|
||||||
|
The `intake crontab` command will synchronize source cron jobs to your crontab.
|
||||||
|
|
||||||
### Action API
|
### Action API
|
||||||
|
|
||||||
The Intake action API defines how programs should behave to be used with Intake sources.
|
The Intake action API defines how programs should behave to be used with Intake sources.
|
||||||
@ -103,23 +107,28 @@ Parity features
|
|||||||
* [x] item punt
|
* [x] item punt
|
||||||
* [x] web feed paging
|
* [x] web feed paging
|
||||||
* [x] web fetch
|
* [x] web fetch
|
||||||
* [ ] crontab integration
|
* [x] crontab integration
|
||||||
* [ ] source batching
|
* [ ] source batching
|
||||||
* [x] add item from web
|
* [x] add item from web
|
||||||
|
* [ ] web edit channels
|
||||||
|
* web edit sources
|
||||||
|
* [ ] edit action argv
|
||||||
|
* [ ] edit source envs
|
||||||
|
* [ ] edit source crontab
|
||||||
* [x] Nix build
|
* [x] Nix build
|
||||||
* [ ] NixOS module
|
* [ ] NixOS module
|
||||||
* [ ] NixOS vm demo
|
* [ ] NixOS vm demo
|
||||||
* [ ] Nix flake templates
|
|
||||||
|
|
||||||
Future features
|
Future features
|
||||||
|
|
||||||
* [ ] on_delete triggers
|
* [ ] on_delete triggers
|
||||||
* [ ] manual item edits, CLI
|
* [ ] manual item edits, CLI
|
||||||
* [ ] manual item edits, web
|
* [ ] manual item edits, web
|
||||||
* [ ] source-level TTS
|
* [x] source-level TTS
|
||||||
* [ ] metric reporting
|
* [ ] metric reporting
|
||||||
* [x] on action failure, create an error item with logs
|
* [x] on action failure, create an error item with logs
|
||||||
* [ ] items gracefully add new fields and `action` keys
|
* [ ] items gracefully add new fields and `action` keys
|
||||||
* [ ] arbitrary date punt
|
* [ ] arbitrary date punt
|
||||||
* [ ] sort crontab entries
|
* [x] sort crontab entries
|
||||||
* [ ] TUI feed view
|
* [ ] TUI feed view
|
||||||
|
* [ ] Nix flake templates
|
||||||
|
51
cmd/crontab.go
Normal file
51
cmd/crontab.go
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"sort"
|
||||||
|
|
||||||
|
"github.com/Jaculabilis/intake/core"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
var crontabCmd = &cobra.Command{
|
||||||
|
Use: "crontab",
|
||||||
|
Short: "Update crontab entries",
|
||||||
|
Long: `Update crontab entries.
|
||||||
|
|
||||||
|
A source's cron job is defined by its INTAKE_CRON environment variable.
|
||||||
|
`,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
crontab(boolArg(cmd, "list"))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
rootCmd.AddCommand(crontabCmd)
|
||||||
|
|
||||||
|
crontabCmd.Flags().BoolP("list", "l", false, "List crontab entries")
|
||||||
|
}
|
||||||
|
|
||||||
|
func crontab(list bool) {
|
||||||
|
db := openAndMigrateDb()
|
||||||
|
|
||||||
|
specs, err := core.GetCronSources(db)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("error: failed to get crontab sources: %v", err)
|
||||||
|
}
|
||||||
|
if list {
|
||||||
|
var sources []string
|
||||||
|
for source := range specs {
|
||||||
|
sources = append(sources, source)
|
||||||
|
}
|
||||||
|
sort.Strings(sources)
|
||||||
|
for _, source := range sources {
|
||||||
|
fmt.Println(specs[source])
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := core.UpdateCrontab(db, specs); err != nil {
|
||||||
|
log.Fatalf("error: failed to update crontab: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,7 @@
|
|||||||
package core
|
package core
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
func AddSourceToChannel(db DB, channel string, source string) error {
|
func AddSourceToChannel(db DB, channel string, source string) error {
|
||||||
_, err := db.Exec(`
|
_, err := db.Exec(`
|
||||||
insert into channels (name, source)
|
insert into channels (name, source)
|
||||||
@ -41,12 +43,16 @@ func GetSourcesInChannel(db DB) (map[string][]string, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func GetChannelsAndActiveCounts(db DB) (map[string]int, error) {
|
func GetChannelsAndActiveCounts(db DB) (map[string]int, error) {
|
||||||
|
now := int(time.Now().Unix()) // TODO pass this value in
|
||||||
rows, err := db.Query(`
|
rows, err := db.Query(`
|
||||||
select c.name, count(i.id) as count
|
select c.name, count(i.id) as count
|
||||||
from channels c
|
from channels c
|
||||||
left outer join items i on c.source = i.source and i.active = 1
|
left outer join items i
|
||||||
|
on c.source = i.source
|
||||||
|
and i.active <> 0
|
||||||
|
and (i.tts = 0 or i.created + i.tts < ?)
|
||||||
group by c.name
|
group by c.name
|
||||||
`)
|
`, now)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
121
core/crontab.go
Normal file
121
core/crontab.go
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
package core
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
var IntakeCronBegin = "### begin intake-managed crontab entries"
|
||||||
|
var IntakeCronEnd = "### end intake-managed crontab entries"
|
||||||
|
|
||||||
|
func makeCrontabEntry(source string, spec string) string {
|
||||||
|
// TODO the /etc/profile setup is NixOS-specific, maybe there's another way to do this
|
||||||
|
return fmt.Sprintf("%-20s . /etc/profile; intake source fetch -s %s", spec, source)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetCronSources(db DB) (specs map[string]string, err error) {
|
||||||
|
res, err := db.Query(`
|
||||||
|
select source, value
|
||||||
|
from envs
|
||||||
|
where name = 'INTAKE_CRON'
|
||||||
|
`)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to get source crontabs: %v", err)
|
||||||
|
}
|
||||||
|
specs = make(map[string]string)
|
||||||
|
for res.Next() {
|
||||||
|
var source string
|
||||||
|
var value string
|
||||||
|
if err = res.Scan(&source, &value); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to scan source crontab: %v", err)
|
||||||
|
}
|
||||||
|
specs[source] = makeCrontabEntry(source, value)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the intake-managed section of the user's crontab.
|
||||||
|
func UpdateCrontab(db DB, specs map[string]string) (err error) {
|
||||||
|
// If there is no crontab command available, quit early.
|
||||||
|
crontabPath, err := exec.LookPath("crontab")
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("no crontab found")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the current crontab without extra header lines via `EDITOR=cat crontab -e`
|
||||||
|
cmdLoad := exec.Command(crontabPath, "-e")
|
||||||
|
cmdLoad.Env = append(os.Environ(), "EDITOR=cat")
|
||||||
|
output, err := cmdLoad.Output()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error: failed to get current crontab: %v", err)
|
||||||
|
}
|
||||||
|
lines := strings.Split(string(output), "\n")
|
||||||
|
|
||||||
|
// Sort the new intake crons
|
||||||
|
var sources []string
|
||||||
|
for source := range specs {
|
||||||
|
sources = append(sources, source)
|
||||||
|
}
|
||||||
|
sort.Strings(sources)
|
||||||
|
|
||||||
|
// Splice the intake crons into the crontab
|
||||||
|
var newCrontab []string
|
||||||
|
headerFound := false
|
||||||
|
inSection := false
|
||||||
|
for i := range lines {
|
||||||
|
switch {
|
||||||
|
case !headerFound && lines[i] == IntakeCronBegin:
|
||||||
|
headerFound = true
|
||||||
|
inSection = true
|
||||||
|
newCrontab = append(newCrontab, IntakeCronBegin)
|
||||||
|
for _, source := range sources {
|
||||||
|
newCrontab = append(newCrontab, specs[source])
|
||||||
|
}
|
||||||
|
|
||||||
|
case lines[i] == IntakeCronEnd:
|
||||||
|
newCrontab = append(newCrontab, IntakeCronEnd)
|
||||||
|
inSection = false
|
||||||
|
|
||||||
|
case !inSection:
|
||||||
|
newCrontab = append(newCrontab, lines[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the splice mark was never found, append the whole section to the end
|
||||||
|
if !headerFound {
|
||||||
|
newCrontab = append(newCrontab, IntakeCronBegin)
|
||||||
|
for _, source := range sources {
|
||||||
|
newCrontab = append(newCrontab, specs[source])
|
||||||
|
}
|
||||||
|
newCrontab = append(newCrontab, IntakeCronEnd)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("Updating %d crontab entries", len(specs))
|
||||||
|
|
||||||
|
// Save the updated crontab
|
||||||
|
cmdSave := exec.Command(crontabPath, "-")
|
||||||
|
stdin, err := cmdSave.StdinPipe()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to open stdin: %v", err)
|
||||||
|
}
|
||||||
|
if _, err := io.WriteString(stdin, strings.Join(newCrontab, "\n")); err != nil {
|
||||||
|
return fmt.Errorf("failed to write to crontab: %v", err)
|
||||||
|
}
|
||||||
|
if err := stdin.Close(); err != nil {
|
||||||
|
return fmt.Errorf("failed to close stdin: %v", err)
|
||||||
|
}
|
||||||
|
output, err = cmdSave.CombinedOutput()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("failed to read crontab output: %v", err)
|
||||||
|
}
|
||||||
|
if len(output) > 0 {
|
||||||
|
log.Printf("crontab output: %s", string(output))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
@ -57,3 +57,8 @@ tmp/intake item add -s page --id 212 --title "Item 212" --body "This is the body
|
|||||||
# default password, comment out to test no password
|
# default password, comment out to test no password
|
||||||
echo "hello" | tmp/intake passwd --stdin
|
echo "hello" | tmp/intake passwd --stdin
|
||||||
echo "hello" | tmp/intake passwd --stdin --verify
|
echo "hello" | tmp/intake passwd --stdin --verify
|
||||||
|
|
||||||
|
# crontab integration
|
||||||
|
tmp/intake source env -s page --set "INTAKE_CRON=0 0 * * *"
|
||||||
|
tmp/intake source env -s spook --set "INTAKE_CRON=0 0 * * *"
|
||||||
|
tmp/intake source env -s feedtest --set "INTAKE_CRON=0 0 * * *"
|
||||||
|
@ -1,12 +1,76 @@
|
|||||||
package web
|
package web
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/Jaculabilis/intake/core"
|
"github.com/Jaculabilis/intake/core"
|
||||||
"github.com/Jaculabilis/intake/web/html"
|
"github.com/Jaculabilis/intake/web/html"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func (env *Env) getChannels(writer http.ResponseWriter, req *http.Request) {
|
||||||
|
allSources, err := core.GetSources(env.db)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(writer, err.Error(), 400)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
channelSources, err := core.GetSourcesInChannel(env.db)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(writer, err.Error(), 400)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
data := html.EditChannelsData{
|
||||||
|
Sources: allSources,
|
||||||
|
ChannelSources: channelSources,
|
||||||
|
}
|
||||||
|
html.EditChannels(writer, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (env *Env) editChannel(writer http.ResponseWriter, req *http.Request) {
|
||||||
|
if err := req.ParseForm(); err != nil {
|
||||||
|
http.Error(writer, err.Error(), 400)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
channel := req.Form.Get("channel")
|
||||||
|
source := req.Form.Get("source")
|
||||||
|
|
||||||
|
if channel == "" {
|
||||||
|
http.Error(writer, "missing channel", 400)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if source == "" {
|
||||||
|
http.Error(writer, "missing source", 400)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if exists, err := core.SourceExists(env.db, source); err != nil || !exists {
|
||||||
|
http.Error(writer, fmt.Sprintf("could not find source %s: %v", source, err), 500)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Method == http.MethodPost {
|
||||||
|
if err := core.AddSourceToChannel(env.db, channel, source); err != nil {
|
||||||
|
http.Error(writer, err.Error(), 500)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
writer.Header()["HX-Refresh"] = []string{"true"}
|
||||||
|
writer.WriteHeader(http.StatusNoContent)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Method == http.MethodDelete {
|
||||||
|
if err := core.DeleteSourceFromChannel(env.db, channel, source); err != nil {
|
||||||
|
http.Error(writer, err.Error(), 500)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
writer.Header()["HX-Refresh"] = []string{"true"}
|
||||||
|
writer.WriteHeader(http.StatusNoContent)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (env *Env) getChannel(writer http.ResponseWriter, req *http.Request) {
|
func (env *Env) getChannel(writer http.ResponseWriter, req *http.Request) {
|
||||||
channel := req.PathValue("channel")
|
channel := req.PathValue("channel")
|
||||||
|
|
||||||
|
49
web/html/editChannels.html
Normal file
49
web/html/editChannels.html
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
{{ define "title" }}Channels - Intake{{ end }}
|
||||||
|
|
||||||
|
{{ define "content" -}}
|
||||||
|
<nav class="center">
|
||||||
|
<span class="feed-controls">
|
||||||
|
<a href="/">Home</a>
|
||||||
|
</span>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<nav>
|
||||||
|
<span class="feed-controls">Edit channels</span>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<form>
|
||||||
|
<label for="source">Add</label>
|
||||||
|
<select name="source">{{ range .Sources }}
|
||||||
|
<option value="{{ . }}">{{ . }}</option>{{ end }}
|
||||||
|
</select>
|
||||||
|
<label for="channel">to</label>
|
||||||
|
<input type="text" name="channel" list="channel-options">
|
||||||
|
<button
|
||||||
|
hx-post="/channel/"
|
||||||
|
>Submit</button>
|
||||||
|
</form>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<datalist id="channel-options">{{ range $channel, $_ := .ChannelSources }}
|
||||||
|
<option value="{{ $channel }}"></option>{{ end }}
|
||||||
|
</datalist>
|
||||||
|
|
||||||
|
{{ range $channel, $sources := .ChannelSources }}
|
||||||
|
<p><b><a href="/channel/{{ $channel }}">{{ $channel }}</a></b></p>
|
||||||
|
<table>
|
||||||
|
{{- range $sources }}
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<button
|
||||||
|
hx-delete="/channel/?channel={{ $channel }}&source={{ . }}"
|
||||||
|
>✕</button>
|
||||||
|
</td>
|
||||||
|
</form>
|
||||||
|
<td><a href="/source/{{ . }}">{{ . }}</a></td>
|
||||||
|
</tr>
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
</table>
|
||||||
|
{{ end }}
|
||||||
|
</nav>
|
||||||
|
{{- end }}
|
@ -1,7 +1,7 @@
|
|||||||
{{ define "title" }}{{ if .Items }}({{ len .Items }}) {{ end }}Intake{{ end }}
|
{{ define "title" }}{{ if .Items }}({{ len .Items }}) {{ end }}Intake{{ end }}
|
||||||
|
|
||||||
{{ define "content" -}}
|
{{ define "content" -}}
|
||||||
<article class="center">
|
<nav class="center">
|
||||||
<span class="feed-controls flex-between">
|
<span class="feed-controls flex-between">
|
||||||
<a href="?hidden={{ .ShowHidden }}&page={{ page .Page -1 }}&count={{ .Count }}"><--</a>
|
<a href="?hidden={{ .ShowHidden }}&page={{ page .Page -1 }}&count={{ .Count }}"><--</a>
|
||||||
<a href="/">Home</a>
|
<a href="/">Home</a>
|
||||||
@ -14,20 +14,20 @@
|
|||||||
</span>
|
</span>
|
||||||
<a href="?hidden={{ .ShowHidden }}&page={{ page .Page 1 }}&count={{ .Count }}">--></a>
|
<a href="?hidden={{ .ShowHidden }}&page={{ page .Page 1 }}&count={{ .Count }}">--></a>
|
||||||
</span>
|
</span>
|
||||||
</article>
|
</nav>
|
||||||
|
|
||||||
{{ if .Items }}
|
{{ if .Items }}
|
||||||
{{ range .Items }}
|
{{ range .Items }}
|
||||||
{{ template "item" . }}
|
{{ template "item" . }}
|
||||||
{{ end }}
|
{{ end }}
|
||||||
|
|
||||||
<article class="center">
|
<nav class="center">
|
||||||
<button
|
<button
|
||||||
hx-post="/mass-deactivate"
|
hx-post="/mass-deactivate"
|
||||||
hx-vals='{{ massDeacVars .Items }}'
|
hx-vals='{{ massDeacVars .Items }}'
|
||||||
hx-confirm="Deactivate {{ len .Items }} items?"
|
hx-confirm="Deactivate {{ len .Items }} items?"
|
||||||
>Deactivate All</button>
|
>Deactivate All</button>
|
||||||
</article>
|
</nav>
|
||||||
|
|
||||||
{{ else }}
|
{{ else }}
|
||||||
<article class="center">
|
<article class="center">
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
{{ define "title" }}Intake - fetch result for {{ .Source }}{{ end }}
|
{{ define "title" }}Intake - fetch result for {{ .Source }}{{ end }}
|
||||||
|
|
||||||
{{ define "content" -}}
|
{{ define "content" -}}
|
||||||
<article class="center">
|
<nav class="center">
|
||||||
<span class="feed-controls">Fetch results for <a href="/source/{{ .Source }}">{{ .Source }}</a></span>
|
<span class="feed-controls">Fetch results for <a href="/source/{{ .Source }}">{{ .Source }}</a></span>
|
||||||
</article>
|
</nav>
|
||||||
|
|
||||||
<article>
|
<article>
|
||||||
<p>{{ .Added }} new items, {{ .Updated }} updated items, {{ .Deleted }} deleted items</p>
|
<p>{{ .Added }} new items, {{ .Updated }} updated items, {{ .Deleted }} deleted items</p>
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
{{ define "title" }}Intake{{ end }}
|
{{ define "title" }}Intake{{ end }}
|
||||||
|
|
||||||
{{ define "content" -}}
|
{{ define "content" -}}
|
||||||
<article>
|
<nav>
|
||||||
<details open>
|
<details open>
|
||||||
<summary><span class="feed-controls">Channels</span></summary>
|
<summary><span class="feed-controls">Channels</span></summary>
|
||||||
{{ if .Channels }}
|
{{ if .Channels }}
|
||||||
{{ range .Channels }}
|
{{ range .Channels }}
|
||||||
<p><a href="/channel/{{ .Name }}">
|
<p><a href="/channel/{{ .Name }}">
|
||||||
{{ if .Active }}
|
{{ if .Active }}
|
||||||
{{ .Name }} ({{ .Active }})
|
({{ .Active }}) {{ .Name }}
|
||||||
{{ else }}
|
{{ else }}
|
||||||
{{ .Name }}
|
{{ .Name }}
|
||||||
{{ end }}
|
{{ end }}
|
||||||
@ -17,10 +17,12 @@
|
|||||||
{{ else }}
|
{{ else }}
|
||||||
<p>No channels found.</p>
|
<p>No channels found.</p>
|
||||||
{{ end }}
|
{{ end }}
|
||||||
</details>
|
|
||||||
</article>
|
|
||||||
|
|
||||||
<article>
|
<p><a href="/channel/">(Edit channels)</a></p>
|
||||||
|
</details>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<nav>
|
||||||
<details>
|
<details>
|
||||||
<summary><span class="feed-controls">Sources</span></summary>
|
<summary><span class="feed-controls">Sources</span></summary>
|
||||||
{{ if .Sources }}
|
{{ if .Sources }}
|
||||||
@ -41,9 +43,9 @@
|
|||||||
<p>No sources found.</p>
|
<p>No sources found.</p>
|
||||||
{{ end }}
|
{{ end }}
|
||||||
</details>
|
</details>
|
||||||
</article>
|
</nav>
|
||||||
|
|
||||||
<article>
|
<nav>
|
||||||
<details open>
|
<details open>
|
||||||
<summary><span class="feed-controls">Add item</span></summary>
|
<summary><span class="feed-controls">Add item</span></summary>
|
||||||
<form action="/item" method="post">
|
<form action="/item" method="post">
|
||||||
@ -73,5 +75,5 @@
|
|||||||
</p>
|
</p>
|
||||||
</form>
|
</form>
|
||||||
</details>
|
</details>
|
||||||
</article>
|
</nav>
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
@ -169,3 +169,16 @@ func Fetch(writer io.Writer, data FetchData) {
|
|||||||
log.Printf("error: failed to render fetch: %v", err)
|
log.Printf("error: failed to render fetch: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var editChannels = load("editChannels.html")
|
||||||
|
|
||||||
|
type EditChannelsData struct {
|
||||||
|
Sources []string
|
||||||
|
ChannelSources map[string][]string
|
||||||
|
}
|
||||||
|
|
||||||
|
func EditChannels(writer io.Writer, data EditChannelsData) {
|
||||||
|
if err := editChannels.Execute(writer, data); err != nil {
|
||||||
|
log.Printf("error: failed to render edit channels: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -3,7 +3,7 @@ main {
|
|||||||
max-width: 700px;
|
max-width: 700px;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
article {
|
article, nav {
|
||||||
border: 1px solid black; border-radius: 6px;
|
border: 1px solid black; border-radius: 6px;
|
||||||
padding: 5px;
|
padding: 5px;
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
@ -88,7 +88,7 @@ table.feed-control td {
|
|||||||
.intake-sources form {
|
.intake-sources form {
|
||||||
margin: 0
|
margin: 0
|
||||||
}
|
}
|
||||||
article.center {
|
.center {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
article textarea {
|
article textarea {
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
{{ define "title" }}{{ if .Item.Title }}{{ .Item.Title }}{{ else }}{{ .Item.Source }}/{{ .Item.Id }}{{ end }} - Intake [{{ .Item.Source }}]{{ end }}
|
{{ define "title" }}{{ if .Item.Title }}{{ .Item.Title }}{{ else }}{{ .Item.Source }}/{{ .Item.Id }}{{ end }} - Intake [{{ .Item.Source }}]{{ end }}
|
||||||
|
|
||||||
{{ define "content" -}}
|
{{ define "content" -}}
|
||||||
<article class="center">
|
<nav class="center">
|
||||||
<span class="feed-controls">
|
<span class="feed-controls">
|
||||||
<a href="/">Home</a>
|
<a href="/">Home</a>
|
||||||
</span>
|
</span>
|
||||||
</article>
|
</nav>
|
||||||
|
|
||||||
{{ template "item" .Item }}
|
{{ template "item" .Item }}
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{{ define "title" }}Intake - Login{{ end }}
|
{{ define "title" }}Intake - Login{{ end }}
|
||||||
|
|
||||||
{{ define "content" -}}
|
{{ define "content" -}}
|
||||||
<article class="center">
|
<nav class="center">
|
||||||
<form method="post" action="/login">
|
<form method="post" action="/login">
|
||||||
<p>
|
<p>
|
||||||
<input name="password" type="password"/>
|
<input name="password" type="password"/>
|
||||||
@ -13,5 +13,5 @@
|
|||||||
>Submit</button>
|
>Submit</button>
|
||||||
</form>
|
</form>
|
||||||
<p id="errors">{{ .Error }}</p>
|
<p id="errors">{{ .Error }}</p>
|
||||||
</article>
|
</nav>
|
||||||
{{ end }}
|
{{ end }}
|
||||||
|
@ -41,6 +41,9 @@ func RunServer(db core.DB, addr string, port string) {
|
|||||||
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("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("POST /channel/", env.editChannel, env.authed, logged)
|
||||||
|
handleFunc("DELETE /channel/", env.editChannel, env.authed, logged)
|
||||||
handleFunc("GET /channel/{channel}", env.getChannel, env.authed, logged)
|
handleFunc("GET /channel/{channel}", env.getChannel, env.authed, logged)
|
||||||
handleFunc("POST /item", env.addItem, env.authed, logged)
|
handleFunc("POST /item", env.addItem, env.authed, logged)
|
||||||
handleFunc("GET /item/{source}/{id}", env.getItem, env.authed, logged)
|
handleFunc("GET /item/{source}/{id}", env.getItem, env.authed, logged)
|
||||||
|
Loading…
Reference in New Issue
Block a user