├── .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 | 
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 |
--------------------------------------------------------------------------------