mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-14 07:26:47 +00:00
* Remove old concept. Introduce new job types and candle scaling * Adds extra processing, commands * new concept for queued jobs. Jobs can pause. New commands to manage status * =End of day commit designing tables and implementing prerequisites further. * Adds postgres data history relations * Fixes table design for sqlite. Fixes all issues from merge * Fixes craziness of database design. Adds some functions to get related jobs * Fixes errors * Updates some documentation, manages prerequisite jobs a little better, adds rpc funcs * Fixes database design and adjust repo functions * Tests database relationship * Test coverage of new job functions * Finishes coverage of new functions * Commands and RPC coverage * New database modifications for new job types * Adds db support of new columns. Adds conversion validation. lint * command blurb changes * Allows websocket test to pass consistently * Fixes merge issue preventing datahistorymanager from starting via config * Minor fixes for different job type processing * Fixes rangeholder issue, fixes validation, does not address jobs not starting or wrong status * Fixes database tests, but at what cost. Fixes dhm tests * Fixes dhj completion issue. Adds prerequisite by nickname * Fixes validation processing. Adds db tests and validation * Fixes validation job processing range * Fixes trade sql. Reduces defaults. Validation processing and errors * Updates cli job commands. adds validation decimal. fix job validation * Expands run job handling and tests * Validation work * Fixes validation processing * candle relations. new job type. updating database design * Adds secondary exchange support. Sets stage for candle override * Re adds accidentally deleted relationship * Updates loading and saving candles to have relationship data when relevant * Now validates and replaces candle data appropriately * Fixes getting and setting datahistory data. Neatens DHM * Test coverage * Updates proto for new db types. New test coverage. Secondary exchange work * Investigation into never-ending validation jobs. Now that intervals are ruled out, now need to complete the job.... * Fixes issues with validation job completion. Fixes validation volume issue for secondary exchange * Adds candle warning support to the backtester * Fixes warnings * lint and begin docs * Documentation updates. Final testing changes * Minor fixes * docs, prerequisite checks, more testing * Fixes binance trade test. Rename err * Documentation fixes. Figure fixes * documentation update * Fixes remote PSQL tests * Fix binance mock test * Remove unnecessary JSON * regen proto * Some minor nit fixes * Var usage, query sorting, log improving, sql mirroring * Extra coverage * Experimental removal of m.jobs and mutex. Fix messaging * Fixes error * Lint fixes, command description improvements. More isRunning gates * description improvements * Lint * BUFF regenerate * Rough concept to fix insertions taking up long periods of time * New calculation for trade data. Adds batch saving This also adds an experimental request feature to shut down lingering requests. However, its uncertain whether or not this is having any impact. Initially thought it was the trades that was taking time and not SQL. Will investigate further * Removes experimental requester. Adds documentation. Fixes typo * rm unused error * re-adds more forgotten contributors * Now with proper commit count
578 lines
14 KiB
Go
578 lines
14 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"html/template"
|
|
"io/ioutil"
|
|
"log"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/thrasher-corp/gocryptotrader/common"
|
|
"github.com/thrasher-corp/gocryptotrader/common/file"
|
|
"github.com/thrasher-corp/gocryptotrader/core"
|
|
)
|
|
|
|
const (
|
|
// DefaultRepo is the main example repository
|
|
DefaultRepo = "https://api.github.com/repos/thrasher-corp/gocryptotrader"
|
|
|
|
// GithubAPIEndpoint allows the program to query your repository
|
|
// contributor list
|
|
GithubAPIEndpoint = "/contributors"
|
|
|
|
// LicenseFile defines a license file
|
|
LicenseFile = "LICENSE"
|
|
|
|
// ContributorFile defines contributor file
|
|
ContributorFile = "CONTRIBUTORS"
|
|
)
|
|
|
|
var (
|
|
// DefaultExcludedDirectories defines the basic directory exclusion list for GCT
|
|
DefaultExcludedDirectories = []string{".github",
|
|
".git",
|
|
"node_modules",
|
|
".vscode",
|
|
".idea",
|
|
"cmd_templates",
|
|
"common_templates",
|
|
"communications_templates",
|
|
"config_templates",
|
|
"currency_templates",
|
|
"events_templates",
|
|
"exchanges_templates",
|
|
"portfolio_templates",
|
|
"root_templates",
|
|
"sub_templates",
|
|
"testdata_templates",
|
|
"tools_templates",
|
|
"web_templates",
|
|
}
|
|
|
|
// global flag for verbosity
|
|
verbose bool
|
|
// current tool directory to specify working templates
|
|
toolDir string
|
|
// exposes root directory if outside of document tool directory
|
|
repoDir string
|
|
// is a broken down version of the documentation tool dir for cross platform
|
|
// checking
|
|
ref = []string{"gocryptotrader", "cmd", "documentation"}
|
|
engineFolder = "engine"
|
|
)
|
|
|
|
// Contributor defines an account associated with this code base by doing
|
|
// contributions
|
|
type Contributor struct {
|
|
Login string `json:"login"`
|
|
URL string `json:"html_url"`
|
|
Contributions int `json:"contributions"`
|
|
}
|
|
|
|
// Config defines the running config to deploy documentation across a github
|
|
// repository including exclusion lists for files and directories
|
|
type Config struct {
|
|
GithubRepo string `json:"githubRepo"`
|
|
Exclusions Exclusions `json:"exclusionList"`
|
|
RootReadme bool `json:"rootReadmeActive"`
|
|
LicenseFile bool `json:"licenseFileActive"`
|
|
ContributorFile bool `json:"contributorFileActive"`
|
|
}
|
|
|
|
// Exclusions defines the exclusion list so documents are not generated
|
|
type Exclusions struct {
|
|
Files []string `json:"Files"`
|
|
Directories []string `json:"Directories"`
|
|
}
|
|
|
|
// DocumentationDetails defines parameters to update documentation
|
|
type DocumentationDetails struct {
|
|
Directories []string
|
|
Tmpl *template.Template
|
|
Contributors []Contributor
|
|
Config *Config
|
|
}
|
|
|
|
// Attributes defines specific documentation attributes when a template is
|
|
// executed
|
|
type Attributes struct {
|
|
Name string
|
|
Contributors []Contributor
|
|
NameURL string
|
|
Year int
|
|
CapitalName string
|
|
DonationAddress string
|
|
}
|
|
|
|
func main() {
|
|
flag.BoolVar(&verbose, "v", false, "Verbose output")
|
|
flag.StringVar(&toolDir, "tooldir", "", "Pass in the documentation tool directory if outside tool folder")
|
|
flag.Parse()
|
|
|
|
wd, err := os.Getwd()
|
|
if err != nil {
|
|
fmt.Println("Documentation tool error cannot get working dir:", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
if strings.Contains(wd, filepath.Join(ref...)) {
|
|
rootDir := filepath.Dir(filepath.Dir(wd))
|
|
repoDir = rootDir
|
|
toolDir = wd
|
|
} else {
|
|
if toolDir == "" {
|
|
fmt.Println("Please set documentation tool directory via the tooldir flag if working outside of tool directory")
|
|
os.Exit(1)
|
|
}
|
|
repoDir = wd
|
|
}
|
|
|
|
fmt.Print(core.Banner)
|
|
fmt.Println("This will update and regenerate documentation for the different packages in your repo.")
|
|
fmt.Println()
|
|
|
|
if verbose {
|
|
fmt.Println("Fetching configuration...")
|
|
}
|
|
|
|
config, err := GetConfiguration()
|
|
if err != nil {
|
|
log.Fatalf("Documentation Generation Tool - GetConfiguration error %s",
|
|
err)
|
|
}
|
|
|
|
if verbose {
|
|
fmt.Println("Fetching project directory tree...")
|
|
}
|
|
|
|
dirList, err := GetProjectDirectoryTree(&config)
|
|
if err != nil {
|
|
log.Fatalf("Documentation Generation Tool - GetProjectDirectoryTree error %s",
|
|
err)
|
|
}
|
|
|
|
var contributors []Contributor
|
|
if config.ContributorFile {
|
|
if verbose {
|
|
fmt.Println("Fetching repository contributor list...")
|
|
}
|
|
contributors, err = GetContributorList(config.GithubRepo)
|
|
if err != nil {
|
|
log.Fatalf("Documentation Generation Tool - GetContributorList error %s",
|
|
err)
|
|
}
|
|
|
|
// Github API missing contributors
|
|
contributors = append(contributors, []Contributor{
|
|
{
|
|
Login: "tk42",
|
|
URL: "https://github.com/tk42",
|
|
Contributions: 2,
|
|
},
|
|
{
|
|
Login: "mshogin",
|
|
URL: "https://github.com/mshogin",
|
|
Contributions: 2,
|
|
},
|
|
{
|
|
Login: "herenow",
|
|
URL: "https://github.com/herenow",
|
|
Contributions: 2,
|
|
},
|
|
{
|
|
Login: "daniel-cohen",
|
|
URL: "https://github.com/daniel-cohen",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "DirectX",
|
|
URL: "https://github.com/DirectX",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "frankzougc",
|
|
URL: "https://github.com/frankzougc",
|
|
Contributions: 1,
|
|
},
|
|
// idoall's contributors were forked and merged, so his contributions
|
|
// aren't automatically retrievable
|
|
{
|
|
Login: "idoall",
|
|
URL: "https://github.com/idoall",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "mattkanwisher",
|
|
URL: "https://github.com/mattkanwisher",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "mKurrels",
|
|
URL: "https://github.com/mKurrels",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "m1kola",
|
|
URL: "https://github.com/m1kola",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "cavapoo2",
|
|
URL: "https://github.com/cavapoo2",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "zeldrinn",
|
|
URL: "https://github.com/zeldrinn",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "starit",
|
|
URL: "https://github.com/starit",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "Jimexist",
|
|
URL: "https://github.com/Jimexist",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "lookfirst",
|
|
URL: "https://github.com/lookfirst",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "merkeld",
|
|
URL: "https://github.com/merkeld",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "CodeLingoTeam",
|
|
URL: "https://github.com/CodeLingoTeam",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "Daanikus",
|
|
URL: "https://github.com/Daanikus",
|
|
Contributions: 1,
|
|
}, {
|
|
Login: "CodeLingoBot",
|
|
URL: "https://github.com/CodeLingoBot",
|
|
Contributions: 1,
|
|
},
|
|
{
|
|
Login: "blombard",
|
|
URL: "https://github.com/blombard",
|
|
Contributions: 1,
|
|
},
|
|
}...)
|
|
|
|
if verbose {
|
|
fmt.Println("Contributor List Fetched")
|
|
for i := range contributors {
|
|
fmt.Println(contributors[i].Login)
|
|
}
|
|
}
|
|
} else {
|
|
fmt.Println("Contributor list file disabled skipping fetching details")
|
|
}
|
|
|
|
if verbose {
|
|
fmt.Println("Fetching template files...")
|
|
}
|
|
|
|
tmpl, err := GetTemplateFiles()
|
|
if err != nil {
|
|
log.Fatalf("Documentation Generation Tool - GetTemplateFiles error %s",
|
|
err)
|
|
}
|
|
|
|
if verbose {
|
|
fmt.Println("All core systems fetched, updating documentation...")
|
|
}
|
|
|
|
UpdateDocumentation(DocumentationDetails{
|
|
dirList,
|
|
tmpl,
|
|
contributors,
|
|
&config})
|
|
|
|
fmt.Println("\nDocumentation Generation Tool - Finished")
|
|
}
|
|
|
|
// GetConfiguration retrieves the documentation configuration
|
|
func GetConfiguration() (Config, error) {
|
|
var c Config
|
|
configFilePath := filepath.Join(toolDir, "config.json")
|
|
|
|
if file.Exists(configFilePath) {
|
|
config, err := ioutil.ReadFile(configFilePath)
|
|
if err != nil {
|
|
return c, err
|
|
}
|
|
|
|
err = json.Unmarshal(config, &c)
|
|
if err != nil {
|
|
return c, err
|
|
}
|
|
|
|
if c.GithubRepo == "" {
|
|
return c, errors.New("repository not set in config.json file, please change")
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
fmt.Println("Creating configuration file, please check to add a different github repository path and change preferences")
|
|
|
|
// Set default params for configuration
|
|
c.GithubRepo = DefaultRepo
|
|
c.ContributorFile = true
|
|
c.LicenseFile = true
|
|
c.RootReadme = true
|
|
c.Exclusions.Directories = DefaultExcludedDirectories
|
|
|
|
data, err := json.MarshalIndent(c, "", " ")
|
|
if err != nil {
|
|
return c, err
|
|
}
|
|
|
|
if err := ioutil.WriteFile(configFilePath, data, 0770); err != nil {
|
|
return c, err
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
// IsExcluded returns if the file path is included in the exclusion list
|
|
func IsExcluded(path string, exclusion []string) bool {
|
|
for i := range exclusion {
|
|
if path == exclusion[i] {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// GetProjectDirectoryTree uses filepath walk functions to get each individual
|
|
// directory name and path to match templates with
|
|
func GetProjectDirectoryTree(c *Config) ([]string, error) {
|
|
var directoryData []string
|
|
if c.RootReadme { // Projects root README.md
|
|
directoryData = append(directoryData, repoDir)
|
|
}
|
|
|
|
if c.LicenseFile { // Standard license file
|
|
directoryData = append(directoryData, filepath.Join(repoDir, LicenseFile))
|
|
}
|
|
|
|
if c.ContributorFile { // Standard contributor file
|
|
directoryData = append(directoryData, filepath.Join(repoDir, ContributorFile))
|
|
}
|
|
|
|
walkFn := func(path string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if info.IsDir() {
|
|
// Bypass what is contained in config.json directory exclusion
|
|
if IsExcluded(info.Name(), c.Exclusions.Directories) {
|
|
if verbose {
|
|
fmt.Println("Excluding Directory:", info.Name())
|
|
}
|
|
return filepath.SkipDir
|
|
}
|
|
// Don't append parent directory
|
|
if strings.EqualFold(info.Name(), "..") {
|
|
return nil
|
|
}
|
|
directoryData = append(directoryData, path)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
return directoryData, filepath.Walk(repoDir, walkFn)
|
|
}
|
|
|
|
// GetTemplateFiles parses and returns all template files in the documentation
|
|
// tree
|
|
func GetTemplateFiles() (*template.Template, error) {
|
|
tmpl := template.New("")
|
|
|
|
walkFn := func(path string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if info.IsDir() {
|
|
if path == "." || path == ".." {
|
|
return nil
|
|
}
|
|
|
|
var parseError error
|
|
tmpl, parseError = tmpl.ParseGlob(filepath.Join(path, "*.tmpl"))
|
|
if parseError != nil {
|
|
if strings.Contains(parseError.Error(), "pattern matches no files") {
|
|
return nil
|
|
}
|
|
return parseError
|
|
}
|
|
return filepath.SkipDir
|
|
}
|
|
return nil
|
|
}
|
|
|
|
return tmpl, filepath.Walk(toolDir, walkFn)
|
|
}
|
|
|
|
// GetContributorList fetches a list of contributors from the github api
|
|
// endpoint
|
|
func GetContributorList(repo string) ([]Contributor, error) {
|
|
var resp []Contributor
|
|
return resp, common.SendHTTPGetRequest(repo+GithubAPIEndpoint, true, false, &resp)
|
|
}
|
|
|
|
// GetDocumentationAttributes returns specific attributes for a file template
|
|
func GetDocumentationAttributes(packageName string, contributors []Contributor) Attributes {
|
|
return Attributes{
|
|
Name: GetPackageName(packageName, false),
|
|
Contributors: contributors,
|
|
NameURL: GetGoDocURL(packageName),
|
|
Year: time.Now().Year(),
|
|
CapitalName: GetPackageName(packageName, true),
|
|
DonationAddress: core.BitcoinDonationAddress,
|
|
}
|
|
}
|
|
|
|
// GetPackageName returns the package name after cleaning path as a string
|
|
func GetPackageName(name string, capital bool) string {
|
|
newStrings := strings.Split(name, " ")
|
|
var i int
|
|
if len(newStrings) > 1 {
|
|
// retrieve the latest spacing to define the most childish package name
|
|
i = len(newStrings) - 1
|
|
}
|
|
if capital {
|
|
return strings.Replace(strings.Title(newStrings[i]), "_", " ", -1)
|
|
}
|
|
return newStrings[i]
|
|
}
|
|
|
|
// GetGoDocURL returns a string for godoc package names
|
|
func GetGoDocURL(name string) string {
|
|
if strings.Contains(name, " ") {
|
|
return strings.Join(strings.Split(name, " "), "/")
|
|
}
|
|
if name == "testdata" ||
|
|
name == "tools" ||
|
|
name == ContributorFile ||
|
|
name == LicenseFile {
|
|
return ""
|
|
}
|
|
return name
|
|
}
|
|
|
|
// UpdateDocumentation generates or updates readme/documentation files across
|
|
// the codebase
|
|
func UpdateDocumentation(details DocumentationDetails) {
|
|
for i := range details.Directories {
|
|
cutSet := details.Directories[i][len(repoDir):]
|
|
if cutSet != "" && cutSet[0] == os.PathSeparator {
|
|
cutSet = cutSet[1:]
|
|
}
|
|
|
|
data := strings.Split(cutSet, string(os.PathSeparator))
|
|
|
|
var temp []string
|
|
for x := range data {
|
|
if data[x] == ".." {
|
|
continue
|
|
}
|
|
if data[x] == "" {
|
|
break
|
|
}
|
|
temp = append(temp, data[x])
|
|
}
|
|
|
|
var name string
|
|
if len(temp) == 0 {
|
|
name = "root"
|
|
} else {
|
|
name = strings.Join(temp, " ")
|
|
}
|
|
|
|
if IsExcluded(name, details.Config.Exclusions.Files) {
|
|
if verbose {
|
|
fmt.Println("Excluding file:", name)
|
|
}
|
|
continue
|
|
}
|
|
if name == engineFolder {
|
|
d, err := ioutil.ReadDir(details.Directories[i])
|
|
if err != nil {
|
|
fmt.Println("Excluding file:", err)
|
|
}
|
|
for x := range d {
|
|
nameSplit := strings.Split(d[x].Name(), ".go")
|
|
engineTemplateName := engineFolder + " " + nameSplit[0]
|
|
if details.Tmpl.Lookup(engineTemplateName) == nil {
|
|
fmt.Printf("Template not found for path %s create new template with {{define \"%s\" -}} TEMPLATE HERE {{end}}\n",
|
|
details.Directories[i],
|
|
name)
|
|
continue
|
|
}
|
|
err = runTemplate(details, filepath.Join(details.Directories[i], nameSplit[0]+".md"), engineTemplateName)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
}
|
|
}
|
|
continue
|
|
}
|
|
if details.Tmpl.Lookup(name) == nil {
|
|
fmt.Printf("Template not found for path %s create new template with {{define \"%s\" -}} TEMPLATE HERE {{end}}\n",
|
|
details.Directories[i],
|
|
name)
|
|
continue
|
|
}
|
|
var mainPath string
|
|
switch {
|
|
case name == LicenseFile || name == ContributorFile:
|
|
mainPath = details.Directories[i]
|
|
default:
|
|
mainPath = filepath.Join(details.Directories[i], "README.md")
|
|
}
|
|
|
|
if err := runTemplate(details, mainPath, name); err != nil {
|
|
log.Println(err)
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
func runTemplate(details DocumentationDetails, mainPath, name string) error {
|
|
err := os.Remove(mainPath)
|
|
if err != nil && !(strings.Contains(err.Error(), "no such file or directory") ||
|
|
strings.Contains(err.Error(), "The system cannot find the file specified.")) {
|
|
return err
|
|
}
|
|
|
|
f, err := os.Create(mainPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func(f *os.File) {
|
|
err := f.Close()
|
|
if err != nil {
|
|
log.Printf("could not close file %s: %v", mainPath, err)
|
|
}
|
|
}(f)
|
|
|
|
attr := GetDocumentationAttributes(name, details.Contributors)
|
|
return details.Tmpl.ExecuteTemplate(f, name, attr)
|
|
}
|