aboutsummaryrefslogtreecommitdiff
path: root/ccwl
diff options
context:
space:
mode:
authorArun Isaac2021-12-13 17:26:25 +0530
committerArun Isaac2021-12-13 18:05:36 +0530
commitb42c56555a06cdd75070fcef64587f67f09f91a9 (patch)
treec45618411b8d5893fedd3f7e950a1ce0424a70f3 /ccwl
parent781e504bcc50bee661cecd55377b55ab2834e01d (diff)
downloadccwl-b42c56555a06cdd75070fcef64587f67f09f91a9.tar.gz
ccwl-b42c56555a06cdd75070fcef64587f67f09f91a9.tar.lz
ccwl-b42c56555a06cdd75070fcef64587f67f09f91a9.zip
ccwl: Support graphviz HTML strings.
* ccwl/graphviz.scm (<html-string>): New type. (escape-id): Delete function. (serialize): New function. (graph->dot): Use serialize instead of escape-id. * tests/graphviz.scm: New file. ("serialize HTML strings correctly"): New test case. * Makefile.am (SCM_TESTS): Register it.
Diffstat (limited to 'ccwl')
-rw-r--r--ccwl/graphviz.scm48
1 files changed, 31 insertions, 17 deletions
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)