From 4ad6649c5087f6be9117af4473de8dabf585792e Mon Sep 17 00:00:00 2001 From: mazrean Date: Fri, 6 Dec 2024 21:11:13 +0900 Subject: [PATCH 1/5] add webui --- .gitignore | 3 +- dbdoc/dbdoc.go | 22 +---- dbdoc/mermaid.go | 165 -------------------------------- internal/ui/asset/template.html | 27 ++++++ internal/ui/asset/template.md | 7 ++ internal/ui/mermaid.go | 133 +++++++++++++++++++++++++ internal/ui/render.go | 94 ++++++++++++++++++ main.go | 29 +++++- 8 files changed, 296 insertions(+), 184 deletions(-) delete mode 100644 dbdoc/mermaid.go create mode 100644 internal/ui/asset/template.html create mode 100644 internal/ui/asset/template.md create mode 100644 internal/ui/mermaid.go create mode 100644 internal/ui/render.go diff --git a/.gitignore b/.gitignore index 53c37a1..a1ed284 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,2 @@ -dist \ No newline at end of file +dist +/isucrud diff --git a/dbdoc/dbdoc.go b/dbdoc/dbdoc.go index 3c478c6..b0aaa7c 100644 --- a/dbdoc/dbdoc.go +++ b/dbdoc/dbdoc.go @@ -3,7 +3,6 @@ package dbdoc import ( "fmt" "go/token" - "os" "golang.org/x/tools/go/packages" "golang.org/x/tools/go/ssa" @@ -19,7 +18,7 @@ type Config struct { DestinationFilePath string } -func Run(conf Config) error { +func Run(conf Config) ([]*Node, error) { ctx := &Context{ FileSet: token.NewFileSet(), WorkDir: conf.WorkDir, @@ -27,17 +26,17 @@ func Run(conf Config) error { ssaProgram, pkgs, err := BuildSSA(ctx, conf.BuildArgs) if err != nil { - return fmt.Errorf("failed to build ssa: %w", err) + return nil, fmt.Errorf("failed to build ssa: %w", err) } loopRangeMap, err := BuildLoopRangeMap(ctx) if err != nil { - return fmt.Errorf("failed to build loop range map: %w", err) + return nil, fmt.Errorf("failed to build loop range map: %w", err) } funcs, err := BuildFuncs(ctx, pkgs, ssaProgram, loopRangeMap) if err != nil { - return fmt.Errorf("failed to build funcs: %w", err) + return nil, fmt.Errorf("failed to build funcs: %w", err) } nodes := BuildGraph( @@ -46,18 +45,7 @@ func Run(conf Config) error { conf.IgnoreMain, conf.IgnoreInitialize, ) - f, err := os.Create(conf.DestinationFilePath) - if err != nil { - return fmt.Errorf("failed to make directory: %w", err) - } - defer f.Close() - - err = WriteMermaid(f, nodes) - if err != nil { - return fmt.Errorf("failed to write mermaid: %w", err) - } - - return nil + return nodes, nil } func BuildSSA(ctx *Context, args []string) (*ssa.Program, []*packages.Package, error) { diff --git a/dbdoc/mermaid.go b/dbdoc/mermaid.go deleted file mode 100644 index a65944a..0000000 --- a/dbdoc/mermaid.go +++ /dev/null @@ -1,165 +0,0 @@ -package dbdoc - -import ( - "fmt" - "io" - "log" - "strconv" - "strings" -) - -const ( - funcNodeColor = "1976D2" - tableNodeColor = "795548" - insertLinkColor = "CDDC39" - deleteLinkColor = "F44336" - selectLinkColor = "78909C" - updateLinkColor = "FF9800" - callLinkColor = "BBDEFB" -) - -var ( - nodeTypes = []struct { - name string - label string - color string - valid bool - }{ - NodeTypeTable: {"table", "テーブル", tableNodeColor, true}, - NodeTypeFunction: {"func", "関数", funcNodeColor, true}, - } - edgeTypes = []struct { - label string - color string - valid bool - }{ - EdgeTypeInsert: {"INSERT", insertLinkColor, true}, - EdgeTypeUpdate: {"UPDATE", updateLinkColor, true}, - EdgeTypeDelete: {"DELETE", deleteLinkColor, true}, - EdgeTypeSelect: {"SELECT", selectLinkColor, true}, - EdgeTypeCall: {"関数呼び出し", callLinkColor, true}, - } -) - -func WriteMermaid(w io.StringWriter, nodes []*Node) error { - _, err := w.WriteString("# DB Graph\n") - if err != nil { - return fmt.Errorf("failed to write header: %w", err) - } - - _, err = w.WriteString("node: ") - if err != nil { - return fmt.Errorf("failed to write node description start: %w", err) - } - - for _, nodeType := range nodeTypes { - if nodeType.valid { - _, err = w.WriteString(fmt.Sprintf("![](https://via.placeholder.com/16/%s/FFFFFF/?text=%%20) `%s` ", nodeType.color, nodeType.label)) - if err != nil { - return fmt.Errorf("failed to write node description: %w", err) - } - } - } - - _, err = w.WriteString("\n\n") - if err != nil { - return fmt.Errorf("failed to write node description end: %w", err) - } - - _, err = w.WriteString("edge: ") - if err != nil { - return fmt.Errorf("failed to write edge description start: %w", err) - } - - for _, edgeType := range edgeTypes { - if edgeType.valid { - _, err = w.WriteString(fmt.Sprintf("![](https://via.placeholder.com/16/%s/FFFFFF/?text=%%20) `%s` ", edgeType.color, edgeType.label)) - if err != nil { - return fmt.Errorf("failed to write edge description: %w", err) - } - } - } - - _, err = w.WriteString("\n") - if err != nil { - return fmt.Errorf("failed to write edge description end: %w", err) - } - - _, err = w.WriteString("```mermaid\n" + - "graph LR\n") - if err != nil { - return fmt.Errorf("failed to write header: %w", err) - } - - for _, nodeType := range nodeTypes { - if nodeType.valid { - _, err = w.WriteString(fmt.Sprintf(" classDef %s fill:#%s,fill-opacity:0.5\n", nodeType.name, nodeType.color)) - if err != nil { - return fmt.Errorf("failed to write class def: %w", err) - } - } - } - - edgeLinksMap := map[EdgeType][]string{} - edgeID := 0 - for _, node := range nodes { - var src string - if nodeType := nodeTypes[node.NodeType]; nodeType.valid { - src = fmt.Sprintf("%s[%s]:::%s", node.ID, node.Label, nodeType.name) - } else { - log.Printf("unknown node type: %v\n", node.NodeType) - continue - } - - for _, edge := range node.Edges { - var dst string - if nodeType := nodeTypes[edge.Node.NodeType]; nodeType.valid { - dst = fmt.Sprintf("%s[%s]:::%s", edge.Node.ID, edge.Node.Label, nodeType.name) - } else { - log.Printf("unknown node type: %v\n", node.NodeType) - continue - } - - line := "--" - if edge.InLoop { - line = "==" - } - - var edgeExpr string - if edge.Label == "" { - edgeExpr = fmt.Sprintf("%s>", line) - } else { - edgeExpr = fmt.Sprintf("%s %s %s>", line, edge.Label, line) - } - _, err = w.WriteString(fmt.Sprintf(" %s %s %s\n", src, edgeExpr, dst)) - if err != nil { - return fmt.Errorf("failed to write edge: %w", err) - } - - edgeLinksMap[edge.EdgeType] = append(edgeLinksMap[edge.EdgeType], strconv.Itoa(edgeID)) - - edgeID++ - } - } - - for edgeType, links := range edgeLinksMap { - if len(links) == 0 { - continue - } - if info := edgeTypes[edgeType]; info.valid { - _, err = w.WriteString(fmt.Sprintf(" linkStyle %s stroke:#%s,stroke-width:2px\n", strings.Join(links, ","), info.color)) - if err != nil { - return fmt.Errorf("failed to write link style: %w", err) - } - } else { - log.Printf("unknown edge type: %v\n", edgeType) - } - } - - _, err = w.WriteString("```") - if err != nil { - return fmt.Errorf("failed to write footer: %w", err) - } - - return nil -} diff --git a/internal/ui/asset/template.html b/internal/ui/asset/template.html new file mode 100644 index 0000000..4ac41fd --- /dev/null +++ b/internal/ui/asset/template.html @@ -0,0 +1,27 @@ + + + + + + Mermaid + + + + + +
{{.MermaidData}}
+ + + \ No newline at end of file diff --git a/internal/ui/asset/template.md b/internal/ui/asset/template.md new file mode 100644 index 0000000..ccd3222 --- /dev/null +++ b/internal/ui/asset/template.md @@ -0,0 +1,7 @@ +# DB Graph +node: {{range .NodeTypes}}![](https://via.placeholder.com/16/{{.Color}}/FFFFFF/?text=%20) `{{.Label}}` {{end}} + +edge: {{range .EdgeTypes}}![](https://via.placeholder.com/16/{{.Color}}/FFFFFF/?text=%20) `{{.Label}}` {{end}} +```mermaid +{{.MermaidData}} +``` diff --git a/internal/ui/mermaid.go b/internal/ui/mermaid.go new file mode 100644 index 0000000..2f8d4f1 --- /dev/null +++ b/internal/ui/mermaid.go @@ -0,0 +1,133 @@ +package ui + +import ( + "fmt" + "io" + "log" + "strconv" + "strings" + + "github.com/mazrean/isucrud/dbdoc" +) + +const ( + funcNodeColor = "1976D2" + tableNodeColor = "795548" + insertLinkColor = "CDDC39" + deleteLinkColor = "F44336" + selectLinkColor = "78909C" + updateLinkColor = "FF9800" + callLinkColor = "BBDEFB" +) + +type NodeType struct { + name string + Label string + Color string + valid bool +} + +type EdgeType struct { + Label string + Color string + valid bool +} + +var ( + nodeTypes = []NodeType{ + dbdoc.NodeTypeTable: {"table", "テーブル", tableNodeColor, true}, + dbdoc.NodeTypeFunction: {"func", "関数", funcNodeColor, true}, + } + edgeTypes = []EdgeType{ + dbdoc.EdgeTypeInsert: {"INSERT", insertLinkColor, true}, + dbdoc.EdgeTypeUpdate: {"UPDATE", updateLinkColor, true}, + dbdoc.EdgeTypeDelete: {"DELETE", deleteLinkColor, true}, + dbdoc.EdgeTypeSelect: {"SELECT", selectLinkColor, true}, + dbdoc.EdgeTypeCall: {"関数呼び出し", callLinkColor, true}, + } +) + +func RenderMermaid( + w io.StringWriter, + nodes []*dbdoc.Node, + targetNodeIDs []string, + isHttp bool, +) error { + _, err := w.WriteString("graph LR\n") + if err != nil { + return fmt.Errorf("failed to write header: %w", err) + } + + for _, nodeType := range nodeTypes { + if nodeType.valid { + _, err = w.WriteString(fmt.Sprintf(" classDef %s fill:#%s,fill-opacity:0.5\n", nodeType.name, nodeType.Color)) + if err != nil { + return fmt.Errorf("failed to write class def: %w", err) + } + } + } + + edgeLinksMap := map[dbdoc.EdgeType][]string{} + edgeID := 0 + for _, node := range nodes { + var src string + if nodeType := nodeTypes[node.NodeType]; nodeType.valid { + src = fmt.Sprintf("%s[%s]:::%s", node.ID, node.Label, nodeType.name) + } else { + log.Printf("unknown node type: %v\n", node.NodeType) + continue + } + + for _, edge := range node.Edges { + var dst string + if nodeType := nodeTypes[edge.Node.NodeType]; nodeType.valid { + dst = fmt.Sprintf("%s[%s]:::%s", edge.Node.ID, edge.Node.Label, nodeType.name) + } else { + log.Printf("unknown node type: %v\n", node.NodeType) + continue + } + + line := "--" + if edge.InLoop { + line = "==" + } + + var edgeExpr string + if edge.Label == "" { + edgeExpr = fmt.Sprintf("%s>", line) + } else { + edgeExpr = fmt.Sprintf("%s %s %s>", line, edge.Label, line) + } + _, err = w.WriteString(fmt.Sprintf(" %s %s %s\n", src, edgeExpr, dst)) + if err != nil { + return fmt.Errorf("failed to write edge: %w", err) + } + + edgeLinksMap[edge.EdgeType] = append(edgeLinksMap[edge.EdgeType], strconv.Itoa(edgeID)) + + edgeID++ + } + } + + if isHttp { + for _, node := range nodes { + w.WriteString(fmt.Sprintf(" click %s \"/?node=%s\"\n", node.ID, node.ID)) + } + } + + for edgeType, links := range edgeLinksMap { + if len(links) == 0 { + continue + } + if info := edgeTypes[edgeType]; info.valid { + _, err = w.WriteString(fmt.Sprintf(" linkStyle %s stroke:#%s,stroke-width:2px\n", strings.Join(links, ","), info.Color)) + if err != nil { + return fmt.Errorf("failed to write link style: %w", err) + } + } else { + log.Printf("unknown edge type: %v\n", edgeType) + } + } + + return nil +} diff --git a/internal/ui/render.go b/internal/ui/render.go new file mode 100644 index 0000000..3525a7d --- /dev/null +++ b/internal/ui/render.go @@ -0,0 +1,94 @@ +package ui + +import ( + _ "embed" + "fmt" + htmlTemplate "html/template" + "io" + "os" + "strings" + "text/template" + + "github.com/mazrean/isucrud/dbdoc" +) + +var ( + //go:embed asset/template.md + templateMarkdown string + //go:embed asset/template.html + templateHTML string +) + +type TemplateParam struct { + NodeTypes []NodeType + EdgeTypes []EdgeType + Nodes []*dbdoc.Node + MermaidData string +} + +func RenderMarkdown(dest string, nodes []*dbdoc.Node) error { + f, err := os.Create(dest) + if err != nil { + return fmt.Errorf("failed to make directory: %w", err) + } + defer f.Close() + + sb := &strings.Builder{} + err = RenderMermaid( + sb, + nodes, + nil, + false, + ) + if err != nil { + return fmt.Errorf("failed to write mermaid: %w", err) + } + + tmpl, err := template.New("markdown").Parse(templateMarkdown) + if err != nil { + return fmt.Errorf("failed to parse template: %w", err) + } + + err = tmpl.Execute(f, TemplateParam{ + NodeTypes: nodeTypes, + EdgeTypes: edgeTypes, + Nodes: nodes, + MermaidData: sb.String(), + }) + if err != nil { + return fmt.Errorf("failed to execute template: %w", err) + } + + return nil +} + +func RenderHTML(w io.Writer, nodes []*dbdoc.Node) error { + + sb := &strings.Builder{} + err := RenderMermaid( + sb, + nodes, + nil, + true, + ) + if err != nil { + return fmt.Errorf("failed to write mermaid: %w", err) + } + + tmpl, err := htmlTemplate.New("html").Parse(templateHTML) + if err != nil { + return fmt.Errorf("failed to parse template: %w", err) + } + + err = tmpl.Execute(w, TemplateParam{ + NodeTypes: nodeTypes, + EdgeTypes: edgeTypes, + Nodes: nodes, + MermaidData: sb.String(), + }) + if err != nil { + return fmt.Errorf("failed to execute template: %w", err) + } + + return nil +} diff --git a/main.go b/main.go index b84f054..86b00d4 100644 --- a/main.go +++ b/main.go @@ -3,9 +3,12 @@ package main import ( "flag" "fmt" + "log" + "net/http" "os" "github.com/mazrean/isucrud/dbdoc" + "github.com/mazrean/isucrud/internal/ui" ) var ( @@ -17,6 +20,8 @@ var ( ignores sliceString ignorePrefixes sliceString ignoreMain, ignoreInitialize bool + web bool + addr string ) func init() { @@ -27,6 +32,8 @@ func init() { flag.Var(&ignorePrefixes, "ignorePrefix", "ignore function") flag.BoolVar(&ignoreMain, "ignoreMain", true, "ignore main function") flag.BoolVar(&ignoreInitialize, "ignoreInitialize", true, "ignore functions with 'initialize' in the name") + flag.BoolVar(&web, "web", false, "run as web server") + flag.StringVar(&addr, "addr", "localhost:7070", "address to listen on") } func main() { @@ -42,7 +49,7 @@ func main() { panic(fmt.Errorf("failed to get working directory: %w", err)) } - err = dbdoc.Run(dbdoc.Config{ + nodes, err := dbdoc.Run(dbdoc.Config{ WorkDir: wd, BuildArgs: flag.Args(), IgnoreFuncs: ignores, @@ -54,4 +61,24 @@ func main() { if err != nil { panic(fmt.Errorf("failed to run dbdoc: %w", err)) } + + if web { + mux := http.NewServeMux() + mux.Handle("GET /", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + r.Header.Set("Content-Type", "text/html") + + err := ui.RenderHTML(w, nodes) + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + } + })) + + log.Printf("open http://%s/ in your browser\n", addr) + http.ListenAndServe(addr, mux) + } else { + err = ui.RenderMarkdown(dst, nodes) + if err != nil { + panic(fmt.Errorf("failed to render markdown: %w", err)) + } + } } From 216da18f40be98a6014c89206dbad9594e5b1081 Mon Sep 17 00:00:00 2001 From: mazrean Date: Sat, 7 Dec 2024 14:51:24 +0900 Subject: [PATCH 2/5] =?UTF-8?q?=E3=83=A9=E3=83=99=E3=83=AB=E4=BF=AE?= =?UTF-8?q?=E6=AD=A3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- internal/ui/asset/template.html | 13 +++++++++++++ internal/ui/render.go | 8 ++++---- 2 files changed, 17 insertions(+), 4 deletions(-) diff --git a/internal/ui/asset/template.html b/internal/ui/asset/template.html index 4ac41fd..9b3db9a 100644 --- a/internal/ui/asset/template.html +++ b/internal/ui/asset/template.html @@ -18,6 +18,19 @@ +
+ nodes: + {{range .NodeTypes}} + + {{.Label}} + {{end}} +
+ edges: + {{range .EdgeTypes}} + + {{.Label}} + {{end}} +
Date: Sat, 7 Dec 2024 15:24:21 +0900 Subject: [PATCH 3/5] =?UTF-8?q?=E3=82=B5=E3=83=96=E3=82=B0=E3=83=A9?= =?UTF-8?q?=E3=83=95=E5=8F=96=E3=82=8A=E5=87=BA=E3=81=97=E6=A9=9F=E8=83=BD?= =?UTF-8?q?=E5=AE=9F=E8=A3=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- go.mod | 2 +- internal/pkg/list/queue.go | 13 ++++++ internal/ui/mermaid.go | 1 - internal/ui/render.go | 87 ++++++++++++++++++++++++++++++++++++-- main.go | 4 +- 5 files changed, 100 insertions(+), 7 deletions(-) diff --git a/go.mod b/go.mod index c57c850..f917535 100644 --- a/go.mod +++ b/go.mod @@ -1,6 +1,6 @@ module github.com/mazrean/isucrud -go 1.22.0 +go 1.23 toolchain go1.22.9 diff --git a/internal/pkg/list/queue.go b/internal/pkg/list/queue.go index 0c13853..1f76794 100644 --- a/internal/pkg/list/queue.go +++ b/internal/pkg/list/queue.go @@ -39,6 +39,19 @@ func (q Queue[T]) Peek() (T, bool) { return e.Value() } +func (q *Queue[T]) Iter(yield func(T) bool) { + for e := q.l.Front(); e.e != nil; e = q.l.Front() { + q.l.Remove(e) + + v, ok := e.Value() + if !ok { + break + } + + yield(v) + } +} + func (q Queue[T]) Clear() { q.l.Init() } diff --git a/internal/ui/mermaid.go b/internal/ui/mermaid.go index 2f8d4f1..476cea6 100644 --- a/internal/ui/mermaid.go +++ b/internal/ui/mermaid.go @@ -50,7 +50,6 @@ var ( func RenderMermaid( w io.StringWriter, nodes []*dbdoc.Node, - targetNodeIDs []string, isHttp bool, ) error { _, err := w.WriteString("graph LR\n") diff --git a/internal/ui/render.go b/internal/ui/render.go index 5c7ea2d..be3ea5e 100644 --- a/internal/ui/render.go +++ b/internal/ui/render.go @@ -10,6 +10,7 @@ import ( "text/template" "github.com/mazrean/isucrud/dbdoc" + "github.com/mazrean/isucrud/internal/pkg/list" ) var ( @@ -37,7 +38,6 @@ func RenderMarkdown(dest string, nodes []*dbdoc.Node) error { err = RenderMermaid( sb, nodes, - nil, false, ) if err != nil { @@ -62,13 +62,16 @@ func RenderMarkdown(dest string, nodes []*dbdoc.Node) error { return nil } -func RenderHTML(w io.Writer, nodes []*dbdoc.Node) error { +func RenderHTML(w io.Writer, nodes []*dbdoc.Node, targetNodeID string) error { + filteredNodes := nodes + if targetNodeID != "" { + filteredNodes = filterNodes(targetNodeID, nodes) + } sb := &strings.Builder{} err := RenderMermaid( sb, - nodes, - nil, + filteredNodes, true, ) if err != nil { @@ -92,3 +95,79 @@ func RenderHTML(w io.Writer, nodes []*dbdoc.Node) error { return nil } + +func filterNodes(targetNodeID string, nodes []*dbdoc.Node) []*dbdoc.Node { + nodeMap := make(map[string]*dbdoc.Node) + for _, node := range nodes { + nodeMap[node.ID] = node + } + + targetNode, exists := nodeMap[targetNodeID] + if !exists { + return []*dbdoc.Node{} + } + + parentMap := make(map[string][]*dbdoc.Node) + for _, node := range nodes { + for _, edge := range node.Edges { + if edge.Node != nil { + parentMap[edge.Node.ID] = append(parentMap[edge.Node.ID], node) + } + } + } + + visited := make(map[string]struct{}, len(nodes)) + relatedNodeMap := make(map[string]*dbdoc.Node) + + childNodeQueue := list.NewQueue[*dbdoc.Node]() + childNodeQueue.Push(targetNode) + for current := range childNodeQueue.Iter { + // Skip if already visited. + if _, ok := visited[current.ID]; ok { + continue + } + visited[current.ID] = struct{}{} + + relatedNodeMap[current.ID] = current + + for _, edge := range current.Edges { + childNodeQueue.Push(edge.Node) + } + } + + visited = make(map[string]struct{}, len(nodes)) + parentNodeQueue := list.NewQueue[*dbdoc.Node]() + parentNodeQueue.Push(targetNode) + for current := range parentNodeQueue.Iter { + // Skip if already visited. + if _, ok := visited[current.ID]; ok { + continue + } + visited[current.ID] = struct{}{} + + relatedNodeMap[current.ID] = current + + for _, parent := range parentMap[current.ID] { + parentNodeQueue.Push(parent) + } + } + + relatedNodes := make([]*dbdoc.Node, 0, len(relatedNodeMap)) + for _, node := range relatedNodeMap { + newEdges := []dbdoc.Edge{} + for _, edge := range node.Edges { + if _, ok := relatedNodeMap[edge.Node.ID]; ok { + newEdges = append(newEdges, edge) + } + } + + relatedNodes = append(relatedNodes, &dbdoc.Node{ + ID: node.ID, + Label: node.Label, + Edges: newEdges, + NodeType: node.NodeType, + }) + } + + return relatedNodes +} diff --git a/main.go b/main.go index 86b00d4..0fd6a56 100644 --- a/main.go +++ b/main.go @@ -67,7 +67,9 @@ func main() { mux.Handle("GET /", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { r.Header.Set("Content-Type", "text/html") - err := ui.RenderHTML(w, nodes) + targetNodeID := r.URL.Query().Get("node") + + err := ui.RenderHTML(w, nodes, targetNodeID) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) } From 140fb33126bd156b312fef84ba1b8dd29275e5cd Mon Sep 17 00:00:00 2001 From: mazrean Date: Sat, 7 Dec 2024 15:30:46 +0900 Subject: [PATCH 4/5] =?UTF-8?q?=E3=83=95=E3=82=A3=E3=83=AB=E3=82=BF?= =?UTF-8?q?=E3=83=BC=E8=A7=A3=E9=99=A4=E3=83=9C=E3=82=BF=E3=83=B3=E8=BF=BD?= =?UTF-8?q?=E5=8A=A0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- internal/ui/asset/template.html | 1 + internal/ui/render.go | 5 +++++ 2 files changed, 6 insertions(+) diff --git a/internal/ui/asset/template.html b/internal/ui/asset/template.html index 9b3db9a..55de7a6 100644 --- a/internal/ui/asset/template.html +++ b/internal/ui/asset/template.html @@ -18,6 +18,7 @@ + {{if .IsFiltered}}{{end}}
nodes: {{range .NodeTypes}} diff --git a/internal/ui/render.go b/internal/ui/render.go index be3ea5e..ebdcde1 100644 --- a/internal/ui/render.go +++ b/internal/ui/render.go @@ -21,6 +21,7 @@ var ( ) type TemplateParam struct { + IsFiltered bool NodeTypes []NodeType EdgeTypes []EdgeType Nodes []*dbdoc.Node @@ -50,6 +51,7 @@ func RenderMarkdown(dest string, nodes []*dbdoc.Node) error { } err = tmpl.Execute(f, TemplateParam{ + IsFiltered: false, NodeTypes: nodeTypes[1:], EdgeTypes: edgeTypes[1:], Nodes: nodes, @@ -63,8 +65,10 @@ func RenderMarkdown(dest string, nodes []*dbdoc.Node) error { } func RenderHTML(w io.Writer, nodes []*dbdoc.Node, targetNodeID string) error { + filtered := false filteredNodes := nodes if targetNodeID != "" { + filtered = true filteredNodes = filterNodes(targetNodeID, nodes) } @@ -84,6 +88,7 @@ func RenderHTML(w io.Writer, nodes []*dbdoc.Node, targetNodeID string) error { } err = tmpl.Execute(w, TemplateParam{ + IsFiltered: filtered, NodeTypes: nodeTypes[1:], EdgeTypes: edgeTypes[1:], Nodes: nodes, From 53153166e6d935b9947955bf19788770782541d4 Mon Sep 17 00:00:00 2001 From: mazrean Date: Sat, 7 Dec 2024 15:34:40 +0900 Subject: [PATCH 5/5] =?UTF-8?q?lint=E3=82=A8=E3=83=A9=E3=83=BC=E4=BF=AE?= =?UTF-8?q?=E6=AD=A3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- internal/ui/mermaid.go | 5 ++++- main.go | 4 +++- 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/internal/ui/mermaid.go b/internal/ui/mermaid.go index 476cea6..51e41f9 100644 --- a/internal/ui/mermaid.go +++ b/internal/ui/mermaid.go @@ -110,7 +110,10 @@ func RenderMermaid( if isHttp { for _, node := range nodes { - w.WriteString(fmt.Sprintf(" click %s \"/?node=%s\"\n", node.ID, node.ID)) + _, err = w.WriteString(fmt.Sprintf(" click %s \"/?node=%s\"\n", node.ID, node.ID)) + if err != nil { + return fmt.Errorf("failed to write click event: %w", err) + } } } diff --git a/main.go b/main.go index 0fd6a56..d1101e2 100644 --- a/main.go +++ b/main.go @@ -76,7 +76,9 @@ func main() { })) log.Printf("open http://%s/ in your browser\n", addr) - http.ListenAndServe(addr, mux) + if err := http.ListenAndServe(addr, mux); err != nil { + panic(fmt.Errorf("server exit: %w", err)) + } } else { err = ui.RenderMarkdown(dst, nodes) if err != nil {