├── .npmignore ├── example.png ├── .prettierrc ├── rollup.config.js ├── example.html ├── package.json ├── .gitignore ├── README.md └── lib ├── nodeTypes.js ├── lib.js └── icons ├── genericIcons.js └── databaseIcons.js /.npmignore: -------------------------------------------------------------------------------- 1 | rollup.config.js 2 | .prettierrc 3 | example.html 4 | example.png -------------------------------------------------------------------------------- /example.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/RaoulMeyer/diagram-as-code/HEAD/example.png -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "endOfLine": "lf", 3 | "semi": true, 4 | "singleQuote": true, 5 | "tabWidth": 4, 6 | "trailingComma": "es5" 7 | } 8 | -------------------------------------------------------------------------------- /rollup.config.js: -------------------------------------------------------------------------------- 1 | import resolve from '@rollup/plugin-node-resolve'; 2 | import { terser } from 'rollup-plugin-terser'; 3 | 4 | export default { 5 | input: 'lib/lib.js', 6 | output: { 7 | dir: 'dist', 8 | format: 'umd', 9 | name: 'diagram-as-code-js', 10 | }, 11 | plugins: [resolve(), terser()], 12 | }; 13 | -------------------------------------------------------------------------------- /example.html: -------------------------------------------------------------------------------- 1 | 2 | 14 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "diagram-as-code", 3 | "version": "1.0.0", 4 | "description": "This library allows you to easily create diagrams of your infrastructure in code.", 5 | "main": "dist/lib.js", 6 | "unpkg": "dist/lib.js", 7 | "directories": { 8 | "lib": "lib" 9 | }, 10 | "scripts": { 11 | "test": "echo \"Error: no test specified\" && exit 1", 12 | "build": "rollup --config rollup.config.js" 13 | }, 14 | "repository": { 15 | "type": "git", 16 | "url": "git+https://github.com/RaoulMeyer/diagram-as-code.git" 17 | }, 18 | "keywords": [ 19 | "diagrams" 20 | ], 21 | "author": "Raoul Meyer", 22 | "contributors": [ 23 | "Matthew Scott" 24 | ], 25 | "license": "MIT", 26 | "bugs": { 27 | "url": "https://github.com/RaoulMeyer/diagram-as-code/issues" 28 | }, 29 | "homepage": "https://github.com/RaoulMeyer/diagram-as-code#readme", 30 | "dependencies": {}, 31 | "devDependencies": { 32 | "@rollup/plugin-node-resolve": "^6.1.0", 33 | "rollup": "^1.28.0", 34 | "rollup-plugin-terser": "^5.1.3", 35 | "vis-network": "^6.5.0" 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | lerna-debug.log* 8 | .vscode 9 | 10 | # Built files 11 | dist/ 12 | 13 | # Diagnostic reports (https://nodejs.org/api/report.html) 14 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json 15 | 16 | # Runtime data 17 | pids 18 | *.pid 19 | *.seed 20 | *.pid.lock 21 | 22 | # Directory for instrumented libs generated by jscoverage/JSCover 23 | lib-cov 24 | 25 | # Coverage directory used by tools like istanbul 26 | coverage 27 | 28 | # nyc test coverage 29 | .nyc_output 30 | 31 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) 32 | .grunt 33 | 34 | # Bower dependency directory (https://bower.io/) 35 | bower_components 36 | 37 | # node-waf configuration 38 | .lock-wscript 39 | 40 | # Compiled binary addons (https://nodejs.org/api/addons.html) 41 | build/Release 42 | 43 | # Dependency directories 44 | node_modules/ 45 | jspm_packages/ 46 | 47 | # TypeScript v1 declaration files 48 | typings/ 49 | 50 | # Optional npm cache directory 51 | .npm 52 | 53 | # Optional eslint cache 54 | .eslintcache 55 | 56 | # Optional REPL history 57 | .node_repl_history 58 | 59 | # Output of 'npm pack' 60 | *.tgz 61 | 62 | # Yarn Integrity file 63 | .yarn-integrity 64 | 65 | # dotenv environment variables file 66 | .env 67 | .env.test 68 | 69 | # parcel-bundler cache (https://parceljs.org/) 70 | .cache 71 | 72 | # next.js build output 73 | .next 74 | 75 | # nuxt.js build output 76 | .nuxt 77 | 78 | # vuepress build output 79 | .vuepress/dist 80 | 81 | # Serverless directories 82 | .serverless/ 83 | 84 | # FuseBox cache 85 | .fusebox/ 86 | 87 | # DynamoDB Local files 88 | .dynamodb/ 89 | 90 | # Mac .DS_Store cache 91 | .DS_Store 92 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Diagram as code 2 | 3 | This library allows you to easily create diagrams of your infrastructure in code. The library aims to make creating a new diagram and changing an existing one extremely easy, requiring only a text editor. 4 | 5 | An example of a diagram that can be created with this library: 6 | 7 | ![Example diagram](./example.png) 8 | 9 | To generate this diagram, we use the following code as specified in [example.html](./example.html): 10 | 11 | ```js 12 | const client = new Client(); 13 | const loadbalancer = new Elb(); 14 | const webserver = new Ec2Cluster(); 15 | const databases = new RdsCluster(); 16 | 17 | client.getsDataFrom(loadbalancer); 18 | loadbalancer.getsDataFrom(webserver); 19 | webserver.getsDataFrom(databases); 20 | 21 | diagram.render(); 22 | ``` 23 | 24 | ## Getting started 25 | 26 | Follow these steps to get started: 27 | 28 | - Download the file [example.html](./example.html). 29 | - Change the body to reflect your infrastructure. See the reference below for a list of what's possible. 30 | - Open it in your browser. 31 | 32 | ## Reference 33 | 34 | ### Adding nodes to a diagram 35 | 36 | To create a new node, create a new instance of the class you want: 37 | 38 | ```js 39 | const customer = new Client('Customer'); 40 | ``` 41 | 42 | The node will automatically be added to the diagram. All node types can be supplied with a label as the first argument of the constructor. 43 | 44 | ### Linking nodes together 45 | 46 | There are three ways to specify flow between nodes. An example for the nodes customer and server: 47 | 48 | ```js 49 | customer.exchangesDataWith(server); 50 | customer.getsDataFrom(server); 51 | customer.sendsDataTo(server); 52 | ``` 53 | 54 | You can call any function that ends either in `with`, `from` or `to`. For example you could instead do: 55 | 56 | ```js 57 | customer.requestsLoginPageFrom(server); 58 | ``` 59 | 60 | ### Rendering the diagram 61 | 62 | To render the diagram: 63 | 64 | ```js 65 | diagram.render(); 66 | ``` 67 | 68 | You can also send some settings via the render function: 69 | 70 | ```js 71 | diagram.render({ 72 | leftToRight: false, 73 | container: document.body 74 | }); 75 | ``` 76 | 77 | The values shown are the defaults. 78 | 79 | ### Customizing positioning 80 | 81 | In most cases, the way the nodes are layed out will make sense. If it doesn't, you can customize the layers of the nodes. Add the layer as the second parameter to any node: 82 | 83 | ```js 84 | const layerNumber = 3; 85 | const customer = new Client('Customer', layerNumber); 86 | ``` 87 | 88 | > Important: You'll have to specify the layer for all nodes in your diagram to make this work. 89 | 90 | ### Available node types 91 | 92 | The following node types are currently available: 93 | 94 | ```text 95 | Client, Server, ServerCluster, Database, DatabaseCluster, Mysql, MysqlCluster, Oracle, OracleCluster, PostgreSql, PostgreSqlCluster, Elasticsearch, ElasticsearchCluster, Ec2, Ec2Cluster, Rds, RdsCluster, Elb, S3, DynamoDb, Redshift, Cloudwatch, Elasticache, Iam, SimpleDb, Swf, Cloudfront, Sqs, Sns, Route53, StorageGateway, CloudFormation, CloudSearch, Glacier, ElasticBeanstalk, Ebs, Lambda, ApiGateway 96 | ``` 97 | 98 | Each of these will get a nice icon when you use them. If you want to add something that is not in this list, you can use the `Custom` type: 99 | 100 | ```js 101 | const instanceCount = 17; 102 | const layerNumber = 3; 103 | const customNode = new Custom( 104 | 'Custom label', 105 | 'https://custom.icon.website.org/icon.png', 106 | layerNumber, 107 | instanceCount 108 | ); 109 | ``` 110 | -------------------------------------------------------------------------------- /lib/nodeTypes.js: -------------------------------------------------------------------------------- 1 | import { clientIcon, serverIcon } from './icons/genericIcons'; 2 | 3 | import { 4 | databaseIcon, 5 | mySqlIcon, 6 | oracleIcon, 7 | postgresSqlIcon, 8 | elasticSearchIcon, 9 | } from './icons/databaseIcons'; 10 | 11 | import { 12 | ec2Icon, 13 | rdsIcon, 14 | elbIcon, 15 | s3Icon, 16 | dynamoDbIcon, 17 | redShiftIcon, 18 | cloudWatchIcon, 19 | elasticCacheIcon, 20 | iamIcon, 21 | simpleDbIcon, 22 | swfIcon, 23 | cloudFrontIcon, 24 | sqsIcon, 25 | snsIcon, 26 | route53Icon, 27 | storageGatewayIcon, 28 | cloudFormationIcon, 29 | cloudSearchIcon, 30 | glacierIcon, 31 | elasticBeanstalkIcon, 32 | ebsIcon, 33 | lambdaIcon, 34 | apiGatewayIcon, 35 | } from './icons/awsIcons'; 36 | 37 | const nodeTypes = [ 38 | /** 39 | * Generic 40 | */ 41 | { 42 | name: 'Client', 43 | label: 'Client', 44 | image: clientIcon, 45 | count: 1, 46 | }, 47 | 48 | { 49 | name: 'Server', 50 | label: 'Server', 51 | image: serverIcon, 52 | count: 1, 53 | }, 54 | { 55 | name: 'ServerCluster', 56 | label: 'Server', 57 | image: serverIcon, 58 | count: 2, 59 | }, 60 | 61 | { 62 | name: 'Database', 63 | label: 'Database', 64 | image: databaseIcon, 65 | count: 1, 66 | }, 67 | { 68 | name: 'DatabaseCluster', 69 | label: 'Database', 70 | image: databaseIcon, 71 | count: 2, 72 | }, 73 | 74 | /** 75 | * Databases 76 | */ 77 | { 78 | name: 'Mysql', 79 | label: 'MySQL', 80 | image: mySqlIcon, 81 | count: 1, 82 | }, 83 | { 84 | name: 'MysqlCluster', 85 | label: 'MySQL', 86 | image: mySqlIcon, 87 | count: 2, 88 | }, 89 | 90 | { 91 | name: 'Oracle', 92 | label: 'Oracle', 93 | image: oracleIcon, 94 | count: 1, 95 | }, 96 | { 97 | name: 'OracleCluster', 98 | label: 'Oracle', 99 | image: oracleIcon, 100 | count: 2, 101 | }, 102 | 103 | { 104 | name: 'PostgreSql', 105 | label: 'PostgreSQL', 106 | image: postgresSqlIcon, 107 | count: 1, 108 | }, 109 | { 110 | name: 'PostgreSqlCluster', 111 | label: 'PostgreSQL', 112 | image: postgresSqlIcon, 113 | count: 2, 114 | }, 115 | 116 | { 117 | name: 'Elasticsearch', 118 | label: 'Elasticsearch', 119 | image: elasticSearchIcon, 120 | count: 1, 121 | }, 122 | { 123 | name: 'ElasticsearchCluster', 124 | label: 'Elasticsearch', 125 | image: elasticSearchIcon, 126 | count: 3, 127 | }, 128 | 129 | /** 130 | * AWS 131 | */ 132 | { 133 | name: 'Ec2', 134 | label: 'EC2', 135 | image: ec2Icon, 136 | count: 1, 137 | }, 138 | { 139 | name: 'Ec2Cluster', 140 | label: 'EC2', 141 | image: ec2Icon, 142 | count: 2, 143 | }, 144 | 145 | { 146 | name: 'Rds', 147 | label: 'RDS', 148 | image: rdsIcon, 149 | count: 1, 150 | }, 151 | { 152 | name: 'RdsCluster', 153 | label: 'RDS', 154 | image: rdsIcon, 155 | count: 2, 156 | }, 157 | 158 | { 159 | name: 'Elb', 160 | label: 'ELB', 161 | image: elbIcon, 162 | count: 1, 163 | }, 164 | 165 | { 166 | name: 'S3', 167 | label: 'S3', 168 | image: s3Icon, 169 | count: 1, 170 | }, 171 | 172 | { 173 | name: 'DynamoDb', 174 | label: 'DynamoDB', 175 | image: dynamoDbIcon, 176 | count: 1, 177 | }, 178 | 179 | { 180 | name: 'Redshift', 181 | label: 'Redshift', 182 | image: redShiftIcon, 183 | count: 1, 184 | }, 185 | 186 | { 187 | name: 'Cloudwatch', 188 | label: 'Cloudwatch', 189 | image: cloudWatchIcon, 190 | count: 1, 191 | }, 192 | 193 | { 194 | name: 'Elasticache', 195 | label: 'Elasticache', 196 | image: elasticCacheIcon, 197 | count: 1, 198 | }, 199 | 200 | { 201 | name: 'Iam', 202 | label: 'IAM', 203 | image: iamIcon, 204 | count: 1, 205 | }, 206 | 207 | { 208 | name: 'SimpleDb', 209 | label: 'SimpleDB', 210 | image: simpleDbIcon, 211 | count: 1, 212 | }, 213 | 214 | { 215 | name: 'Swf', 216 | label: 'SWF', 217 | image: swfIcon, 218 | count: 1, 219 | }, 220 | 221 | { 222 | name: 'Cloudfront', 223 | label: 'Cloudfront', 224 | image: cloudFrontIcon, 225 | count: 1, 226 | }, 227 | 228 | { 229 | name: 'Sqs', 230 | label: 'SQS', 231 | image: sqsIcon, 232 | count: 1, 233 | }, 234 | 235 | { 236 | name: 'Sns', 237 | label: 'SNS', 238 | image: snsIcon, 239 | count: 1, 240 | }, 241 | 242 | { 243 | name: 'Route53', 244 | label: 'Route53', 245 | image: route53Icon, 246 | count: 1, 247 | }, 248 | 249 | { 250 | name: 'StorageGateway', 251 | label: 'Storage Gateway', 252 | image: storageGatewayIcon, 253 | count: 1, 254 | }, 255 | 256 | { 257 | name: 'CloudFormation', 258 | label: 'CloudFormation', 259 | image: cloudFormationIcon, 260 | count: 1, 261 | }, 262 | 263 | { 264 | name: 'CloudSearch', 265 | label: 'CloudSearch', 266 | image: cloudSearchIcon, 267 | count: 1, 268 | }, 269 | 270 | { 271 | name: 'Glacier', 272 | label: 'Glacier', 273 | image: glacierIcon, 274 | count: 1, 275 | }, 276 | 277 | { 278 | name: 'ElasticBeanstalk', 279 | label: 'Elastic Beanstalk', 280 | image: elasticBeanstalkIcon, 281 | count: 1, 282 | }, 283 | 284 | { 285 | name: 'Ebs', 286 | label: 'EBS', 287 | image: ebsIcon, 288 | count: 1, 289 | }, 290 | 291 | { 292 | name: 'Lambda', 293 | label: 'Lambda', 294 | image: lambdaIcon, 295 | count: 1, 296 | }, 297 | 298 | { 299 | name: 'ApiGateway', 300 | label: 'API Gateway', 301 | image: apiGatewayIcon, 302 | count: 1, 303 | }, 304 | ]; 305 | 306 | export default nodeTypes; 307 | -------------------------------------------------------------------------------- /lib/lib.js: -------------------------------------------------------------------------------- 1 | import nodeTypes from './nodeTypes'; 2 | import vis from 'vis-network'; 3 | 4 | class NodeInterface { 5 | exchangesDataWith(node) { 6 | throw new Exception('Not implemented'); 7 | } 8 | getsDataFrom(node) { 9 | throw new Exception('Not implemented'); 10 | } 11 | sendsDataTo(node) { 12 | throw new Exception('Not implemented'); 13 | } 14 | getIds(node) { 15 | throw new Exception('Not implemented'); 16 | } 17 | getNodes(node) { 18 | throw new Exception('Not implemented'); 19 | } 20 | getEdges(node) { 21 | throw new Exception('Not implemented'); 22 | } 23 | } 24 | 25 | class Node extends NodeInterface { 26 | /** 27 | * @param {string} label 28 | * @param {string} image 29 | * @param {number|undefined} position 30 | * @param {number} [count = 1] 31 | */ 32 | constructor(label, image, position, count = 1) { 33 | super(); 34 | 35 | this._ids = Array.from(Array(count), () => Math.random().toString(36)); 36 | 37 | this._label = label; 38 | this._image = image; 39 | this._position = position; 40 | this._edges = []; 41 | 42 | window.diagram.add(this); 43 | } 44 | 45 | /** 46 | * @param {NodeInterface} node 47 | */ 48 | exchangesDataWith(node) { 49 | for (const otherId of node.getIds()) { 50 | for (const id of this._ids) { 51 | this._edges.push({ 52 | from: id, 53 | to: otherId, 54 | arrows: { 55 | to: { 56 | enabled: true, 57 | }, 58 | from: { 59 | enabled: true, 60 | }, 61 | }, 62 | }); 63 | } 64 | } 65 | } 66 | 67 | /** 68 | * @param {NodeInterface} node 69 | */ 70 | getsDataFrom(node) { 71 | for (const otherId of node.getIds()) { 72 | for (const id of this._ids) { 73 | this._edges.push({ 74 | from: otherId, 75 | to: id, 76 | arrows: { 77 | to: { 78 | enabled: true, 79 | }, 80 | }, 81 | }); 82 | } 83 | } 84 | } 85 | 86 | /** 87 | * @param {NodeInterface} node 88 | */ 89 | sendsDataTo(node) { 90 | for (const otherId of node.getIds()) { 91 | for (const id of this._ids) { 92 | this._edges.push({ 93 | from: id, 94 | to: otherId, 95 | arrows: { 96 | to: { 97 | enabled: true, 98 | }, 99 | }, 100 | }); 101 | } 102 | } 103 | } 104 | 105 | /** 106 | * @returns string[] 107 | */ 108 | getIds() { 109 | return this._ids; 110 | } 111 | 112 | /** 113 | * @returns Object[] 114 | */ 115 | getNodes() { 116 | return this._ids.map(id => { 117 | const node = { 118 | id, 119 | label: this._label, 120 | image: this._image, 121 | shape: this._image ? 'image' : 'ellipse', 122 | }; 123 | 124 | if (this._position !== undefined) { 125 | node.level = this._position; 126 | } 127 | 128 | return node; 129 | }); 130 | } 131 | 132 | /** 133 | * @returns Object[] 134 | */ 135 | getEdges() { 136 | return this._edges; 137 | } 138 | } 139 | 140 | for (const node of nodeTypes) { 141 | window[node.name] = class extends Node { 142 | constructor(label, position) { 143 | super(label || node.label, node.image, position, node.count); 144 | 145 | return new Proxy(this, { 146 | get: function(object, property) { 147 | if (Reflect.has(object, property)) { 148 | return Reflect.get(object, property); 149 | } else if (property.toLowerCase().slice(-4) === 'from') { 150 | return object.getsDataFrom; 151 | } else if (property.toLowerCase().slice(-2) === 'to') { 152 | return object.sendsDataTo; 153 | } else if (property.toLowerCase().slice(-4) === 'with') { 154 | return object.exchangesDataWith; 155 | } else { 156 | throw new Error(`Method ${property} does not exist.`); 157 | } 158 | }, 159 | }); 160 | } 161 | }; 162 | } 163 | 164 | window.Custom = Node; 165 | 166 | class Diagram { 167 | constructor() { 168 | this._options = {}; 169 | this._items = []; 170 | } 171 | 172 | /** 173 | * @param {NodeInterface[]} items 174 | */ 175 | add(...items) { 176 | this._items.push(...items); 177 | } 178 | 179 | /** 180 | * @param {Object} config 181 | */ 182 | render(config = {}) { 183 | const { leftToRight, container } = config; 184 | 185 | const nodes = []; 186 | const edges = []; 187 | 188 | for (const item of this._items) { 189 | nodes.push(...item.getNodes()); 190 | edges.push(...item.getEdges()); 191 | } 192 | 193 | if (!document.body) { 194 | document.body = document.createElement('body'); 195 | document.body.style.width = '100vw'; 196 | document.body.style.height = '100vh'; 197 | } 198 | 199 | this._network = new vis.Network( 200 | container || document.body, 201 | { 202 | nodes: new vis.DataSet(nodes), 203 | edges: new vis.DataSet(edges), 204 | }, 205 | { 206 | layout: { 207 | hierarchical: { 208 | direction: leftToRight || false ? 'LR' : 'RL', 209 | edgeMinimization: true, 210 | enabled: true, 211 | levelSeparation: 200, 212 | nodeSpacing: 200, 213 | sortMethod: 'directed', 214 | }, 215 | }, 216 | physics: { 217 | enabled: false, 218 | }, 219 | ...this._options, 220 | } 221 | ); 222 | 223 | this._network.on('beforeDrawing', function(context) { 224 | context.save(); 225 | 226 | context.setTransform(1, 0, 0, 1, 0, 0); 227 | context.fillStyle = '#fff'; 228 | context.fillRect(0, 0, context.canvas.width, context.canvas.height); 229 | 230 | context.restore(); 231 | }); 232 | } 233 | } 234 | 235 | window.diagram = new Diagram(); 236 | -------------------------------------------------------------------------------- /lib/icons/genericIcons.js: -------------------------------------------------------------------------------- 1 | export const clientIcon = 2 | ''; 3 | 4 | export const serverIcon = 5 | ''; 6 | 7 | export const databaseIcon = 8 | ''; 9 | -------------------------------------------------------------------------------- /lib/icons/databaseIcons.js: -------------------------------------------------------------------------------- 1 | const databaseIcon = 2 | ''; 3 | 4 | const elasticSearchIcon = 5 | ''; 6 | 7 | const mySqlIcon = 8 | ''; 9 | 10 | const oracleIcon = 11 | ''; 12 | 13 | const postgresSqlIcon = 14 | ''; 15 | 16 | export { 17 | databaseIcon, 18 | elasticSearchIcon, 19 | mySqlIcon, 20 | oracleIcon, 21 | postgresSqlIcon, 22 | }; 23 | --------------------------------------------------------------------------------