1
0
mirror of https://github.com/osmarks/mycorrhiza.git synced 2025-01-23 08:26:51 +00:00
mycorrhiza/history/information.go

183 lines
4.8 KiB
Go
Raw Normal View History

// information.go
// Things related to gathering existing information.
2020-08-19 18:54:23 +00:00
package history
import (
"fmt"
"regexp"
2020-10-03 12:59:53 +00:00
"strconv"
2020-08-19 18:54:23 +00:00
"strings"
2020-11-29 17:06:45 +00:00
"time"
2020-09-26 18:19:17 +00:00
"github.com/bouncepaw/mycorrhiza/templates"
2020-11-29 17:06:45 +00:00
"github.com/bouncepaw/mycorrhiza/util"
"github.com/gorilla/feeds"
2020-08-19 18:54:23 +00:00
)
func recentChangesFeed() *feeds.Feed {
feed := &feeds.Feed{
Title: "Recent changes",
Link: &feeds.Link{Href: util.URL},
Description: "List of 30 recent changes on the wiki",
Author: &feeds.Author{Name: "Wikimind", Email: "wikimind@mycorrhiza"},
Updated: time.Now(),
}
var (
out, err = gitsh(
"log", "--oneline", "--no-merges",
"--pretty=format:\"%h\t%ae\t%at\t%s\"",
"--max-count=30",
)
revs []Revision
)
if err == nil {
for _, line := range strings.Split(out.String(), "\n") {
revs = append(revs, parseRevisionLine(line))
}
}
for _, rev := range revs {
feed.Add(&feeds.Item{
Title: rev.Message,
Author: &feeds.Author{Name: rev.Username},
Id: rev.Hash,
Description: rev.descriptionForFeed(),
Created: rev.Time,
Updated: rev.Time,
Link: &feeds.Link{Href: util.URL + rev.bestLink()},
})
}
return feed
}
func RecentChangesRSS() (string, error) {
return recentChangesFeed().ToRss()
}
func RecentChangesAtom() (string, error) {
return recentChangesFeed().ToAtom()
}
func RecentChangesJSON() (string, error) {
return recentChangesFeed().ToJSON()
}
2020-09-26 18:19:17 +00:00
func RecentChanges(n int) string {
var (
out, err = gitsh(
"log", "--oneline", "--no-merges",
"--pretty=format:\"%h\t%ae\t%at\t%s\"",
2020-10-03 12:59:53 +00:00
"--max-count="+strconv.Itoa(n),
2020-09-26 18:19:17 +00:00
)
revs []Revision
)
if err == nil {
for _, line := range strings.Split(out.String(), "\n") {
revs = append(revs, parseRevisionLine(line))
}
}
entries := make([]string, len(revs))
for i, rev := range revs {
entries[i] = rev.RecentChangesEntry()
}
return templates.RecentChangesHTML(entries, n)
}
2021-01-21 14:21:34 +00:00
// FileChanged tells you if the file has been changed.
func FileChanged(path string) bool {
_, err := gitsh("diff", "--exit-code", path)
return err != nil
}
// Revisions returns slice of revisions for the given hypha name.
2020-08-28 19:10:46 +00:00
func Revisions(hyphaName string) ([]Revision, error) {
2020-08-19 18:54:23 +00:00
var (
out, err = gitsh(
"log", "--oneline", "--no-merges",
2020-11-29 17:06:45 +00:00
// Hash, author email, author time, commit msg separated by tab
"--pretty=format:\"%h\t%ae\t%at\t%s\"",
"--", hyphaName+".*",
2020-08-19 18:54:23 +00:00
)
revs []Revision
)
if err == nil {
for _, line := range strings.Split(out.String(), "\n") {
2020-09-29 15:04:22 +00:00
if line != "" {
revs = append(revs, parseRevisionLine(line))
}
2020-08-19 18:54:23 +00:00
}
}
return revs, err
}
2020-11-29 17:06:45 +00:00
// HistoryWithRevisions returns an html representation of `revs` that is meant to be inserted in a history page.
func HistoryWithRevisions(hyphaName string, revs []Revision) (html string) {
var (
currentYear int
currentMonth time.Month
)
for i, rev := range revs {
if rev.Time.Month() != currentMonth || rev.Time.Year() != currentYear {
currentYear = rev.Time.Year()
currentMonth = rev.Time.Month()
if i != 0 {
html += `
</ul>
</section>`
}
html += fmt.Sprintf(`
<section class="history__month">
<a href="#%[1]d-%[2]d" class="history__month-anchor">
<h2 id="%[1]d-%[2]d" class="history__month-title">%[3]s</h2>
</a>
<ul class="history__entries">`,
currentYear, currentMonth,
strconv.Itoa(currentYear)+" "+rev.Time.Month().String())
}
html += rev.asHistoryEntry(hyphaName)
}
return html
}
func (rev *Revision) asHistoryEntry(hyphaName string) (html string) {
author := ""
if rev.Username != "anon" {
author = fmt.Sprintf(`
2021-01-23 16:37:29 +00:00
<span class="history-entry__author">by <a href="/page/%[1]s/%[2]s" rel="author">%[2]s</span>`, util.UserHypha, rev.Username)
2020-11-29 17:06:45 +00:00
}
return fmt.Sprintf(`
<li class="history__entry">
<a class="history-entry" href="/rev/%[3]s/%[1]s">
<time class="history-entry__time">%[2]s</time>
<span class="history-entry__hash">%[3]s</span>
<span class="history-entry__msg">%[4]s</span>
</a>%[5]s
</li>
2021-01-21 13:49:22 +00:00
`, hyphaName, rev.timeToDisplay(), rev.Hash, rev.Message, author)
2020-11-29 17:06:45 +00:00
}
2021-01-21 13:49:22 +00:00
// Return time like mm-dd 13:42
func (rev *Revision) timeToDisplay() string {
D := rev.Time.Day()
2020-11-29 17:06:45 +00:00
h, m, _ := rev.Time.Clock()
2021-01-21 13:49:22 +00:00
return fmt.Sprintf("%02d — %02d:%02d", D, h, m)
2020-11-29 17:06:45 +00:00
}
2020-08-19 18:54:23 +00:00
// This regex is wrapped in "". For some reason, these quotes appear at some time and we have to get rid of them.
var revisionLinePattern = regexp.MustCompile("\"(.*)\t(.*)@.*\t(.*)\t(.*)\"")
func parseRevisionLine(line string) Revision {
results := revisionLinePattern.FindStringSubmatch(line)
return Revision{
Hash: results[1],
Username: results[2],
Time: *unixTimestampAsTime(results[3]),
Message: results[4],
}
2020-08-19 18:54:23 +00:00
}
// See how the file with `filepath` looked at commit with `hash`.
func FileAtRevision(filepath, hash string) (string, error) {
out, err := gitsh("show", hash+":"+filepath)
return out.String(), err
2020-08-19 18:54:23 +00:00
}