Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update ArcGisMapServerImageryProvider.js to add support for refreshing the token. #25

Merged
merged 28 commits into from
Oct 13, 2017
Merged
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
5b29adb
[TerriaJS/terriaJS#2588] ArcGisMapServerImagryProvider: Remove stray …
a-stacey Sep 27, 2017
d3de10e
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor the…
a-stacey Sep 27, 2017
849a9ba
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor the…
a-stacey Sep 27, 2017
7de4eb2
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Add a .token…
a-stacey Sep 27, 2017
5f77795
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Add a callba…
a-stacey Oct 3, 2017
33eea2e
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Add ._reques…
a-stacey Oct 3, 2017
7b54c8e
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Partially op…
a-stacey Oct 4, 2017
c6209ec
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor to …
a-stacey Oct 4, 2017
ca43393
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Add 499 erro…
a-stacey Oct 4, 2017
06470fe
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Change let's…
a-stacey Oct 9, 2017
58a010d
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Fix spelling…
a-stacey Oct 9, 2017
31ac6ff
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Simplify tok…
a-stacey Oct 9, 2017
e3e4368
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Change from …
a-stacey Oct 9, 2017
8c07960
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Fix a subtle…
a-stacey Oct 10, 2017
ae96af9
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Throttle the…
a-stacey Oct 9, 2017
48d8a44
[TerriaJS/terriaJS#2588] Add change log entry for this branch.
a-stacey Oct 11, 2017
26801a9
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Fix document…
a-stacey Oct 11, 2017
7aca562
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Relocated pr…
a-stacey Oct 11, 2017
d8bd601
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor thr…
a-stacey Oct 12, 2017
2a45ade
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor req…
a-stacey Oct 12, 2017
e3c2f8d
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor req…
a-stacey Oct 12, 2017
9f9bd5d
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Handle if th…
a-stacey Oct 12, 2017
7b931ac
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor han…
a-stacey Oct 12, 2017
b69ab2a
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Refactor han…
a-stacey Oct 12, 2017
43f6783
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Use the refa…
a-stacey Oct 12, 2017
a3f06ef
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Fix lint war…
a-stacey Oct 12, 2017
3b2b9f7
[TerriaJS/terriaJS#2588] ArcGisMapServerImageryProvider: Move functio…
a-stacey Oct 13, 2017
22609c8
Remove redundant section
kring Oct 13, 2017
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGES.md
Original file line number Diff line number Diff line change
@@ -4,6 +4,7 @@ Change Log
### TerriaJS-only

* Fixed a bug that could cause tiles to be missing from the globe surface, especially when starting with the camera zoomed close to the surface.
* Added support for refreshing expired tokens for `ArcGisMapServerImageryProvider` via callback registered with `options.requestNewToken` in constructor.

### 1.33 - 2017-05-01

306 changes: 209 additions & 97 deletions Source/Scene/ArcGisMapServerImageryProvider.js
Original file line number Diff line number Diff line change
@@ -11,11 +11,13 @@ define([
'../Core/Ellipsoid',
'../Core/Event',
'../Core/GeographicTilingScheme',
'../Core/loadImageViaBlob',
'../Core/loadJson',
'../Core/loadJsonp',
'../Core/Math',
'../Core/Rectangle',
'../Core/RuntimeError',
'../Core/throttleRequestByServer',
'../Core/TileProviderError',
'../Core/WebMercatorProjection',
'../Core/WebMercatorTilingScheme',
@@ -35,11 +37,13 @@ define([
Ellipsoid,
Event,
GeographicTilingScheme,
loadImageViaBlob,
loadJson,
loadJsonp,
CesiumMath,
Rectangle,
RuntimeError,
throttleRequestByServer,
TileProviderError,
WebMercatorProjection,
WebMercatorTilingScheme,
@@ -59,6 +63,8 @@ define([
* @param {Object} options Object with the following properties:
* @param {String} options.url The URL of the ArcGIS MapServer service.
* @param {String} [options.token] The ArcGIS token used to authenticate with the ArcGIS MapServer service.
* @param {ArcGisMapServerImageryProvider~requestNewTokenCallback} [options.requestNewToken] A callback to retrieve new tokens if
* its detected that the current token has expired or was not supplied.
* @param {TileDiscardPolicy} [options.tileDiscardPolicy] The policy that determines if a tile
* is invalid and should be discarded. If this value is not specified, a default
* {@link DiscardMissingTileImagePolicy} is used for tiled map servers, and a
@@ -124,6 +130,7 @@ define([

this._url = options.url;
this._token = options.token;
this._requestNewToken = options.requestNewToken;
this._tileDiscardPolicy = options.tileDiscardPolicy;
this._proxy = options.proxy;

@@ -225,19 +232,20 @@ define([
}

function requestMetadata() {
var parameters = {
f: 'json'
};
loadJsonHandleTokenErrors(that, function () {
var parameters = {
f: 'json'
};

if (defined(that._token)) {
parameters.token = that._token;
}
if (defined(that._token)) {
parameters.token = that._token;
}

var metadata = loadJsonp(that._url, {
parameters : parameters,
proxy : that._proxy
});
when(metadata, metadataSuccess, metadataFailure);
return loadJsonp(that._url, {
parameters : parameters,
proxy : that._proxy
});
}).then(metadataSuccess).otherwise(metadataFailure);
}

if (defined(options.mapServerData)) {
@@ -255,48 +263,6 @@ define([
}
}

function buildImageUrl(imageryProvider, x, y, level) {
var url;
if (imageryProvider._useTiles) {
url = imageryProvider._url + '/tile/' + level + '/' + y + '/' + x;
} else {
var nativeRectangle = imageryProvider._tilingScheme.tileXYToNativeRectangle(x, y, level);
var bbox = nativeRectangle.west + '%2C' + nativeRectangle.south + '%2C' + nativeRectangle.east + '%2C' + nativeRectangle.north;

url = imageryProvider._url + '/export?';
url += 'bbox=' + bbox;
if (imageryProvider._tilingScheme instanceof GeographicTilingScheme) {
url += '&bboxSR=4326&imageSR=4326';
} else {
url += '&bboxSR=3857&imageSR=3857';
}
url += '&size=' + imageryProvider._tileWidth + '%2C' + imageryProvider._tileHeight;
url += '&format=png&transparent=true&f=image';

if (imageryProvider.layers) {
url += '&layers=show:' + imageryProvider.layers;
}
}

var token = imageryProvider._token;
if (defined(token)) {
if (url.indexOf('?') === -1) {
url += '?';
}
if (url[url.length - 1] !== '?'){
url += '&';
}
url += 'token=' + token;
}

var proxy = imageryProvider._proxy;
if (defined(proxy)) {
url = proxy.getURL(url);
}

return url;
}

defineProperties(ArcGisMapServerImageryProvider.prototype, {
/**
* Gets the URL of the ArcGIS MapServer.
@@ -311,14 +277,16 @@ define([
},

/**
* Gets the ArcGIS token used to authenticate with the ArcGis MapServer service.
* The ArcGIS token used to authenticate with the ArcGis MapServer service.
* @memberof ArcGisMapServerImageryProvider.prototype
* @type {String}
* @readonly
*/
token : {
get : function() {
return this._token;
},
set : function(token) {
this._token = token;
}
},

@@ -558,7 +526,7 @@ define([
/**
* Gets the comma-separated list of layer IDs to show.
* @memberof ArcGisMapServerImageryProvider.prototype
*
*
* @type {String}
*/
layers : {
@@ -604,8 +572,38 @@ define([
}
//>>includeEnd('debug');

var that = this;
var tokenRetries = 1;
function loadImageWithToken (url) {
var loadPromise = loadImageViaBlob(url);
if (!defined(loadPromise)) {
return loadPromise;
}

return loadPromise.otherwise(function(requestErrorEvent) {
// If the token has expired or was not supplied the server sets the HTTP status code to 498/499 specifically to indicate these errors.
if (((requestErrorEvent.statusCode === 498) || (requestErrorEvent.statusCode === 499)) && (tokenRetries > 0)) {
tokenRetries--;

// Note: The token may have already been updated between the request and now (when the response is received),
// but for now we don't detect and optimize for this case and send off a new token request regardless.
return updateToken(that).then(function () {
// Rebuild the URL now that the token has been updated.
url = buildImageUrl(that, x, y, level);
return loadImageWithToken(url);
});
}

throw requestErrorEvent;
});
}

var url = buildImageUrl(this, x, y, level);
return ImageryProvider.loadImage(this, url);
if (!defined(this._requestNewToken)) {
return ImageryProvider.loadImage(this, url);
} else {
return throttleRequestByServer(url, loadImageWithToken);
}
};

/**
@@ -635,75 +633,189 @@ define([
return undefined;
}

var rectangle = this._tilingScheme.tileXYToNativeRectangle(x, y, level);
var that = this;
return loadJsonHandleTokenErrors(this, function () {
var url = buildPickURL(that, x, y, level, longitude, latitude);
return loadJson(url);
}).then(function(json) {
return jsonToFeatures(json);
});
};

function loadJsonHandleTokenErrors(item, loadJson) {
var tokenRetries = 1;
function loadJsonHandleError() {
return loadJson().then(function(json) {
// In this case if the token fails the server returns with a HTTP status code of 200 and encodes the error as JSON.
if (defined(json.error) && defined(json.error.code)) {
if (((json.error.code === 498) || (json.error.code === 499)) && defined(item._requestNewToken) && (tokenRetries > 0)) {
tokenRetries--;

// Note: The token may have already been updated between the request and now (when the response is received),
// but for now we don't detect and optimize for this case and send off a new token request regardless.
return updateToken(item).then(function () {
return loadJsonHandleError();
});
}
}

return json;
});
}

return loadJsonHandleError();
}

function buildImageUrl(imageryProvider, x, y, level) {
var url;
if (imageryProvider._useTiles) {
url = imageryProvider._url + '/tile/' + level + '/' + y + '/' + x;
} else {
var nativeRectangle = imageryProvider._tilingScheme.tileXYToNativeRectangle(x, y, level);
var bbox = nativeRectangle.west + '%2C' + nativeRectangle.south + '%2C' + nativeRectangle.east + '%2C' + nativeRectangle.north;

url = imageryProvider._url + '/export?';
url += 'bbox=' + bbox;
if (imageryProvider._tilingScheme instanceof GeographicTilingScheme) {
url += '&bboxSR=4326&imageSR=4326';
} else {
url += '&bboxSR=3857&imageSR=3857';
}
url += '&size=' + imageryProvider._tileWidth + '%2C' + imageryProvider._tileHeight;
url += '&format=png&transparent=true&f=image';

if (imageryProvider.layers) {
url += '&layers=show:' + imageryProvider.layers;
}
}

var token = imageryProvider._token;
if (defined(token)) {
if (url.indexOf('?') === -1) {
url += '?';
}
if (url[url.length - 1] !== '?'){
url += '&';
}
url += 'token=' + token;
}

var proxy = imageryProvider._proxy;
if (defined(proxy)) {
url = proxy.getURL(url);
}

return url;
}

function buildPickURL(imageryProvider, x, y, level, longitude, latitude) {
var rectangle = imageryProvider._tilingScheme.tileXYToNativeRectangle(x, y, level);

var horizontal;
var vertical;
var sr;
if (this._tilingScheme instanceof GeographicTilingScheme) {
if (imageryProvider._tilingScheme instanceof GeographicTilingScheme) {
horizontal = CesiumMath.toDegrees(longitude);
vertical = CesiumMath.toDegrees(latitude);
sr = '4326';
} else {
var projected = this._tilingScheme.projection.project(new Cartographic(longitude, latitude, 0.0));
var projected = imageryProvider._tilingScheme.projection.project(new Cartographic(longitude, latitude, 0.0));
horizontal = projected.x;
vertical = projected.y;
sr = '3857';
}

var url = this._url + '/identify?f=json&tolerance=2&geometryType=esriGeometryPoint';
var url = imageryProvider._url + '/identify?f=json&tolerance=2&geometryType=esriGeometryPoint';
url += '&geometry=' + horizontal + ',' + vertical;
url += '&mapExtent=' + rectangle.west + ',' + rectangle.south + ',' + rectangle.east + ',' + rectangle.north;
url += '&imageDisplay=' + this._tileWidth + ',' + this._tileHeight + ',96';
url += '&imageDisplay=' + imageryProvider._tileWidth + ',' + imageryProvider._tileHeight + ',96';
url += '&sr=' + sr;

url += '&layers=visible';
if (defined(this._layers)) {
url += ':' + this._layers;
if (defined(imageryProvider._layers)) {
url += ':' + imageryProvider._layers;
}

if (defined(this._token)) {
url += '&token=' + this._token;
if (defined(imageryProvider._token)) {
url += '&token=' + imageryProvider._token;
}

if (defined(this._proxy)) {
url = this._proxy.getURL(url);
if (defined(imageryProvider._proxy)) {
url = imageryProvider._proxy.getURL(url);
}

return loadJson(url).then(function(json) {
var result = [];
return url;
}

var features = json.results;
if (!defined(features)) {
return result;
}
function jsonToFeatures(json) {
var result = [];

for (var i = 0; i < features.length; ++i) {
var feature = features[i];

var featureInfo = new ImageryLayerFeatureInfo();
featureInfo.data = feature;
featureInfo.name = feature.value;
featureInfo.properties = feature.attributes;
featureInfo.configureDescriptionFromProperties(feature.attributes);

// If this is a point feature, use the coordinates of the point.
if (feature.geometryType === 'esriGeometryPoint' && feature.geometry) {
var wkid = feature.geometry.spatialReference && feature.geometry.spatialReference.wkid ? feature.geometry.spatialReference.wkid : 4326;
if (wkid === 4326 || wkid === 4283) {
featureInfo.position = Cartographic.fromDegrees(feature.geometry.x, feature.geometry.y, feature.geometry.z);
} else if (wkid === 102100 || wkid === 900913 || wkid === 3857) {
var projection = new WebMercatorProjection();
featureInfo.position = projection.unproject(new Cartesian3(feature.geometry.x, feature.geometry.y, feature.geometry.z));
}
}
var features = json.results;
if (!defined(features)) {
return result;
}

result.push(featureInfo);
for (var i = 0; i < features.length; ++i) {
var feature = features[i];

var featureInfo = new ImageryLayerFeatureInfo();
featureInfo.data = feature;
featureInfo.name = feature.value;
featureInfo.properties = feature.attributes;
featureInfo.configureDescriptionFromProperties(feature.attributes);

// If this is a point feature, use the coordinates of the point.
if (feature.geometryType === 'esriGeometryPoint' && feature.geometry) {
var wkid = feature.geometry.spatialReference && feature.geometry.spatialReference.wkid ? feature.geometry.spatialReference.wkid : 4326;
if (wkid === 4326 || wkid === 4283) {
featureInfo.position = Cartographic.fromDegrees(feature.geometry.x, feature.geometry.y, feature.geometry.z);
} else if (wkid === 102100 || wkid === 900913 || wkid === 3857) {
var projection = new WebMercatorProjection();
featureInfo.position = projection.unproject(new Cartesian3(feature.geometry.x, feature.geometry.y, feature.geometry.z));
}
}

return result;
});
};
result.push(featureInfo);
}

return result;
}

function updateToken(imageryProvider) {
if (!defined(imageryProvider._newTokenRequestInFlight) && defined(imageryProvider._requestNewToken)) {
// Due to the promise implementation used the function registered with .then() will be executed immediatly if the imageryProvider._requestNewToken()
// promise has already resolved when .then() is called. This flag allows us to make sure that ._newTokenRequestInFlight is defined correctly in both
// cases (where then runs immediately, when then runs deferred).
// Note: We explicitly set/test alreadyRun from both .then() and .otherwise() rather then using loadPromise.always() so that the order of execution is well
// defined (i.e. these operations will be run before any subsequently chained operations which might then call updateToken() and not want to get this result
// which has been resolved).
var alreadyRun = false;
var loadPromise = imageryProvider._requestNewToken().then(function(newToken) {
alreadyRun = true;
imageryProvider._newTokenRequestInFlight = undefined;

imageryProvider.token = newToken;
return newToken;
}).otherwise(function(requestErrorEvent) {
alreadyRun = true;
imageryProvider._newTokenRequestInFlight = undefined;

throw requestErrorEvent;
});

imageryProvider._newTokenRequestInFlight = alreadyRun ? undefined : loadPromise;
return loadPromise;
}

return imageryProvider._newTokenRequestInFlight;
}

return ArcGisMapServerImageryProvider;
});

/**
* A function that will make a request for a new token.
*
* @callback ArcGisMapServerImageryProvider~requestNewTokenCallback
* @return {Promise.<String>} A promise which will resolve to a new token.
*/