Date | Description of Major Changes | Editor |
2021-08-05 | Turn loose deployment notes into formal guide | Jamie |
2021-08-09 | Complete deployment checklist | Jamie |
2021-09-13 | Review comments + add example code changes | Jamie |
2021-09-15 | Final revisions for V1 | Jamie |
This document will walk you through the process of deploying your own local version of the Urban Pandemic Preparedness Dashboard (referred to as UPPD or ‘the application’).
For your convenience, we have provided a main
git branch that includes all assets required to run the application for Los Angeles. There is also a clean
branch that has cleared out all example information for users wishing to deploy their own version of the tool.
You can view a live demo of the Los Angeles dashboard here
Below are the full requirements for the index and asset files to be compatible with our deployment package as well as examples and instructions for customizing the tool. Please note that the guide below assumes you have already completed the index calculations in a structure somewhat similar to our own - please see the methodology documents in this directory for details on our index methodology.
If you have any questions about deploying your UPPD with additional customization or require assistance with data integration beyond the scope of this document, please contact the SecDev Group at secdev@secdev.com
- Docker Engine
- Server running Ubuntu or Mac OS with >= 4GB RAM
- Approximately 6 GB of hard drive space
- A Mapbox API key with default permissions
- It’s strongly recommended that you set up npm (via installing Node.js) to run the application side independently and test small visual or textual changes
- This application was developed and tested in Mac OS and Ubuntu. It may not behave as expected when deployed to a Windows server.
There are four main subdirectories, each representing a component of the application. These subdirectories have additional README’s composed by our partners in developing the frontend, RS21.
uppd-docker-compose
: Contains docker-compose file for easy deployment of all components. Only potential changes needed involve changing ports depending on your deployment needs.uppd-tile-server
: Contains the code to initalize the tile server for the map - the coloured heatmap tiles and the points of interest are shown by querying this serveruppd-database
: Contains the code to initialize all datasets for the map, including importing indices, points of interest, and shapefile from user-added files.uppd-application
: Contains all the front end code. This is the directory that will require the most customization to deploy your own version.
There are three main types of data files, all of which are located in the uppd-database/docker/data
folder.
- File names and column names must not contain any characters other than a-z (lowercase only), 0-9, and the underscore character (
_
) - Spaces, hyphens, capital letters, or other characters may cause errors when initializing the database or when connecting the database to the app.
- Keep track of the names of the columns you would like to show on the front end of the application
- These files are to enable a view that includes important landmarks
- Files should be separated by landmark type - for example, libraries and hospitals would be in separate .csv files
- They should have at least 3 columns:
- a column for the label you wish to show on the front end
- ‘latitude’
- ‘longitude’
- Note the name of the label column for use with uppd-application later.
- Files must be .csv files located in uppd-database/docker/data/csv/assets
- The primary dataset to be visualized - in our demo, this is the UPPD Index Score or Risk Score. It is also referred to as
city_metrics
in the database - Aside from the points of interest and shapefile, all data that should be included in the dashboard should be in the index files.
- Must be .csv files located in
uppd-database/docker/data/csv/indices
- The most recent index file should be called
current_indices.csv
- Currently, this project does not support automatic data updates. This means that adding additional years of data to V1 of the application will require current_indices to be replaced and the database to be rebuilt
- The most recent index file will be used to create the database table for the indices.
- Any additional index files must have the same column names in the same order as current_indices, or data may be added inaccurately.
- Additional index files should be called
filename_YYYY_indices.csv
, where YYYY is the year of data represented in the indices file - The database expects a column called
tractce10
to be in current_indices to align with the shapefile.- If you have a different column name for the main code or id used for the areas of interest (usually the smallest available unit, such as a census tract, neighborhood, or Lower-layer Super Output Areas (LSOAs)), adjust the name in
uppd-database/src/sql/views/05_view_data.sql
line 4, after them.
- If you have a different column name for the main code or id used for the areas of interest (usually the smallest available unit, such as a census tract, neighborhood, or Lower-layer Super Output Areas (LSOAs)), adjust the name in
- The database expects a column called
source_date
with dates in the format YYYY-MM-DD or MM/DD/YYYY. - Take note of the columns which have data you will want to display on the front end. Consult the methodology documents in
uppd-application/src/assets
for details about how we organized the column names and calculated the values for our deployments
- Must be named
city_geography.shp
and be located inuppd-database/docker/data/shapefile
- Note the name of the column with the id for the regions to match with the indices. If it is not ‘tractce’, update
uppd-database/src/sql/views/05_view_data.sql
on line 4 and put the column name afterg.
- If there are special characters in the shapefile that go beyond UTF-8, you may have to modify line 52 of uppd-database/docker/dbscripts/02-load-data.sh to change the character set used for the postgis database
Example Alteration: uppd-database/src/sql/views/05_view_data.sql
Default - the location id columns are assumed to be tractce10
in both the shapefile (city_geography.shp) and the indices files (current_indices.csv). The city_geography id column is changed to tractce using ALTER TABLE on lines 1-2 before creating the view to avoid errors due to the duplicate name
ALTER TABLE city_geography
RENAME COLUMN tractce10 to tractce;
CREATE VIEW view_data AS
SELECT *
FROM city_metrics AS m, city_geography AS g
WHERE m.tractce10::TEXT = g.tractce;
Example after changing 05_view_data.sql for Amsterdam’s data files - note that the ALTER TABLE rename on lines 1-2 is no longer required since the columns already have two different names
CREATE VIEW view_data AS
SELECT *
FROM city_metrics AS m, city_geography AS g
WHERE m.neighborhood_code::TEXT = g.buurt_code;
Example Alteration: uppd-database/docker/dbscripts/02-load-data.sh, lines 49-53
Default - no -W flag is given, so the conversion from shapefile to postgis compatible data is done assuming the character setting should be in UTF-8
for f in $(ls /usr/local/data/shapefile/*.shp); do
echo " loading shapefile data $f..." 2>&1 | tee -a $logData
NAME=`echo $f | sed 's:.*/::' | cut -d'.' -f1`
shp2pgsql -s 4326 -I $f ${SCHEMA}.${NAME} | psql -d ${DB} -U ${POSTGRES_USER} 2>&1 | tee -a $logData | grep -Ev "INSERT 0 1|^$"
done
Example after changing 02-load-data.sh for Amsterdam’s shapefile.
Because special characters are included in the file, the character set is changed to LATIN1 using the -W tag, which allows the shapefile to be successfully converted into postgis compatible data. See postgis manual section 4.3.2 for details.
for f in $(ls /usr/local/data/shapefile/*.shp); do
echo " loading shapefile data $f..." 2>&1 | tee -a $logData
NAME=`echo $f | sed 's:.*/::' | cut -d'.' -f1`
shp2pgsql -s 4326 -W "LATIN1" -I $f ${SCHEMA}.${NAME} | psql -d ${DB} -U ${POSTGRES_USER} 2>&1 | tee -a $logData | grep -Ev "INSERT 0 1|^$"
done
uppd-application/src/configuration
contains three files for customizing the front end of the application.
img-config.ts lines 1-25
// main logo used at the top left corner
import navLogo from '../assets/appBranding/Seal_of_Los_Angeles.svg';
export const mainLogo = navLogo;
// Background images
import backgroundImageOne from '../assets/img/background-image.jpg';
import backgroundImageTwo from '../assets/img/la-bg2.jpg';
[[...
lines 8-18
...]]
// Background image for the About Page
export const AboutBackgroundImage = backgroundImageOne;
// Background image for the Info Page
export const InfoBackgroundImage = backgroundImageTwo;
Image files can be placed in uppd-application/src/assets and be imported in img-config.ts to customize the logo in the top left corner of the application as well as the background images used on the main landing page (AboutBackgroundImage
) and the info / methodology page (InfoBackgroundImage
)
theme-color-config.ts lines 4 - 67
export const mapGradientDark = {
step6: '#FEE5D9',
step5: '#FCBBA1',
step4: '#FC9272',
step3: '#FB6A4A',
step2: '#DE2D26',
step1: '#A50F15',
};
export const mapGradientLight = {
step6: '#EFF3FF',
step5: '#C6DBEF',
step4: '#9ECAE1',
step3: '#6BAED6',
step2: '#3182BD',
step1: '#08519C',
};
export const themeOverrides: Overrides = {
MuiButton: {
label: {
textTransform: 'capitalize',
},
},
};
export const darkTheme = createMuiTheme({
palette: {
type: 'dark',
primary: {
main: mapGradientDark.step1,
light: mapGradientDark.step2,
},
secondary: {
main: '#FDCDBA',
},
background: {
default: '#1E1E1E',
},
warning: {
main: '#F3F800',
},
},
overrides: themeOverrides,
});
export const lightTheme = createMuiTheme({
palette: {
type: 'light',
primary: {
main: mapGradientLight.step1,
},
secondary: {
main: '#A5CDE4',
},
background: {
default: '#F3F3F3',
},
warning: {
main: '#DBDB0F',
},
},
overrides: themeOverrides,
});
This file allows you to customize the colours used in light and dark mode (lightTheme
and darkTheme
), as well as the colours used in the heatmap gradient (mapGradientLight
and mapGradientDark
). step6
of the map gradient is the lightest colour, used to show the highest values of the index (lowest risk) and step1
is the darkest colour, used to show the lowest values of the index (highest risk). step1
and step2
for the light and dark gradients are re-used in the themes for the overall application to provide colour coherence, but any of these values can be changed and replaced with a hex colour code.
app-config.ts
The bulk of the front end configuration happens in this file. Please be sure to review app-config carefully before deploying.
mapAreaConfig (lines 9-21)
export const mapAreaConfig = {
zoomLevel: 9,
mapCenter: [-118.2437, 34.0522], // [longitude, latitude]
bounds: [
[-119.24, 33.05], // Southwest Coordinates (bottom left)
[-117.24, 35.05], // Northeast Coordinates (top right)
],
style: {
dark: 'mapbox://styles/jharnum/cknsxylu20drd17pbsf07gytm',
light: 'mapbox://styles/jharnum/cknsy11dh0q3h18ql5qeh7f41',
satellite: 'mapbox://styles/jharnum/cknsxxxqq0q1j17qohk46xz4l',
},
};
-
This section allows the user to define the area that should be displayed by the map.
mapCenter
is the longitude and latitude for the central point on the first load of the map.bounds
indicates the limits of the map using the bottom left and top right points. -
style
indicates the Mapbox style to use for the map in each of three ‘modes’ - dark, light, and satellite. You may use the styles provided or see the Mapbox documentation for more details on creating your own styles.
filterScale (lines 23-27)
export const filterScale = {
lowBound: 0,
highBound: 100,
step: 1,
};
- This variable allows the user to set the bounds for the main index and main index subcategories. This is what is used to create the filterable ‘scale’ of index results on the main map page. Here it has been set to display 0 to 100 and allow users to step in increments of 1.
Main Column Variables (lines 29-31)
export const tractId = 'tractce10';
export const primaryScore = 'secdev_res_plr';
export const totalPopCol = 'tot_pop_e';
tractId
is the name of the column to use for matching the region to its shape in the shapefile.primaryScore
is the name of the column with the overall index score - the main value to be displayed when a tract is selected.totalPopCol
is the name of the column with the total population of a given region
Years of Data (lines 33-35 main branch)
export const currentYear = 2019;
export const availableYears: number[] = [2019, 2018, 2017];
export const projectedYears: number[] = [];
currentYear
is the most recent year of data available. We found that data was most rich for 2019 in most cities we have developed for so far.availableYears
is used to create the year slider.- If only one year of data is available, put only one year between the brackets.
- You may also wish to remove the
DateSlider
component from the sidebar - seeuppd-application/src/components/sidebar/Sidebar.tsx
in the main branch and comment out<DateSlider />
on line 207 as well as theDateSlider
import on line 26.
- You may also wish to remove the
- If only one year of data is available, put only one year between the brackets.
projectedYears
allows a warning message to be displayed when certain years are selected. This is to allow for data that has been created using predictive analysis of some kind. Any year that is not projected can be assumed to be real data rather than projected or interpolated data.
mapLayers (lines 37 - 218) The mapLayers variable is a list of dictionary (or key: value) items, starting with the main index value (in our case, the Urban Pandemic Preparedness Index). Below are two key examples of these items:
Line 38-42 - Main index value
{
title: 'Urban Pandemic Preparedness Index',
colName: 'secdev_res_plr',
subcategories: [],
},
- Note that no subcategories are listed for the main value - this is because they are listed as separate dictionary items in the rest of the list.
title
is what will be displayed on the dropdown menu and sidebar for this item andcolName
is the column in the index file that contains the data.
Lines 43-55 - Economic factors
{
title: 'Economic Factors',
colName: 'econ_fctrs_plr',
subcategories: [
{
title: 'Percent living below poverty',
colName: 'pvrty_pe',
},
{ title: 'Percent unemployed', colName: 'unemp_pe' },
{ title: 'Income per capita', colName: 'pci_e' },
{ title: 'Percent without a high school diploma', colName: 'nohsdp_plr' },
],
},
- Here we have four subcategories or metrics listed in the Economic Factors section. When the ‘Economic Factors’ section is clicked in the sidebar, a section will appear showing the individual metrics in this category.
title
will display on the sidebar as the label for the value in the column defined incolName
. - The user can add as many or as few of these dictionary items as they would like to display. Currently, no additional number formatting is available, so it may be useful to state in words if a value is a percentage or not, as shown in the example above.
radarChartConfig (lines 220-236)
The radar or spider chart is used to display the main category values relative to one another in order to view which aspects a particular region may be stronger or weaker in than others. These are the categories defined in mapLayers
that have subcategories.
export const radarChartConfig: RadarChartDataType = {
enabled: true,
fields: [
{ title: 'Econ. Factors', colName: 'econ_fctrs_plr' },
{ title: 'Disease Factors', colName: 'chronic_fctrs_plr' },
{ title: 'Demo. Factors', colName: 'demograph_fctrs_plr' },
{ title: 'Social Factors', colName: 'social_fctrs_plr' },
{
title: 'Lifestyle Factors',
colName: 'clncl_fctrs_plr',
},
{
title: 'Digital Prep.',
colName: 'digital_fctrs_plr',
},
],
};
- Because of the small space available for the radar chart labels, some
title
s have been abbreviated. ThecolName
s should match thecolName
used for that subcategory in themapLayers
variable discussed above.
linearCharts (lines 239-264) These charts can be used to display the relative numbers of binary groups. In our Los Angeles example, we used Under and Over 65 and Male and Female linear charts. See the Age Distribution chart code below.
export const linearCharts: LinearChartDataType[] = [
{
title: 'Age Distribution',
chartId: 'ageDist',
data: {
totalPopulation: totalPopCol,
secondaryCount: 'age65p_pe',
},
labels: {
left: 'Under 65',
right: 'Over 65',
},
title
is the main label for the chart andchartId
provides an id for this component.data
contains two data columns: thetotalPopulation
column (defined above in totalPopCol variable) and the secondaryCount, which is currently assumed to be a percentage value. I.e. the column namedage65p_pe
contains the percentage of people over 65 in that region.labels
are used to label each side of the chart. Here, the left will be the ‘primaryCount’ (calculated by the application) - the percentage of people under 65 and the right will be oursecondaryCount
defined above.
racialDistInfo (lines 266-291) This displays the percentage of people in a region belonging to racial or ethnic categories defined in the main indices dataset. As many of these categories can be displayed as needed. Below is an abbreviated example.
export const racialDistInfo: RacialDistDataType[] = [
{
title: 'African American',
colName: 'african_american',
},
{
title: 'White',
colName: 'white',
},
- The
title
is used to label the value incolName
, which is assumed to be already in percentage format. - A tooltip explaining the origins of the category names and that the values may add to more than 100% is included in
uppd-application/src/components/Sidebar/CensusInfo/RacialDistro.tsx
on line 38 - different regions may define racial and ethnic groups differently, so changes to this tooltip may help the user understand where the groupings are coming from.
PointsOfInterest (lines 293-348) This is where you tell the front end application which points of interest you added to the database. You can have as many or as few as you like.
export const PointsOfInterest: PointsOfInterestType[] = [
{
title: 'Hospitals',
endpoint: 'hospitals_and_medical_centers',
icon: 'hospital-15',
nameField: 'name',
},
{
title: 'Libraries',
endpoint: 'libraries',
icon: 'library-11',
nameField: 'name',
},
title
is the label to display in the toggle dropdown for points of interest.endpoint
is the name of the csv file you added to the database’s csv asset folder without the.csv
extension. So in this example, the file with the hospital locations was calledhospitals_and_medical_centers.csv
.icon
is the name of a maki icon - you can find the full icon set here and here. The number after the dash handles sizing in pixels - so here, the library icon is set to appear smaller than the hospital icon.nameField
is the column name in the original csv file that contains the name of the location you would like to have displayed when that point is selected.
Finally, on line 350 of app-config.ts there is a sidebarText
variable. If desired, users can add a sentence or two of text here (without formatting) and it will display in the sidebar.
- Docker is installed and running - https://docs.docker.com/engine/install/
- Node.js is installed - https://nodejs.org/en/
- Git clone the repository ((link here))
- Get Mapbox API key
- Add Mapbox API key to uppd-application/.env
- Add Mapbox API key to uppd-tile-server/html/index.html, line 29
- Edit src/sql/00_db_setup.sql to include secure passwords instead of default value
- Add changed passwords to uppd-tile-server/.env so it can access uppd-database
- Verify data files fit the requirements detailed above
- Shapefile, city_geography.shp
- Asset or point of interest files, .csv
- Index files, at least current_indices.csv
- Add custom images, if desired, to uppd-application/src/assets and ensure they are imported in uppd-application/src/configuration/img-config.ts 5. navLogo sets the branding image in the top left corner 6. backgroundImageOne sets the main landing page or About page background 7. backgroundImageTwo sets the Info / Methodology page background
- Use google maps or another mapping service to get the
mapCenter
andbounds
to use inuppd-application/src/configuration/app-config.ts
(line 12-15) anduppd-tile-server/html/index.html
(line 34). If using google maps, reverse the order of the numbers - google uses (latitude, longitude) and mapbox uses (longitude, latitude). - Populate the values in lines 29 - 46 of 1uppd-application/src/configuration/app-config.ts1 using the column names from your current_indices.csv file. See Front End Application Configuration above for details.
- Populate the
mapLayers
variable, starting on line 49 ofuppd-application/src/configuration/app-config.ts
. See Front End Application Configuration above for details. - Populate the
radarChartConfig
variable, starting on line 206 ofuppd-application/src/configuration/app-config.ts
. The radar chart displays the main sub-categories of the index to show their distribution. Shortened versions of the titles may look better for this chart. See Front End Application Configuration above for details. - Populate the
linearCharts
variable starting on line 229 ofuppd-application/src/configuration/app-config.ts
. See Front End Application Configuration above for details. - Populate the
racialDistInfo
variable starting on line 256. This is used to show population distribution according to more than two variables, which may add up to more than 100%. See Front End Application Configuration above for details. - Populate the
PointsOfInterest
variable starting on line 285. This is used to tell the map about the asset files. See Front End Application Configuration above for details. - Run
npm run lint
in uppd-application and follow any warnings or error messages that appear - Check docker-compose.yml in uppd-docker-compose to ensure that none of the ports are already in use. If they are, change them only on the left side of the colon (eg.
80:80
may become81:80
if port 80 is already in use - Use the command
docker-compose up --build
to build the entire project. Pay attention to any error messages that may appear, particularly when the database is being initialized - If there are database errors, use
ctrl+c
to stop the programs and then use the commanddocker system prune
to fully remove the database that was set up. This helps to ensure that old errors won’t crop up again unexpectedly - When you have verified everything is working correctly, you can stop (ctrl+c) and restart the program using the command
uppd-docker-compose up --build -d
- this starts it in the background. - Verify that the three program components - database, tile server, and application - are running using the command ‘docker ps’
- Visit
localhost:8080
to view the application on your local server
- Use
uppd-database/data_check.py
to run some preliminary tests on your data files before standing up the application for the first time (please note that this is only a quick script meant for people familiar with coding and will field many changes back to the user) - Running
npm run start
from the command line in the uppd-application directory will stand up a version of the front end atlocalhost:3000
. It can still access the tile server and database as long as they are still running.- If you have not done so before, you should run
npm install
first to ensure all appropriate packages are installed. npm run start
can be used to check changes made to the front end only, such as changes toapp-config.ts
, as it will create a hot-loading instance of the application.
- If you have not done so before, you should run
- If the database is built with inaccurate data, remove the image using
docker rm database
and re-build once the data has been corrected - If changes have been made to different parts of the program, it may be helpful to stop all uppd images and run
docker system prune
to clean them up before rebuilding - If a production deploy is desired, be sure to
- Search all directories for
localhost
and change as appropriate - Ensure that all database credentials have been changed appropriately
- Search all directories for