An Overview of Open Source Data Visualization Tools
Updated by Linode Contributed by Mihalis Tsoukalos
Creating graphic visualizations for a data set is a powerful way to derive meaning from vast amounts of information. It provides a way to extract meaningful relationships between different aspects of your data depending on how the data is mapped and which graphic representations are chosen. Data visualization is a common practice in many sectors, including various scientific disciplines, business settings, the government sector, and education.
There are many open source tools available to create sophisticated data visualizations for complex data sets. This guide will provide an introductory exploration of data analysis and 2D graphics rendering packages that can be used with R, Python, and JavaScript to generate data visualizations.
In this guide you will complete the following steps:
- Create three different data sets to use for your visualizations.
- Visualize your data using R and RStudio as a pie chart.
- Create a word cloud using Python and the pandas library.
- Develop a web browser based pie chart visualization using JavaScript and the D3.js package.
Before You BeginPermalink
Ensure you have the following programs and packages installed on your computer:
- The R programming language
- The RStudio Desktop application
- Python 3.7.0 or higher
- pandas Python package
- Matplotlib Python package
- wordcloud Python package
- The Golang programming language
AssumptionsThis guide assumes you have some basic familiarity with the following concepts and skills:
- Basic programming principles and data structures
- Are able to read code written in Go, Python, HTML, CSS, and JavaScript
Create Your Data SetsPermalink
In this section, you will create a data set using the contents of your Bash history file and optionally, your Zsh history file. You will then create a third data set using a Perl script that will extract information from the first two data sets. In the Create Visualizations for your Data section of the guide, you will use these various data sets to create corresponding visualizations.
Data Set 1 - Bash History FilePermalink
A Bash history file stores all commands executed in your command line interpreter. View your 10 most recently executed commands with the following command:
head ~/.bash_history
Your output should resemble the following:
git commit -a -m "Fixed Constants links"
git push
git diff
docker images
brew update; brew upgrade; brew cleanup
git commit -a -m "Cleanup v2 and v3"
git push
git commit -a -m "Added examples section"
git push
cat ~/.lenses/lenses-cli.yml
Create a new directory named data-sets
to store your data and copy your Bash history file to the directory:
mkdir data-sets && cp ~/.bash_history data-sets/data-1
Data Set 2 - Zsh History FilePermalink
If you are using the Zsh shell interpreter, you can use its history file as a second data set. Zsh’s history file format includes data that you will need to exclude from your data set. Use AWK to clean up your Zsh history file and save the output to a new file in the data-sets
directory:
awk -F ";" '{$1=""; print $0}' ~/.zsh_history | sed -e "s/^[ \t]*//" -e "/^$/d" > data-sets/data-2
Data Set 3 - Perl ScriptPermalink
To create your third data set, you will use a Perl script that categorizes the contents of your data set files. The categorization is based on the word count for each line of text in your data files or, in other words, the length of each command stored in your Bash and Zsh history files. The script creates 5 categories of command lengths; 1 - 2 words, 3 - 5 words, 6 - 10 words, 11 - 15 words, and 16 or more words.
Create a file named
command_length.pl
in your home directory with the following content:- ~/command_length.pl
-
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
#!/usr/bin/perl -w use strict; my $directory = ""; my $filename = ""; my $CAT1 = 0; my $CAT2 = 0; my $CAT3 = 0; my $CAT4 = 0; my $CAT5 = 0; if ( @ARGV != 1 ) { die <<Thanatos usage info: Please use exactly 1 argument! Thanatos } ($directory) = @ARGV; opendir(BIN, $directory) || die "Error opening directory $directory: $!\n"; while (defined ($filename = readdir BIN) ) { # The following command does not process . and .. next if( $filename =~ /^\.\.?$/ ); process_file($directory."/".$filename); } print "Category1\t\t$CAT1\n"; print "Category2\t\t$CAT2\n"; print "Category3\t\t$CAT3\n"; print "Category4\t\t$CAT4\n"; print "Category5\t\t$CAT5\n"; exit 0; sub process_file { my $file = shift; my $line = ""; open (HISTORYFILE, "< $file") || die "Cannot open $file: $!\n"; while (defined($line = <HISTORYFILE>)) { chomp $line; next if ( ! defined($line) ); check_category($line); } } sub check_category { my $command = shift; chomp $command; my $length = length($command); if ( $length <= 2 ) { $CAT1 ++; } elsif ( $length <= 5 ) { $CAT2 ++; } elsif ( $length <= 10 ) { $CAT3 ++; } elsif ( $length <= 15 ) { $CAT4 ++; } else { $CAT5 ++; } }
Run the Perl script. The script expects a single argument; the directory that holds all data files that you want to process. The file’s output will be saved in a new file
command_categories.txt
. Thecommand_categories.txt
file will be used later in this guide to create visualizations using R../command_length.pl data-sets > ~/command_categories.txt
Note
Open the
.command_categories.txt
file to view the categorizations created by your Perl script. Your file should resemble the following example:- command_categories.txt
You now have three sources of data that you can use to explore data visualization tools in the next sections.
~/data-sets/data-1
~/data-stes/data-2
~/command_categories.txt
Create Visualizations for your DataPermalink
In this section you will use the data sets you created in the previous section to generate visualizations for them.
Visualize your Data with R and RStudioPermalink
R is a specialized programming language used for statistical computing and graphics. It is especially good for creating high quality graphs, like density plots, line charts, pie charts, and scatter plots. RStudio is an integrated development environment (IDE) for R that includes debugging and plotting tools that make it easy to write, debug, and run R scripts.
In this section, you will use the command_categories.txt
file created in the Data Set 3 - Perl Script section and RStudio to create a pie chart and simple spreadsheet of your data.
Open RStudio Desktop and create a data frame. In R, a data frame is a table similar to a two-dimensional array.
DATA <- read.table("~/command_categories.txt", header=TRUE)
This command will read the
command_categories.txt
file that was created in the Data Set 3 - Perl Script section of the guide and create a data frame from it that is stored in theDATA
variable.The
header=TRUE
argument indicates that the file’s first row contains variable names for column values. This means thatCategory Name
andNumber of Times
will be used as variable names for the two columns of values incommand_categories.txt
.
Next, create a pie chart visualization for each column of values using R’s
pie()
function.pie(DATA$Number.of.Times, DATA$Category.Name)
The function’s first argument,
DATA$Number.of.Times
andDATA$Category.Name
, provides the x-vector numeric values to use when creating the pie chart visualization.The second argument,
DATA$Category.Name
, provides the labels for each pie slice.
RStudio will display a pie chart visualization of your data in the Plots and Files window similar to the following:
Visualize your data in a spreadsheet style format using R’s
View()
function.View(DATA, "Command Lengths")
RStudio will display a spreadsheet style viewer in a new window pane tab.
Explore R’s graphics package to discover additional functions you can use to create more complex visualizations for your data with RStudio.
Create a Word Cloud using PythonPermalink
Word clouds depict text data using varying font sizes and colors to visually demonstrate the frequency and relative importance of each word. A common application for word clouds is to visualize tag or keyword relevancy. In this section you will use Python3 and your Bash and Zsh history files to generate a word cloud of all your shell commands. The Python packages listed below are commonly used in programs for data analysis and scientific computing and you will use them to generate your word cloud.
- pandas: this package provides data structures that make it easier to work with various types of data, including tabular data, ordered and unordered time series data, and arbitrary matrix data.
- Matplotlib: a plotting library that generates 2D graphics.
- wordcloud: allows you to generate word clouds.
Create a file named
create_wordcloud.py
in your home directory with the following content:- ~/create_wordcloud.py
-
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
#!/usr/bin/env python3 import pandas as pd import matplotlib.pyplot as plt import sys import os from wordcloud import WordCloud, STOPWORDS from random import randrange path = sys.argv[1] if os.path.exists(path): print("Creating word cloud for file: " + path) data = pd.read_table(path, header = None, names=["CMD"], encoding = "iso-8859-1") skipWords = [] skipWords.extend(STOPWORDS) words = ''.join(data['CMD']) w = WordCloud( stopwords=set(skipWords), background_color='gray', width=4000, height=2000 ).generate(words) filename = "word_cloud_" + str(randrange(100)) + ".png" print("Your wordcloud's filename is: " + filename) plt.imshow(w) plt.axis("off") plt.savefig(filename, dpi=1000) else: print("File" + path + "does not exist")
Run your Python script and pass the path of one of your data set files as an argument. The script will read the contents of the file using panda’s
read_table()
function and convert it into a data frame with a column name ofCMD
. It will then use the data in theCMD
column to create a concatenated string representation of the data that can be passed to wordcloud to generate a.png
wordcloud image../create_wordcloud.py ~/data-sets/data-1
You should see a similar output from the Python script:
Creating word cloud for file: /Users/username/data-sets/data-2 Your word cloud's filename is: word_cloud_58.png
Open the
word_cloud_58.png
image file to view your word cloud.
You could use a similar process to create a word cloud visualization for any text you want to analyze.
Visualize Data using D3.jsPermalink
D3.js is a JavaScript library that helps you visualize JSON formatted data using HTML, SVG, and CSS. In this section you will us D3.js to create and embed a pie chart visualization into a web page.
To convert your data set into JSON, you will create a Golang command line utility that generates JSON formatted plain text output. For more complex data sets, you might consider creating a similar command line utility using Golang’s json package.
Create a file named
cToJSON.go
in your home directory with the following content:- ./cToJSON.go
-
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
package main import ( "bufio" "flag" "fmt" "io" "os" "regexp" "sort" ) var DATA = make(map[string]int) func lineByLine(file string) error { var err error f, err := os.Open(file) if err != nil { return err } defer f.Close() r := bufio.NewReader(f) for { line, err := r.ReadString('\n') if err == io.EOF { break } else if err != nil { fmt.Printf("error reading file %s", err) break } r := regexp.MustCompile("[^\\s]+") words := r.FindAllString(line, -1) if len(words) == 0 { continue } if _, ok := DATA[words[0]]; ok { DATA[words[0]]++ } else { DATA[words[0]] = 1 } } return nil } func main() { flag.Parse() if len(flag.Args()) == 0 { fmt.Printf("usage: cToJSON <file1> [<file2> ...]\n") return } for _, file := range flag.Args() { err := lineByLine(file) if err != nil { fmt.Println(err) } } n := map[int][]string{} var a []int for k, v := range DATA { n[v] = append(n[v], k) } for k := range n { a = append(a, k) } fmt.Println("[") sort.Sort(sort.Reverse(sort.IntSlice(a))) counter := 0 for _, k := range a { if counter >= 10 { break } for _, s := range n[k] { if counter >= 10 { break } counter++ fmt.Printf("{\"command\":\"%s\",\"count\":%d},\n", s, k) } } fmt.Println("];") }
- The utility expects file paths to your Bash and Zsh data sets as arguments.
- It will then read the files and find the 10 most popular commands and output it as JSON formatted data.
- Several Golang standard library packages are used in the utility to perform operations liking reading files, using regular expressions, and sorting collections.
Run the command line utility and pass in the paths to each command history data set:
go run cToJSON.go data-set/data-1 data-set/data-2
Your output should resemble the following:
[ {"command":"ll","count":1832}, {"command":"git","count":1567}, {"command":"cd","count":982}, {"command":"brew","count":926}, {"command":"unison","count":916}, {"command":"gdf","count":478}, {"command":"ssh","count":474}, {"command":"rm","count":471}, {"command":"sync","count":440}, {"command":"ls","count":421}, ];
You are now ready to create your pie chart visualization and embed it into a web page using D3.js
Create an HTML file named
pieChart.html
and copy and paste the following content. TheDATA
variable on line 31 contains the JSON data that was created by thecToJSON.go
script in the previous step. Remove the JSON data in the example and replace it with your own JSON data.Note
In this example, your JSON data is hardcoded inpieChart.html
for simplicity. Web browser security constraints restrict how a document or script loaded from one origin can interact with a resource from another origin. However, you may consider using the d3-fetch module to fetch your JSON data from a specific URL.- ~/pieChart.html
-
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
<!DOCTYPE html> <html lang="en"> <head> <meta charset="utf-8"> <title>History Visualization</title> <style type="text/css"> * { margin: 0; padding: 0; } #chart { background-color: white; font: 14px sans-serif; margin: 0 auto 50px; width: 600px; height: 600px; } #chart .label{ fill: #404040; font-size: 12px; } </style> </head> <body> <div id="chart"></div> </body> <script src="https://d3js.org/d3.v3.min.js"></script> <script type="text/javascript"> var DATA = [ {"command":"ll","count":1832}, {"command":"git","count":1567}, {"command":"cd","count":982}, {"command":"brew","count":926}, {"command":"unison","count":916}, {"command":"gdf","count":478}, {"command":"ssh","count":474}, {"command":"rm","count":471}, {"command":"sync","count":440}, {"command":"ls","count":421} ]; var width = 600; height = 600; radius = width / 2.5; var pie = d3.layout.pie() .value(function(d) { return d.count; }) var pieData = pie(DATA); var color = d3.scale.category20(); var arc = d3.svg.arc() .innerRadius(0) .outerRadius(radius - 7); var svg = d3.select("#chart").append("svg") .attr("width", width) .attr("height", height) .append("g") .attr("transform", "translate(" + width / 2 + "," + height / 2 + ")"); var ticks = svg.selectAll("line") .data(pieData) .enter() .append("line"); ticks.attr("x1", 0) .attr("x2", 0) .attr("y1", -radius+4) .attr("y2", -radius-2) .attr("stroke", "black") .attr("transform", function(d) { return "rotate(" + (d.startAngle+d.endAngle)/2 * (180/Math.PI) + ")"; }); var labels = svg.selectAll("text") .data(pieData) .enter() .append("text"); labels.attr("class", "label") .attr("transform", function(d) { var dist = radius + 25; angle = (d.startAngle + d.endAngle) / 2; x = dist * Math.sin(angle); y = -dist * Math.cos(angle); return "translate(" + x + "," + y + ")"; }) .attr("dy", "0.35em") .attr("text-anchor", "middle") .text(function(d){ return d.data.command + " (" + d.data.count + ")"; }); var path = svg.selectAll("path") .data(pieData) .enter() .append("path") .attr("fill", function(d, i) { return color(i); }) .attr("d", arc); </script> </html>
Navigate to your preferred browser and enter the HTML file’s absolute path to view the pie chart. For a macOS user that has stored the HTML file in their home directory, the path would resemble the following:
/Users/username/pieChart.html
Next StepsPermalink
Now that you are familiar with some data visualization tools and simple techniques, you can begin to explore more sophisticated approaches using the same tools explored in this guide. Here are a few ideas you can consider:
- Create a new data set by extracting all
git
related commands from your history files; analyze and visualize them. - Automate some of the techniques discussed in this guide using Cron jobs to generate your data sets automatically.
- Explore the Python for Data Science eBook’s data visualization section for a deeper dive into using pandas.
Join our CommunityPermalink
Find answers, ask questions, and help others.
This guide is published under a CC BY-ND 4.0 license.