Security News
cURL Project and Go Security Teams Reject CVSS as Broken
cURL and Go security teams are publicly rejecting CVSS as flawed for assessing vulnerabilities and are calling for more accurate, context-aware approaches.
A dashboard to visualize a synthesis on a structured corpus, using several charts (pie, histogram, ...)
A web dashboard to visualize a synthesis on a structured corpus, using several charts (pies, histograms, ...), powered by castor, and based on SB Admin v2.0.
ezVIS stands for easy visualization of information in web report.
You have to install mongodb first, and node too.
$ npm install --production ezvis -g
Note: the
--production
options is not required, it prevents the installation of development dependencies.
Make sure mongodb is running, and then.
$ ezvis /path/to/data/repository
Then, point your browser to http://localhost:3000/
If you don't have a data repository, but already loaded data in mongodb, you can use:
$ ezvis $PWD/data
Before that, you have to configure your mongo connection, by creating a
./data.json
file containing something like:
{
"port": 3000,
"collectionName" : "insu"
}
Then, point your browser to http://localhost:3000/
To make charts appear on the dashboard, you have to configure them.
The configuration is done in the JSON file of
castor (e.g.data.json
),
it's a file with the same name as the data directory
(besides that directory), appended with .json
.
The whole dashboard configuration is done inside the dashboard
key of the
JSON configuration file. Except, the documentFields
configuration, and
corpusFields
.
Each chart has to be described in the dashboard.charts
key.
Below is an example with an histogram, and a pie chart. There are two types of
charts: histogram
and pie
.
{
"theme": "/path/to/ezvis",
"documentFields": {
"$fields.Themes" : {
"path" : "content.json.DiscESI",
"parseCSV" : ";",
"foreach": {
"trim": true
}
},
},
"dashboard" : {
"charts": [
{
"field": "content.json.Py",
"type": "histogram"
},
{
"field": "fields.Themes",
"type": "pie"
}
]
}
}
documentFields are fields added to each document at loading/synchronizing files.
They are declared in the JSON settings, in the documentFields
object.
A simplistic example of a document is:
{
"wid": "2rgwJl",
"content": {
"json": {
"title": "2001: A Space Odyssey",
"year": "1968",
"director": "Stanley Kubrick",
"actors": "Keir Dullea/Gary Lockwood/William Sylvester/Daniel Richter/Leonard Rossiter/Douglas Rain"
}
}
}
If you want to easily access the year of document, you can declare a $year
documentField:
{
"documentFields": {
"$year": {
"get": "content.json.year"
}
}
}
which will modify the former document to the following:
{
"wid": "2rgwJl",
"content": {
"json": {
"title": "2001: A Space Odyssey",
"year": "1968",
"director": "Stanley Kubrick",
"actors": "Keir Dullea/Gary Lockwood/William Sylvester/Daniel Richter/Leonard Rossiter/Douglas Rain"
}
},
"year": "1968"
}
$year
indicates to create a year
property at the document's root, and the
get
JBJ action points to the
location of the source field in the same document.
All JBJ actions are
applicable, and for example a "cast": "number"
after the get
action will
transtype the year "1968"
into a number 1968
.
A more useful usage is to separate the content.json.actors
field into one actors
array:
{
"documentFields": {
"$year": {
"get": "content.json.year"
},
"$actors": {
"path": "content.json.actors",
"parseCSV" : "/",
"foreach": {
"trim": true
}
}
}
}
would produce
{
"wid": "2rgwJl",
"content": {
"json": {
"title": "2001: A Space Odyssey",
"year": "1968",
"director": "Stanley Kubrick",
"actors": "Keir Dullea/Gary Lockwood/William Sylvester/Daniel Richter/Leonard Rossiter/Douglas Rain"
}
},
"year": "1968",
"actors": ["Keir Dullea","Gary Lockwood","William Sylvester","Daniel Richter","Leonard Rossiter","Douglas Rain"]
}
Note: you can use a dot notation in the name of the field to be created. Using
"$my.fields.year"
will create ayear
field within thefields
field within themy
field at the root of the document.
Note 2: the generated fields are truncated at 1000 characters (if they are of string type), except if you add
"noindex": true
to the field (in this case, performance may be lower, but only if later operations use the field; that is to say that a field created only to be displayed, not to be used in computations -like charts- is a good candidate to be noindexed).
The $text
field is used in the documents table to search the documents, as a
full-text index.
Thus, for the document table to be searchable, you have to build a $text
field, using technique similar to this:
"$text": {
"get" : ["title", "year", "director", "actors"],
"join": "|"
}
This field is not truncated at 1000 characters.
The nosave
property of a variable prevent its value to be saved in the
document.
This is useful for external resources, like http:
or local:
protocols.
"$country2iso": {
"nosave": true,
"$?" : "http://localhost:35000/country2iso3.json",
"parseJSON": true
},
"$codes": {
"mappingVar": ["country","country2iso"]
},
corpusFields are computed after documents loading/synchronizing.
They are used to compute metrics on the whold corpus (hence, the name).
For example, to get the number of documents in the corpus:
"corpusFields": {
"$filmsNb": {
"visible": true,
"label" : "films",
"icon" : "hand-o-right",
"$?" : "local:///compute.json?operator=count&field=wid",
"parseJSON": true,
"get" : "data.0.value",
"cast" : "number"
},
The filmsNb
corpusFields above is visible
on the dashboard page, the
label
displayed after its value is "films", the icon
at its left is a
hand-o-right
from font-awesome.
From "$?"
on, the properties are JBJ actions.
The "$?"
action means that the remaining actions will be applied to the
result of the /compute
route of ezvis, using the
count
operator
on the wid
field
.
It's a source using the local
protocol, which is a shortcut to http://localhost:port
(useful because the
port number is not always known before the launch of the server). This one
could return a page like:
{
template: "compute.html",
url: {
protocol: "http:",
slashes: true,
auth: null,
host: "localhost:3000",
port: "3000",
hostname: "localhost",
hash: null,
search: "?operator=count&field=wid",
query: "operator=count&field=wid",
pathname: "/compute.json",
path: "/compute.json?operator=count&field=wid",
href: "http://localhost:3000/compute.json?operator=count&field=wid"
},
parameters: {
field: [
"wid"
],
operator: "count",
selector: null,
query: null,
itemsPerPage: 30,
startIndex: 0,
startPage: null,
search: null,
order: [
null
],
columns: [
null
],
flying: [
null
],
resource: "data5"
},
headers: {
Content-Type: "application/json"
},
recordsTotal: 1,
recordsFiltered: 1,
data: [{
_id: "wid",
value: 29
}]
}
This page is a text, containing JSON. You have to parse it, using
"parseJSON": true
, then get the value #0 of the data
array, using the
get
action and the dot notation: data.0.value
(it's the object-path
notation, see
the examples).
Tip: You can transform a
local:///compute.json?operator=count&field=wid
intohttp://localhost:3000/compute.json?operator=count&field=wid
and copy-paste its content into the input area of the JBJ Playground, and try to enter in the stylesheet area the JBJ actions you want to test, and click "Try it" to see if the result matches what you want in the corpusField.
Once in a while, you need to combine a corpusField
and a documentField
(to normalize a value, or to use reference table in a corpusField
).
flyingFields
are like documentFields
, except that they are computed just in time, thus they can interoperate with corpusFields
.
Warning: if you have a
corpusFields
with the same name as adocumentFields
, one of them will be replaced by the other. So don't name them the same way.
A flyingField
can be seen as a post-treatment (written in JBJ) applicable
through the use of a compute operator on a specific field. The
URL begins with /compute.json?operator=
followed by the operator name (often
distinct
), followed by &field=
and by a field name. Then you can add
&flying=
and the name of the flyingField
.
Example where you replace the name of a country by the ISO code of the country, using an external table.
If you have a documentFields
named country
:
{
"documentFields": {
"country": {
"get": "content.json.countries",
"parseCSV": ";",
"foreach": {
"trim": true
}
}
}
}
And a corpusFields
containing a matching table from a external URL:
{
"corpusFields": {
"$country2iso": {
"$?" : "http://external.domain.org/country2iso3.json",
"parseJSON": true
}
}
}
where the country2iso3.json
file contains some keys like (that's what the
URL in $?
returns):
{
"Albania": "ALB",
"Algeria": "DZA",
"Zaire": "COD",
"Zambia": "ZMB"
}
The distinct
operator on country
documentFields
will be called with http://localhost:3000/compute.json?operator=distinct&field=country:
{
"data": [
{
"_id": "Albania",
"value": 2
},
{
"_id": "Algeria",
"value": 15
}
]
}
With a country2isoTreatment
flyingFields
:
{
"flyingFields": {
"$country2isoTreatment": {
"$_id": {
"combine" : ["_id", "country2iso"]
},
"mask": "_id,value"
}
}
}
which combine
(or mappingVar
) the _id
given by the operator and the
country2iso
corpusFields
, and then mask all corpusFields
and other
fields, to only keep _id
and value
, the previous URL can be added with
&flying=country2isoTreatment
to give:
{
"data": [
{
"_id": "ALB",
"value": 2
},
{
"_id": "DZA",
"value": 15
}
]
}
All dashboard settings are inside a dashboard
key.
At the moment, there is only the charts
setting.
Used to represent evolution of the number of documents along the time (so, this field is often a publication year, or anything indicating a point in time).
Possible configuration: size
, and color
.
If you want to display the holes in time too (e.g., year with no documents),
add "parseDates": true
in the configuration:
{
"field": "content.json.year",
"type": "histogram",
"title": "Per year",
"parseDates": true
}
If you want to shorten the field value to display on the chart, use an associative array to replace too long fields values with shorter ones:
{
"field": "fields.Themes",
"type": "histogram",
"title": "Themes (histogram)",
"maxItems" : 10,
"labels": {
"Biology & Biochemistry" : "Bio & Bio",
"Pharmacology & Toxicology" : "Pharmaco & Toxico",
"Plant & Animal Science" : "Plant & Animal"
}
},
Used to fill the pie chart quarters.
There are some configuration possible: size
of the pie,
colors
of the slices, and position of the legend
.
You can also use groupPercent
: if there is more than one slice whose
percentage of the pie is less than this number, those slices will be grouped
together into one slice. This is the "other" slice. It will always be the last
slice in a pie.
To name the possible grouped slice, use groupedTitle
.
To remove the labels around the slices, set removeLabels
to true
.
If you want to shorten the field value to display in the legend, use an associative array to replace too long fields values with shorter ones:
{
"field": "Thematique",
"type": "pie",
"title": "Themes (pie)",
"labels": {
"Clinical Medicine": "Clinic",
"Biology & Biochemistry": "Bio",
"Neurosciences & Behavior": "Neuro"
}
},
Used to display the number of documents associated to a field value (for example, for keywords: how many documents match a keyword?). Bars are sorted by descending number of documents.
Possible configuration: size
, color
, and maxItems
.
maxItems
limit the number of bars to its value (default value: 100
).
If you want to shorten the field value to display on the chart, use an associative array to replace too long fields values with shorter ones:
{
"field": "fields.Themes",
"type": "horizontalbars",
"title": "Themes (bars)",
"maxItems" : 10,
"labels": {
"Biology & Biochemistry" : "Bio & Bio",
"Pharmacology & Toxicology" : "Pharmaco & Toxico",
"Plant & Animal Science" : "Plant & Animal"
}
},
Used to project country-related numbers on a geographical map.
At the moment, there is only one usable map: "world".
To be able to project the numbers to the areas on the map, you have to match
numbers and ISO-36166-1 ALPHA-3 codes. To do this, you may need the mapping
key in the declaration of a new documentFields
instance.
Here an example:
"dashboard" : {
"charts": [
{
"type": "map",
"fields": [ "fields.country" ],
"title": "Countries map"
}
]
},
"documentFields" : {
"country" : {
"label": "Country",
"path" : "content.json.country",
"parseCSV": ";",
"foreach" : {
"mapping": {
"Afghanistan" : "AFG",
"Angola": "AGO",
"Albania" : "ALB",
"Andorra" : "AND",
"United Arab Emirates" : "ARE",
"Argentina" : "ARG",
"Armenia" : "ARM",
"Fr. S. and Antarctic Lands" : "ATF",
"Australia" : "AUS",
"Austria" : "AUT",
"Azerbaijan" : "AZE",
"Burundi" : "BDI",
"Belgium" : "BEL",
"Benin": "BEN",
"Burkina Faso" : "BFA",
"Bangladesh": "BGD",
"Bulgaria" : "BGR",
"Bosnia and Herz." : "BIH",
"Belarus" : "BLR",
"Belize" : "BLZ",
"Bolivia": "BOL",
"Brazil" : "BRA",
"Brunei" : "BRN",
"Bhutan" : "BTN",
"Botswana" : "BWA",
"Central African Republic": "CAF",
"Canada" : "CAN",
"Switzerland" : "CHE",
"Chile" : "CHL",
"People's Republic of China" : "CHN",
"Ivory Coast" : "CIV",
"Cameroon": "CMR",
"Zaire" : "COD",
"Congo" : "COG",
"Colombia" : "COL",
"Costa Rica" : "CRI",
"Cuba" : "CUB",
"N. Cyprus" : "CYN",
"Cyprus" : "CYP",
"Czech Republic": "CZE",
"Germany" : "DEU",
"Djibouti" : "DJI",
"Denmark" : "DNK",
"Dominican Rep." : "DOM",
"Algeria" : "DZA",
"Ecuador": "ECU",
"Egypt" : "EGY",
"Eritrea" : "ERI",
"Spain" : "ESP",
"Estonia" : "EST",
"Ethiopia" : "ETH",
"Finland" : "FIN",
"Falkland Is." : "FLK",
"France" : "FRA",
"Gabon": "GAB",
"Gaza" : "GAZ",
"United Kingdom" : "GBR",
"Georgia" : "GEO",
"Ghana": "GHA",
"Guinea" : "GIN",
"Gambia" : "GMB",
"Guinea Bissau" : "GNB",
"Eq. Guinea" : "GNQ",
"Greece": "GRC",
"Greenland" : "GRL",
"Guatemala": "GTM",
"Guyana" : "GUY",
"Hong Kong" : "HKG",
"Honduras" : "HND",
"Croatia" : "HRV",
"Haiti" : "HTI",
"Hungary": "HUN",
"Iceland": "ISL",
"India": "IND",
"Indonesia": "IDN",
"Ireland": "IRL",
"Iran": "IRN",
"Iraq" : "IRQ",
"Israel": "ISR",
"Italy" : "ITA",
"Jamaica" : "JAM",
"Jordan": "JOR",
"Japan" : "JPN",
"Kazakhstan" : "KAZ",
"Kenya" : "KEN",
"Kyrgyzstan" : "KGZ",
"Cambodia" : "KHM",
"South Korea" : "KOR",
"Kosovo" : "KOS",
"Kuwait" : "KWT",
"Laos" : "LAO",
"Lebanon" : "LBN",
"Liberia" : "LBR",
"Libya" : "LBY",
"Liechtenstein" : "LIE",
"Sri Lanka" : "LKA",
"Lesotho" : "LSO",
"Lithuania" : "LTU",
"Luxembourg" : "LUX",
"Latvia" : "LVA",
"Macau" : "MAC",
"St. Martin" : "MAF",
"Morocco" : "MAR",
"Monaco" : "MCO",
"Moldova" : "MDA",
"Madagascar" : "MDG",
"Mexico" : "MEX",
"Macedonia, the Former Yugoslave Republic of": "MKD",
"Mali" : "MLI",
"Myanmar" : "MMR",
"Montenegro" : "MNE",
"Mongolia" : "MNG",
"Mozambique" : "MOZ",
"Mauritania" : "MRT",
"Malawi" : "MWI",
"Malaysia" : "MYS",
"Namibia" : "NAM",
"New Caledonia" : "NCL",
"Niger" : "NER",
"Nigeria" : "NGA",
"Nicaragua" : "NIC",
"Netherlands" : "NLD",
"Norway" : "NOR",
"Nepal" : "NPL",
"New Zealand" : "NZL",
"Oman" : "OMN",
"Pakistan" : "PAK",
"Panama" : "PAN",
"Peru": "PER",
"Philippines" : "PHL",
"Papua New Guinea" : "PNG",
"Poland": "POL",
"N. Korea" : "PRK",
"Portugal" : "PRT",
"Paraguay" : "PRY",
"Qatar" : "QAT",
"Romania" : "ROU",
"Russia" : "RUS",
"Rwanda" : "RWA",
"W. Sahara" : "SAH",
"Saudi Arabia" : "SAU",
"Sudan" : "SDN",
"S. Sudan" : "SDS",
"Senegal" : "SEN",
"Sierra Leone" : "SLE",
"El Salvador" : "SLV",
"Somaliland" : "SOL",
"Somalia" : "SOM",
"Serbia" : "SRB",
"Suriname" : "SUR",
"Slovakia" : "SVK",
"Slovenia" : "SVN",
"Sweden" : "SWE",
"Swaziland" : "SWZ",
"Sint Maarten" : "SXM",
"Syria" : "SYR",
"Chad" : "TCD",
"Togo" : "TGO",
"Thailand" : "THA",
"Tibet" : "TIB",
"Tajikistan" : "TJK",
"Turkmenistan" : "TKM",
"East Timor" : "TLS",
"Trinidad and Tobago" : "TTO",
"Tunisia" : "TUN",
"Turkey" : "TUR",
"Taiwan" : "TWN",
"Tanzania" : "TZA",
"Uganda" : "UGA",
"Ukraine" : "UKR",
"Uruguay" : "URY",
"United States" : "USA",
"Uzbekistan" : "UZB",
"Venezuela" : "VEN",
"Vietnam" : "VNM",
"West Bank" : "WEB",
"Yemen" : "YEM",
"South Africa" : "ZAF",
"Zambia" : "ZMB",
"Zimbabwe" : "ZWE"
}
}
}
}
You can use the colors
option to set the color scale.
ex:
Classical Network, with all links (at least until 100k).
{
"type": "network",
"fields": [ "actors" ],
"maxItems": 100000,
"title": "Actors net",
"help": "Links between actors."
},
Centered Network, where only the actors "near" Arnold Schwarzenegger
and
Peter Weller
are visible.
{
"type": "network",
"fields": [ "actors" ],
"maxItems": 100000,
"title": "Arnold & Peter",
"help": "Actors linked to Arnold Schwarzenegger and Peter Weller",
"centerOn": ["Arnold Schwarzenegger", "Peter Weller"]
}
To change the color of a node, in order to make it more visible, add a nodes
array, in which each element is an object with a value
and a color
property. If you don't provide a color, their color will be the current one,
saturated.
{
"type": "network",
"fields": [ "actors" ],
"maxItems": 100000,
"threshold": 3,
"title": "Actors net with colors",
"help": "Network of actors, which common films are at least 3, and where Carrie Fisher is highlighed",
"nodes": [{
"value": "Carrie Fisher",
"color": "#33a02c"
}]
}
You can also precise in which field the value has to be colored.
{
"type": "network",
"fields": ["actors", "director"],
"maxItems": 100000,
"title": "Sylvester Stallone and himself",
"help": "A network centered on Sylvester Stallone, as director",
"centerOn": ["Sylvester Stallone"],
"nodes": [{
"field": "director",
"value": "Sylvester Stallone",
"color": "red"
}]
}
Selected Network, where only documents matching the mongodb selector
are
visible (using a MongoDB Match Query Criteria, but within a JSON, thus using
quotes around operators).
{
"type": "network",
"fields": [ "actors" ],
"maxItems": 100000,
"title": "Actors net 2000",
"help": "Actors from films after year 2000",
"selector": {
"year": { "$gte": "2000" }
}
}
Threshold Network, where only links above the threshold value are displayed (in the example: the actors linked are those who played at least in 3 films together).
{
"type": "network",
"fields": [ "actors" ],
"maxItems": 100000,
"threshold": 3,
"title": "Actors net > 3 films",
"help": "Network of actors, which common films are at least 3"
}
To display several fields in the same network, put their names in the fields
property:
{
"type": "network",
"fields": [ "director", "year", "title" ],
"maxItems": 100000,
"title": "Director, year, films",
"help": "Network of directors, year and films"
}
You can set the color of the nodes field by field.
Example: actors
field in green and director
field in blue.
{
"type": "network",
"fields": ["actors", "director"],
"maxItems": 100000,
"title": "Actors and director colored",
"help": "A network with actors in green and directors in blue",
"fieldsColor": {
"actors": "green",
"director": "blue"
}
}
To specify the size of the pie, add the size
key to your chart.
The height
you specify is in pixels, and is used both on dashboard, and on chart pages.
Ex:
{
"field": "fields.Themes",
"type": "pie",
"size": {
"height": 400
}
}
You can add a columns
property too, knowing that the display has a "width"
of 12 columns (Twitter bootstrap).
Here is an example where the pie should take half of the page's width:
{
"field": "fields.Themes",
"type": "pie",
"size": {
"height": 400,
"columns": 6
}
}
If you need to separate two charts, you can add an offset before a chart, using
offset
property. It is a number which represent the "width" of offset
columns.
Below is an example where the horizontal bars should take 5 columns, with a preceding offset of 1 column.
{
"field": "fields.Themes",
"type": "horizontalbars",
"title": "Thèmes",
"size": {
"height": 420,
"columns": 5,
"offset": 1
}
}
Note:
columns
andoffset
properties are taken into account only on dashboard (index) page.
To specify where you want the legend to be, add the legend
key to your chart, with an object as a value.
In general, you can find what's possible in legend
in
amCharts documentation.
The position
can take 4 values:
Ex:
{
"field": "fields.Themes",
"type": "pie",
"legend": {
"position": "left"
}
}
If you don't want a legend, remove legend
key from the chart.
If you need one, simply add "legend": {}
in the chart.
{
"field": "fields.Themes",
"type": "pie",
"legend": {}
}
Or use the enabled
property:
{
"field": "fields.Themes",
"type": "pie",
"legend": {
"enabled": true
}
}
You can set a color
value (hexadecimal color value).
Example with a red histogram:
{
"dashboard": {
"charts" : [
{
"field": "content.json.Year",
"type": "histogram",
"color": "#ff0000"
}
]
}
}
In a pie, you can parameter a set of colors to be used:
{
"dashboard": {
"charts" : [
{
"field": "content.json.Keywords",
"type": "pie",
"colors": [ "#BB9FF5", "#ff7a85", "#44b2ba", "#ffa65a", "#34cdb8"]
}
]
}
}
In a map, you can also parameter a set of colors, but you can use ColorBrewer to choose only a color scale name (default value: "YlOrRd" -Yellow, Orange, Red-, but you can try "RdYlBu" -Red, Yellow Blue-, or "BuGn", -Blue, Green-):
{
"dashboard": {
"charts" : [
{
"field": "content.json.country",
"type": "map",
"colors": "OrRd"
}
]
}
}
Or you can use the following syntax (which allows you to add other color options):
{
"dashboard": {
"charts" : [
{
"field": "content.json.country",
"type": "map",
"colors": {
"scale" : "OrRd"
}
}
]
}
}
You can set the type scale you want (or the distribution of the values in the colors):
{
"dashboard": {
"charts" : [
{
"field": "content.json.country",
"type": "map",
"colors": {
"scale" : "OrRd",
"distrib" : "linear"
}
}
]
}
}
You can add a helping text for a particular chart, in the help
key:
{
"dashboard": {
"charts": [
{
"field": "fields.Themes",
"type": "pie",
"help": "Each document may be labelled with several themes"
}
]
}
}
This help text is a markdown text, and will appear after the title of the page
pages.chart.title
, and possibly after the charts' help text
pages.chart.help
, which appear on all charts' pages.
To indicate which field is used by a chart, you have to specify it inside the chart.
These are used to point inside the mongodb document, using the dot notation.
Often, they are placed in the content
field, or in fields
.
Ex:
"dashboard" : {
"charts": [
{
"field": "content.json.Py",
"type": "histogram"
},
{
"field": "content.json.DiscESI",
"type": "pie"
}
]
}
Maybe your fields are multivalued, for example, if you load csv
files.
For example, in a Keywords
columns, you have such values:
Dashboard; Nodejs; Github
Web; Dashboard; Statistics
The direct way, is to point to content.json.keywords
, but that will
distinguish the Dashboard
from the first row to the one from the second row.
Moreover, they will be bound to other keywords on the same row.
The solution is to add a document field in the JSON configuration file, using JBJ's syntax:
"documentFields" : {
"$Keywords" : {
"path" : "content.json.Keywords",
"parseCSV" : ";",
"foreach" : {
"trim" : true
}
}
},
Then, you have to add
"dashboard" : {
"charts": [
{
"field": "content.json.Py",
"type": "histogram"
},
{
"field": "Keywords",
"type": "pie"
}
]
}
Here is an example with a normal field Py
(Publication year, which
is unique in each row), and a multivalued one, Keywords
(several
keywords):
"documentFields" : {
"$Keywords" : {
"content.json.Keywords",
"parseCSV" : ";",
"foreach" : {
"trim" : true
}
}
},
"dashboard" : {
"charts": [
{
"field": "content.json.Py",
"type": "histogram"
},
{
"field": "Keywords",
"type": "pie"
}
]
}
By default, the charts operator is distinct
, which count the distinct values
of one field.
There are other operators, and some of them take more than one field.
For these cases, use operator
and fields
keys (fields
replaces field
,
or you can set only one field in this table).
Example: to display, in an histogram
, the total of citations (by year):
{
"fields": [
"content.json.Tc",
"content.json.Py"
],
"type": "histogram",
"operator": "sum_field1_by_field2",
"title": "Citations",
"help": "Total number of citations per year"
}
See operators.
flying
indicates to a chart that you want to apply the JBJ actions of the
said flyingFields
to the data elements returned by the operator.
Often, the operator will return a JSON page containing at least:
{
"recordsTotal": 108,
"recordsFiltered": 108,
"data": [
{
"_id": "Albania",
"value": 2
},
{
"_id": "Algeria",
"value": 15
}
]
}
If you declared a flyingFields
like:
"$country2iso": {
"$_id": {
"get": "_id",
"mapping": {
"Albania": "ALB",
"Algeria": "DZA"
}
},
"mask": "_id,value"
}
and added a flying
key to the chart, and its value was country2iso
, then
the data projected to the chart will be the previous data
, treated by
country2iso
:
{
"recordsTotal": 108,
"recordsFiltered": 108,
"data": [
{
"_id": "ALB",
"value": 2
},
{
"_id": "DZA",
"value": 15
}
]
}
In /chart.html
pages, you can see a chart, and a table with documents. This table display the fields you chose to put in the documentFields
key.
Here is an example, displaying Year
, Title
, Authors
, and Keywords
:
"documentFields" : {
"$year" : {
"visible": true,
"label": "Publication Year",
"path" : "content.json.Py"
},
"$title" : {
"visible": true,
"label": "Title",
"path" : "content.json.Ti"
},
"$authors": {
"visible": true,
"label": "Authors",
"path" : "content.json.Af"
},
"$keywords" : {
"visible": true,
"label": "Keywords",
"path" : "content.json.DiscESI"
}
}
All document fields which visible
key is set to true
will be
present in the table.
By default, visible
key value is false
.
The Search
field above the documents table uses a field named text
, which
must contain the content of all fields you want to be able to search.
Ex:
"documentFields" : {
"$text": {
"get" : ["content.json.Py", "content.json.Ti", "content.json.Af"],
"join": "|"
}
}
In a chart page, you can add facets: others fields' values.
Thus, you can have facets in each chart. Example:
"charts": [
{
"field": "fields.Section",
"type": "pie",
"title": "Sections",
"facets": [
{
"path": "content.json.Py",
"label": "Year",
"help": "Production per year",
"paging": false
},
{
"path": "fields.Themes",
"label": "Theme",
"column2": "#Publi"
}
]
},
Here, you have a pie displaying sections, and two facets:
content.json.Py
in the documentfields.Theme
in the documentThe first facet has a help
field, which will appear as a tooltip.
The first facet won't have pagination buttons, as its paging
field is set to
false
(this is useful when there is only one page to display).
The second facet changes the lable of its second columns to #Publi
, instead
of Occ
by default.
To indicate the title of a document, use the documentFields
named $title
.
In order to make the /display/id.html
page work, you have to declare all the
fields you want in the document's page.
They have to be in display.fields
, they'll be displayed in the same
order as their declaration's order.
Use "path": "label"
Ex:
{
"display" : {
"fields" : {
"fields.title": "Titre",
"fields.authors": "Auteurs",
"fields.year": "Année de publication",
"content.json.SourceCorrigee": "Source",
"content.json.DiscESI": "Discipline ESI",
"content.json.SectionEtude": "Marquage INSU - Section",
"content.json.La": "Langue de la publication",
"content.json.PaysFRERegroupe": "Pays",
"content.json.Di": "DOI",
"content.json.Ut": "Identifiant WoS"
}
}
}
To change the fields column's width, use fieldsWidth
within display
key:
{
"display": {
"fields": {
"title" : "Title",
"year" : "Year",
"director": "Director",
"vactors" : "Actors"
},
"fieldsWidth": "50%"
}
}
Any CSS width will work. However, be aware that it is preferable to let it be responsive.
Each URL of the theme may be customized:
title
(appears in the head of the page, and is a part of the browser's tab)description
(short description of the page)help
(first paragraph in the page)For example, you may customize the /index.html
page using:
"pages" : {
"index" : {
"title" : "Dashboard",
"description" : "Study Foo's dashboard",
"help" : "This comes from Web Of Science, and does only contain documents from Foo University."
}
}
Notice that the path for index
settings is pages.index
.
By default, this castor theme is able to load CSV files (each line being a document).
Here is the default settings:
"loaders": [
{
"script": "castor-load-csv",
"pattern": "**/*.csv"
}
]
These settings mean that all .csv
files within the data directory (and all
its descendants, thanks to **/
) will be loaded by
castor-load-csv
.
You can add any castor loader you find in castorjs repositories (except castor-load, which manages (all loaders).
For example, to enable castor to load XML files, each containing several documents (in /RDF/Topic
), use:
"loaders" : [
{
"script" : "castor-load-xmlcorpus",
"pattern" : "**/*.xml",
"options" : {
"cutter" : "/RDF/Topic"
}
}
]
There is an example of TSV settings in the showcase.
If you want restrict access to your ezVIS, add an access
key containing
login
and plain
or sha1
subkeys.
Using plain
will bypass sha1
value.
login
is a username.
plain
is plain password.
sha1
is the SHA-1 hash of the password (so that it will not be stored in the settings).
Example for a pwd
value of the password:
"access": {
"login": "user",
"sha1" : "37fa265330ad83eaa879efb1e2db6380896cf639"
}
Warning: when you access the ezVIS report from the same machine as the one running the server, you will not be asked for your identity. This is to allow
local:///
protocol to work, even when not knowing the password (see #corpusFields).
Tip: to generate a SHA1, either use a Linux commande like
sha1sum
orshasum
(be careful: don't integrate any carriage return, use^D
at the end of plain password), or online services like SHA-1 online
FAQs
A dashboard to visualize a synthesis on a structured corpus, using several charts (pie, histogram, ...)
The npm package ezvis receives a total of 0 weekly downloads. As such, ezvis popularity was classified as not popular.
We found that ezvis demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 2 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
cURL and Go security teams are publicly rejecting CVSS as flawed for assessing vulnerabilities and are calling for more accurate, context-aware approaches.
Security News
Bun 1.2 enhances its JavaScript runtime with 90% Node.js compatibility, built-in S3 and Postgres support, HTML Imports, and faster, cloud-first performance.
Security News
Biden's executive order pushes for AI-driven cybersecurity, software supply chain transparency, and stronger protections for federal and open source systems.