about summary refs log tree commit diff
diff options
context:
space:
mode:
-rw-r--r--Makefile.am2
-rw-r--r--ccwl/graphviz.scm48
-rw-r--r--tests/graphviz.scm48
3 files changed, 80 insertions, 18 deletions
diff --git a/Makefile.am b/Makefile.am
index 9250230..218ccc7 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -75,7 +75,7 @@ bin_SCRIPTS = scripts/ccwl
 
 SOURCES = ccwl/ccwl.scm ccwl/cwl.scm ccwl/graphviz.scm ccwl/yaml.scm ccwl/utils.scm
 TEST_EXTENSIONS = .scm
-SCM_TESTS = tests/ccwl.scm tests/utils.scm tests/yaml.scm
+SCM_TESTS = tests/ccwl.scm tests/graphviz.scm tests/utils.scm tests/yaml.scm
 TESTS = $(SCM_TESTS)
 
 SCM_LOG_DRIVER =		\
diff --git a/ccwl/graphviz.scm b/ccwl/graphviz.scm
index 7093bdb..fbef971 100644
--- a/ccwl/graphviz.scm
+++ b/ccwl/graphviz.scm
@@ -62,6 +62,11 @@ language."
   "Construct <graph-node> object."
   (make-graph-node name properties))
 
+(define-immutable-record-type <html-string>
+  (html-string str)
+  html-string?
+  (str html-string-underlying))
+
 (define (workflow->graph workflow)
   "Convert WORKFLOW, a <workflow> object, to a <graph> object."
   (graph 'workflow
@@ -112,18 +117,27 @@ language."
                                                                (style . "filled"))))
                                                (workflow-outputs workflow))))))
 
-(define (escape-id id)
-  "Escape string ID if necessary according to graphviz dot syntax."
-  (let ((id (if (symbol? id)
-                (symbol->string id)
-                id)))
-    (if (string-every (char-set-union (char-set-intersection char-set:letter+digit
-                                                             char-set:ascii)
-                                      (char-set #\_))
-                      id)
-        id
-        (call-with-output-string
-          (cut write id <>)))))
+(define (serialize object)
+  "Serialize OBJECT according to graphviz dot syntax. OBJECT may a
+symbol, a string, or a <html-string> object."
+  (let ((str (cond
+              ((symbol? object) (symbol->string object))
+              ((string? object) object)
+              ((html-string? object) (html-string-underlying object))
+              (else (error "Unknown object type to serialize to graphviz dot:" object)))))
+    (cond
+     ;; Surround HTML strings in <>, and don't escape.
+     ((html-string? object)
+      (format "<~a>" str))
+     ;; Don't escape safe strings.
+     ((string-every (char-set-union (char-set-intersection char-set:letter+digit
+                                                           char-set:ascii)
+                                    (char-set #\_))
+                    str)
+      str)
+     ;; Escape strings with unsafe characters.
+     (else (call-with-output-string
+             (cut write str <>))))))
 
 (define* (graph->dot graph #:optional (port (current-output-port)) (level 0))
   "Render GRAPH, a <graph> object, in the graphviz dot syntax to
@@ -136,16 +150,16 @@ PORT."
   (for-each (match-lambda
               ((key . value)
                (indent-level port (1+ level))
-               (display (format "~a=~a;~%" key (escape-id value)) port)))
+               (display (format "~a=~a;~%" key (serialize value)) port)))
             (graph-properties graph))
   (for-each (lambda (node)
               (indent-level port (1+ level))
-              (display (escape-id (graph-node-name node)) port)
+              (display (serialize (graph-node-name node)) port)
               (unless (null? (graph-node-properties node))
                 (display (format " [~a]"
                                  (string-join (map (match-lambda
                                                      ((key . value)
-                                                      (format "~a=~a" key (escape-id value))))
+                                                      (format "~a=~a" key (serialize value))))
                                                    (graph-node-properties node))
                                               ", "))
                          port))
@@ -155,8 +169,8 @@ PORT."
               ((from . to)
                (indent-level port (1+ level))
                (display (format "~a -> ~a;~%"
-                                (escape-id from)
-                                (escape-id to))
+                                (serialize from)
+                                (serialize to))
                         port)))
             (graph-edges graph))
   (for-each (lambda (subgraph)
diff --git a/tests/graphviz.scm b/tests/graphviz.scm
new file mode 100644
index 0000000..5853616
--- /dev/null
+++ b/tests/graphviz.scm
@@ -0,0 +1,48 @@
+;;; ccwl --- Concise Common Workflow Language
+;;; Copyright © 2021 Arun Isaac <arunisaac@systemreboot.net>
+;;;
+;;; This file is part of ccwl.
+;;;
+;;; ccwl is free software: you can redistribute it and/or modify it
+;;; under the terms of the GNU General Public License as published by
+;;; the Free Software Foundation, either version 3 of the License, or
+;;; (at your option) any later version.
+;;;
+;;; ccwl is distributed in the hope that it will be useful, but
+;;; WITHOUT ANY WARRANTY; without even the implied warranty of
+;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+;;; General Public License for more details.
+;;;
+;;; You should have received a copy of the GNU General Public License
+;;; along with ccwl.  If not, see <https://www.gnu.org/licenses/>.
+
+(use-modules (srfi srfi-64))
+
+(define graph->dot
+  (@@ (ccwl graphviz) graph->dot))
+
+(define graph
+  (@@ (ccwl graphviz) graph))
+
+(define graph-node
+  (@@ (ccwl graphviz) graph-node))
+
+(define html-string
+  (@@ (ccwl graphviz) html-string))
+
+(test-begin "graphviz")
+
+(test-equal "serialize HTML strings correctly"
+  "digraph foo {
+  bar [label=<<table><tr><td>bar</td></tr></table>>];
+}
+"
+  (call-with-output-string
+    (lambda (port)
+      (graph->dot
+       (graph 'foo
+              #:nodes (list (graph-node 'bar
+                                        `((label . ,(html-string "<table><tr><td>bar</td></tr></table>"))))))
+       port))))
+
+(test-end "graphviz")