diff --git a/CHANGELOG.md b/CHANGELOG.md
index a5ad3220e6091ead4a5d29c4a7610787d80b568e..eb1780c9b5bef24533b225a4301f618255a71bc9 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -8,6 +8,7 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.
 ### Added
 - Image server now accepts the period parameter which should be a ISO-8601 duration format. 
 However please note that NWIS only accepts periods using xxD.
+- Image server shows site name and number above the graph
 
 ## [0.22.0](https://github.com/usgs/waterdataui/compare/waterdataui-0.21.0...waterdataui-0.22.0) - 2019-12-12
 ### Changed
diff --git a/assets/src/scripts/components/hydrograph/index.js b/assets/src/scripts/components/hydrograph/index.js
index e2344c94e923e184f47604d80ce361d17630b66a..c8e6b5e155f4c1bafd574cfce4d992672a2ae480 100644
--- a/assets/src/scripts/components/hydrograph/index.js
+++ b/assets/src/scripts/components/hydrograph/index.js
@@ -12,7 +12,8 @@ import { dispatch, link, provide } from '../../lib/redux';
 
 import { addSVGAccessibility } from '../../accessibility';
 import config from '../../config';
-import { isLoadingTS, hasAnyTimeSeries } from '../../selectors/time-series-selector';
+import {isLoadingTS, hasAnyTimeSeries, getMonitoringLocationName,
+    getAgencyCode} from '../../selectors/time-series-selector';
 import { Actions } from '../../store';
 import { callIf, mediaQuery } from '../../utils';
 
@@ -231,9 +232,20 @@ const plotAllMedianPoints = function (elem, {visible, xscale, yscale, seriesPoin
 };
 
 
-const createTitle = function(elem) {
-    elem.append('div')
-        .classed('time-series-graph-title', true)
+const createTitle = function(elem, siteNo, showMLName) {
+    let titleDiv = elem.append('div')
+        .classed('time-series-graph-title', true);
+
+    if (showMLName) {
+        titleDiv.append('div')
+            .call(link((elem, {mlName, agencyCode}) => {
+                elem.html(`${mlName}, ${agencyCode} ${siteNo}`);
+            }, createStructuredSelector({
+                mlName: getMonitoringLocationName(siteNo),
+                agencyCode: getAgencyCode(siteNo)
+            })));
+    }
+    titleDiv.append('div')
         .call(link((elem, title) => {
             elem.html(title);
         }, titleSelector));
@@ -262,11 +274,11 @@ const watermark = function (elem) {
         }, layoutSelector));
 };
 
-export const timeSeriesGraph = function(elem) {
+export const timeSeriesGraph = function(elem, siteNo, showMLName) {
     elem.append('div')
         .attr('class', 'hydrograph-container')
         .call(watermark)
-        .call(createTitle)
+        .call(createTitle, siteNo, showMLName)
         .call(createTooltipText)
         .append('svg')
             .attr('xmlns', 'http://www.w3.org/2000/svg')
@@ -501,7 +513,17 @@ const dataLoadingAlert = function(elem, message) {
     }
 };
 
-export const attachToNode = function (store, node, {siteno, parameter, compare, period, cursorOffset, showOnlyGraph = false} = {}) {
+export const attachToNode = function (store,
+                                      node,
+                                      {
+                                          siteno,
+                                          parameter,
+                                          compare,
+                                          period,
+                                          cursorOffset,
+                                          showOnlyGraph = false,
+                                          showMLName = false
+                                      } = {}) {
     const nodeElem = select(node);
     if (!siteno) {
         select(node).call(drawMessage, 'No data is available.');
@@ -541,7 +563,7 @@ export const attachToNode = function (store, node, {siteno, parameter, compare,
     // Set up rendering functions for the graph-container
     nodeElem.select('.graph-container')
         .call(link(controlDisplay, hasAnyTimeSeries))
-        .call(timeSeriesGraph, siteno)
+        .call(timeSeriesGraph, siteno, showMLName)
         .call(callIf(!showOnlyGraph, cursorSlider))
         .append('div')
             .classed('ts-legend-controls-container', true)
diff --git a/assets/src/scripts/selectors/time-series-selector.js b/assets/src/scripts/selectors/time-series-selector.js
index 3d140531de630d56979e27f61a3112994d264cc3..c6e7b5fbed85f04d4d7febe88cc276afc2664e4b 100644
--- a/assets/src/scripts/selectors/time-series-selector.js
+++ b/assets/src/scripts/selectors/time-series-selector.js
@@ -8,6 +8,10 @@ import { createSelector } from 'reselect';
  */
 export const getVariables = state => state.series.variables ? state.series.variables : null;
 
+export const getSourceInfo = state => state.series.sourceInfo || {};
+
+export const getSiteCodes = state => state.series.siteCodes || {};
+
 export const getMethods = state => state.series.methods ? state.series.methods : {};
 
 export const getQueryInfo = state => state.series.queryInfo || {};
@@ -33,6 +37,23 @@ export const hasAnyTimeSeries = state => state.series && state.series.timeSeries
  * Selectors the return derived data from the state
  */
 
+/*
+ * @param {String} siteno
+ * @return {String} monitoring loation name. Returns empty string if state does not contain siteNo.
+ */
+export const getMonitoringLocationName = memoize((siteNo) => createSelector(
+    getSourceInfo,
+    (sourceInfo) => siteNo in sourceInfo ? sourceInfo[siteNo].siteName || '' : ''
+));
+
+/*
+ * @param {String} siteno
+ * @return {String} agency code for siteno
+ */
+export const getAgencyCode = memoize((siteNo) => createSelector(
+    getSiteCodes,
+    (siteCodes) => siteNo in siteCodes ? siteCodes[siteNo].agencyCode || '' : ''
+));
 /*
  * @return {Object}     Variable details for the currently selected variable or null.
  */
diff --git a/assets/src/scripts/selectors/time-series-selector.spec.js b/assets/src/scripts/selectors/time-series-selector.spec.js
index 46ebe46339963d0817cf1cd270d06f50c9d2b036..2eed1e93bfa4a3908d13ffbccf027bb93cf0300e 100644
--- a/assets/src/scripts/selectors/time-series-selector.spec.js
+++ b/assets/src/scripts/selectors/time-series-selector.spec.js
@@ -1,4 +1,7 @@
-import { getVariables, getCurrentVariableID, getCurrentDateRange, getCurrentVariable, getQueryInfo, getRequests, getCurrentParmCd, hasTimeSeries, getTsRequestKey, getTsQueryInfo, getRequestTimeRange, isLoadingTS, getTSRequest, getTimeSeriesCollectionIds, getIanaTimeZone, getNwisTimeZone } from './time-series-selector';
+import { getVariables, getSourceInfo, getSiteCodes, getCurrentVariableID, getCurrentDateRange,
+    getMonitoringLocationName, getAgencyCode, getCurrentVariable, getQueryInfo, getRequests, getCurrentParmCd,
+    hasTimeSeries, getTsRequestKey, getTsQueryInfo, getRequestTimeRange, isLoadingTS, getTSRequest,
+    getTimeSeriesCollectionIds, getIanaTimeZone, getNwisTimeZone } from './time-series-selector';
 
 describe('timeSeriesSelector', () => {
     const TEST_VARS = {
@@ -30,6 +33,54 @@ describe('timeSeriesSelector', () => {
         });
     });
 
+    describe('getSourceInfo', () => {
+        it('Return an empty object if series is empty', () => {
+            expect(getSourceInfo({
+                series: {}
+            })).toEqual({});
+        });
+
+        it('Return the sourceInfo if in series', () => {
+            expect(getSourceInfo({
+                series: {
+                    sourceInfo: {
+                        '0537000': {
+                            siteName: 'Site Name'
+                        }
+                    }
+                }
+            })).toEqual({
+                '0537000': {
+                    siteName: 'Site Name'
+                }
+            });
+        });
+    });
+
+    describe('getSiteCodes', () => {
+        it('Return an empty object if series is empty', () => {
+            expect(getSiteCodes({
+                series: {}
+            })).toEqual({});
+        });
+
+        it('Return the siteCodes if in series', () => {
+            expect(getSiteCodes({
+                series: {
+                    siteCodes: {
+                        '0537000': {
+                            agencyCode: 'USGS'
+                        }
+                    }
+                }
+            })).toEqual({
+                '0537000': {
+                    agencyCode: 'USGS'
+                }
+            });
+        });
+    });
+
     describe('getQueryInfo', () => {
         it('Return empty object if series is empty', () => {
             expect(getQueryInfo({
@@ -90,6 +141,56 @@ describe('timeSeriesSelector', () => {
         });
     });
 
+    describe('getMonitoringLocationName', () => {
+        const TEST_INFO = {
+            series: {
+                sourceInfo: {
+                    '01010101': {
+                        'siteName': 'My Site Name'
+                    }
+                }
+            }
+        };
+        it('Returns empty string if state has no sourceInfo', () => {
+           expect(getMonitoringLocationName('12345678')({
+               series: {}
+           })).toBe('');
+        });
+
+        it('Returns empty string if siteNo is not in sourceInfo', () => {
+            expect(getMonitoringLocationName('12345678')(TEST_INFO)).toBe('');
+        });
+
+        it('Returns the monitoring location name for the site', () => {
+            expect(getMonitoringLocationName('01010101')(TEST_INFO)).toBe('My Site Name');
+        });
+    });
+
+    describe('getAgencyCode', () => {
+        const TEST_SITE_CODES = {
+            series: {
+                siteCodes: {
+                    '01010101': {
+                        'agencyCode': 'USGS'
+                    }
+                }
+            }
+        };
+        it('Returns empty string if state has no siteCodes ', () => {
+           expect(getAgencyCode('12345678')({
+               series: {}
+           })).toBe('');
+        });
+
+        it('Returns empty string if siteNo is not in siteCodes', () => {
+            expect(getAgencyCode('12345678')(TEST_SITE_CODES)).toBe('');
+        });
+
+        it('Returns the agency code  for the site', () => {
+            expect(getAgencyCode('01010101')(TEST_SITE_CODES)).toBe('USGS');
+        });
+    });
+
     describe('getCurrentVariableID', () => {
         it('Return the current variable ID', () => {
             expect(getCurrentVariableID({
diff --git a/graph-server/src/renderer/index.js b/graph-server/src/renderer/index.js
index 22ea5b9212ea4c146b727b4dfe53e67b0bca3dd7..10c24f38f908a52e6e7766b2052b1ea4a52b8a96 100644
--- a/graph-server/src/renderer/index.js
+++ b/graph-server/src/renderer/index.js
@@ -13,7 +13,8 @@ const renderToResponse = function (res, {siteID, parameterCode, compare, period}
         compare: compare,
         period: period,
         cursorOffset: false,
-        showOnlyGraph : true
+        showOnlyGraph : true,
+        showMLName: true
     };
     renderPNG({
         pageURL: 'http://wdfn-graph-server',