2020-05-02 02:16:49 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-05-24 06:39:24 +00:00
|
|
|
"context"
|
2020-05-02 02:16:49 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
2020-11-08 00:25:36 +00:00
|
|
|
"log"
|
2020-05-02 02:16:49 +00:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Repo struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
Description string `json:"description"`
|
|
|
|
CreatedAt time.Time `json:"created_at"`
|
|
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type ContentsResponse struct {
|
|
|
|
Content []byte `json:"content"` // Assume base64 "encoding" parameter in Gitea response, and use Go's auto decode
|
|
|
|
}
|
|
|
|
|
|
|
|
type TopicsResponse struct {
|
|
|
|
Topics []string `json:"topics"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type ReaddirEntry struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
Path string `json:"path"`
|
|
|
|
Size int `json:"size"`
|
|
|
|
RawURL string `json:"download_url"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rde ReaddirEntry) isImage() bool {
|
|
|
|
return strings.HasSuffix(rde.Name, `.png`) || strings.HasSuffix(rde.Name, `.jpg`) || strings.HasSuffix(rde.Name, `.jpeg`)
|
|
|
|
}
|
|
|
|
|
|
|
|
type MarkdownRequest struct {
|
|
|
|
Context string
|
|
|
|
Mode string
|
|
|
|
Text string
|
|
|
|
Wiki bool
|
|
|
|
}
|
|
|
|
|
2020-11-08 00:25:36 +00:00
|
|
|
// reposPage gets a single page of the list of Git repositories in this organisation.
|
|
|
|
func (this *Application) reposPage(ctx context.Context, page, limit int) ([]Repo, error) {
|
2020-06-06 23:31:25 +00:00
|
|
|
err := this.apiSem.Acquire(ctx, 1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err // e.g. ctx closed
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
defer this.apiSem.Release(1)
|
|
|
|
|
2020-11-08 00:25:36 +00:00
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, this.cfg.Gitea.URL+`api/v1/orgs/`+url.PathEscape(this.cfg.Gitea.Org)+fmt.Sprintf(`/repos?page=%d&limit=%d`, page, limit), nil)
|
2020-05-24 06:39:24 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, fmt.Errorf("HTTP %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
var repos []Repo
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&repos)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return repos, nil
|
|
|
|
}
|
|
|
|
|
2020-11-08 00:25:36 +00:00
|
|
|
// repos gets a list of Git repositories in this organisation. It may have to
|
|
|
|
// make multiple network requests.
|
|
|
|
func (this *Application) repos(ctx context.Context) ([]Repo, error) {
|
|
|
|
// Seems like gitea-1.13.0-rc1 returns 30 results by default, and supports up to a limit of 100
|
|
|
|
// Make a much larger request
|
|
|
|
|
|
|
|
ret := make([]Repo, 0)
|
|
|
|
|
|
|
|
nextPage := 1 // Counting starts at 1
|
|
|
|
for {
|
|
|
|
page, err := this.reposPage(ctx, nextPage, 300)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Page %d with %d results", nextPage, len(page))
|
|
|
|
|
|
|
|
if len(page) == 0 && len(ret) > 0 {
|
|
|
|
return ret, nil // Found enough already
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append(ret, page...)
|
|
|
|
nextPage += 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-02 02:16:49 +00:00
|
|
|
// repoFile gets a single file from the default branch of the git repository
|
|
|
|
// Usually the default branch is `master`.
|
2020-05-24 06:39:24 +00:00
|
|
|
func (this *Application) repoFile(ctx context.Context, repo, filename string) ([]byte, error) {
|
2020-06-06 23:31:25 +00:00
|
|
|
err := this.apiSem.Acquire(ctx, 1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err // e.g. ctx closed
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
defer this.apiSem.Release(1)
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, this.cfg.Gitea.URL+`api/v1/repos/`+url.PathEscape(this.cfg.Gitea.Org)+`/`+url.PathEscape(repo)+`/contents/`+url.PathEscape(filename), nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, fmt.Errorf("HTTP %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
var cr ContentsResponse
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&cr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return cr.Content, nil
|
|
|
|
}
|
|
|
|
|
2020-05-24 06:39:24 +00:00
|
|
|
func (this *Application) filesInDirectory(ctx context.Context, repo, dir string) ([]ReaddirEntry, error) {
|
2020-06-06 23:31:25 +00:00
|
|
|
err := this.apiSem.Acquire(ctx, 1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err // e.g. ctx closed
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
defer this.apiSem.Release(1)
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, this.cfg.Gitea.URL+`api/v1/repos/`+url.PathEscape(this.cfg.Gitea.Org)+`/`+url.PathEscape(repo)+`/contents/`+dir, nil) // n.b. $dir param not escaped
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
|
|
|
|
// "No files found" happens with a HTTP 500/404 error depending on Gitea version. Catch this special case
|
|
|
|
if resp.StatusCode == 500 || resp.StatusCode == 404 {
|
|
|
|
b, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if strings.Contains(string(b), `does not exist`) {
|
|
|
|
return []ReaddirEntry{}, nil // no files found
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("HTTP %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
var ret []ReaddirEntry
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&ret)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// imageFilesForRepo finds documentation images for the repository.
|
|
|
|
// It searches the dist/ and doc/ subdirectories.
|
2020-05-24 06:39:24 +00:00
|
|
|
func (this *Application) imageFilesForRepo(ctx context.Context, repo string) ([]ReaddirEntry, error) {
|
2020-05-02 02:16:49 +00:00
|
|
|
|
|
|
|
ret := []ReaddirEntry{}
|
|
|
|
|
|
|
|
for _, dirName := range []string{`dist`, `doc`} {
|
|
|
|
|
2020-05-24 06:39:24 +00:00
|
|
|
files, err := this.filesInDirectory(ctx, repo, dirName)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("readdir(%s): %w", dirName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, f := range files {
|
|
|
|
if f.isImage() {
|
|
|
|
ret = append(ret, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2020-05-24 06:39:24 +00:00
|
|
|
func (this *Application) topicsForRepo(ctx context.Context, repo string) ([]string, error) {
|
2020-06-06 23:31:25 +00:00
|
|
|
err := this.apiSem.Acquire(ctx, 1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err // e.g. ctx closed
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
defer this.apiSem.Release(1)
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, this.cfg.Gitea.URL+`api/v1/repos/`+url.PathEscape(this.cfg.Gitea.Org)+`/`+url.PathEscape(repo)+`/topics`, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, fmt.Errorf("HTTP %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
var tr TopicsResponse
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&tr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return tr.Topics, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// renderMarkdown calls the remote Gitea server's own markdown renderer.
|
2020-05-24 06:39:24 +00:00
|
|
|
func (this *Application) renderMarkdown(ctx context.Context, repoName string, body string) ([]byte, error) {
|
2020-06-06 23:31:25 +00:00
|
|
|
err := this.apiSem.Acquire(ctx, 1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err // e.g. ctx closed
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
defer this.apiSem.Release(1)
|
|
|
|
|
|
|
|
jb, err := json.Marshal(MarkdownRequest{
|
2020-05-02 02:16:49 +00:00
|
|
|
Context: this.cfg.Gitea.URL + url.PathEscape(this.cfg.Gitea.Org) + `/` + url.PathEscape(repoName) + `/src/branch/master`,
|
|
|
|
Mode: "gfm", // magic constant - Github Flavoured Markdown
|
|
|
|
Text: body,
|
2020-05-24 06:39:24 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2020-05-02 02:16:49 +00:00
|
|
|
}
|
|
|
|
|
2020-05-24 06:39:24 +00:00
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, this.cfg.Gitea.URL+`api/v1/markdown/`, bytes.NewReader(jb))
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
req.Header.Set(`Content-Type`, `application/json`)
|
|
|
|
req.Header.Set(`Content-Length`, fmt.Sprintf("%d", len(jb)))
|
2020-05-02 02:16:49 +00:00
|
|
|
|
2020-05-24 06:39:24 +00:00
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, fmt.Errorf("HTTP %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ioutil.ReadAll(resp.Body)
|
|
|
|
}
|
|
|
|
|
|
|
|
// renderMarkdownRaw calls the remote Gitea server's own markdown renderer.
|
2020-05-24 06:39:24 +00:00
|
|
|
func (this *Application) renderMarkdownRaw(ctx context.Context, body []byte) ([]byte, error) {
|
2020-06-06 23:31:25 +00:00
|
|
|
err := this.apiSem.Acquire(ctx, 1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err // e.g. ctx closed
|
|
|
|
}
|
2020-05-24 06:39:24 +00:00
|
|
|
defer this.apiSem.Release(1)
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, this.cfg.Gitea.URL+`api/v1/markdown/raw`, bytes.NewReader(body))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Header.Set(`Content-Type`, `text/plain`)
|
|
|
|
req.Header.Set(`Content-Length`, fmt.Sprintf("%d", len(body)))
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2020-05-02 02:16:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, fmt.Errorf("HTTP %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ioutil.ReadAll(resp.Body)
|
|
|
|
}
|