2016-04-26 21:59:41 +08:00
|
|
|
require('../../support/test_helper');
|
2016-03-09 03:06:43 +08:00
|
|
|
|
2016-04-26 21:59:41 +08:00
|
|
|
var assert = require('../../support/assert');
|
|
|
|
var TestClient = require('../../support/test-client');
|
2016-03-09 03:06:43 +08:00
|
|
|
|
2016-05-17 20:54:44 +08:00
|
|
|
function makeMapconfig(sql, cartocss) {
|
2016-03-09 03:06:43 +08:00
|
|
|
return {
|
|
|
|
"version": "1.4.0",
|
|
|
|
"layers": [
|
|
|
|
{
|
|
|
|
"type": 'mapnik',
|
|
|
|
"options": {
|
|
|
|
"cartocss_version": '2.3.0',
|
2016-05-17 20:54:44 +08:00
|
|
|
"sql": sql,
|
2016-03-09 03:06:43 +08:00
|
|
|
"cartocss": cartocss
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-04-26 21:59:41 +08:00
|
|
|
describe('turbo-carto regressions', function() {
|
2016-03-09 18:48:07 +08:00
|
|
|
|
|
|
|
afterEach(function (done) {
|
2016-05-17 20:54:44 +08:00
|
|
|
if (this.testClient) {
|
|
|
|
this.testClient.drain(done);
|
2016-06-20 19:09:01 +08:00
|
|
|
} else {
|
|
|
|
done();
|
2016-05-17 20:54:44 +08:00
|
|
|
}
|
2016-03-09 18:48:07 +08:00
|
|
|
});
|
2016-03-09 03:06:43 +08:00
|
|
|
|
|
|
|
it('should accept // comments', function(done) {
|
2016-05-17 20:54:44 +08:00
|
|
|
var cartocss = [
|
|
|
|
"/** simple visualization */",
|
|
|
|
"",
|
|
|
|
"Map {",
|
|
|
|
" buffer-size: 256;",
|
|
|
|
"}",
|
|
|
|
"",
|
|
|
|
"#county_points_with_population{",
|
|
|
|
" marker-fill-opacity: 0.1;",
|
|
|
|
" marker-line-color:#FFFFFF;//#CF1C90;",
|
|
|
|
" marker-line-width: 0;",
|
|
|
|
" marker-line-opacity: 0.3;",
|
|
|
|
" marker-placement: point;",
|
|
|
|
" marker-type: ellipse;",
|
|
|
|
" //marker-comp-op: overlay;",
|
|
|
|
" marker-width: [cartodb_id];",
|
|
|
|
" [zoom=5]{marker-width: [cartodb_id]*2;}",
|
|
|
|
" [zoom=6]{marker-width: [cartodb_id]*4;}",
|
|
|
|
" marker-fill: #000000;",
|
|
|
|
" marker-allow-overlap: true;",
|
|
|
|
" ",
|
|
|
|
"",
|
|
|
|
"}"
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
this.testClient = new TestClient(makeMapconfig('SELECT * FROM populated_places_simple_reduced', cartocss));
|
|
|
|
this.testClient.getLayergroup(function(err, layergroup) {
|
|
|
|
assert.ok(!err, err);
|
|
|
|
|
|
|
|
assert.ok(layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.ok(!layergroup.hasOwnProperty('errors'));
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-06-21 18:08:40 +08:00
|
|
|
it('should fail for private tables', function(done) {
|
|
|
|
var cartocss = [
|
|
|
|
"#private_table {",
|
|
|
|
" marker-placement: point;",
|
|
|
|
" marker-allow-overlap: true;",
|
|
|
|
" marker-line-width: 0;",
|
|
|
|
" marker-fill-opacity: 1.0;",
|
|
|
|
" marker-width: ramp([cartodb_id], 10, 20);",
|
|
|
|
" marker-fill: red;",
|
|
|
|
"}"
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
this.testClient = new TestClient(makeMapconfig('SELECT * FROM test_table_private_1', cartocss));
|
2017-12-05 19:09:31 +08:00
|
|
|
this.testClient.getLayergroup({ response: TestClient.RESPONSE.ERROR }, function(err, layergroup) {
|
2016-06-21 18:08:40 +08:00
|
|
|
assert.ok(!err, err);
|
|
|
|
|
|
|
|
assert.ok(!layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.ok(layergroup.hasOwnProperty('errors'));
|
|
|
|
|
|
|
|
var turboCartoError = layergroup.errors_with_context[0];
|
|
|
|
assert.ok(turboCartoError);
|
2016-06-27 00:43:04 +08:00
|
|
|
assert.equal(turboCartoError.type, 'layer');
|
2018-03-20 18:33:31 +08:00
|
|
|
assert.ok(turboCartoError.message.match(/permission\sdenied\sfor\s.+?test_table_private_1/));
|
2016-06-21 18:08:40 +08:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work for private tables with api key', function(done) {
|
|
|
|
var cartocss = [
|
|
|
|
"#private_table {",
|
|
|
|
" marker-placement: point;",
|
|
|
|
" marker-allow-overlap: true;",
|
|
|
|
" marker-line-width: 0;",
|
|
|
|
" marker-fill-opacity: 1.0;",
|
|
|
|
" marker-width: ramp([cartodb_id], 10, 20);",
|
|
|
|
" marker-fill: red;",
|
|
|
|
"}"
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
this.testClient = new TestClient(makeMapconfig('SELECT * FROM test_table_private_1', cartocss), 1234);
|
|
|
|
this.testClient.getLayergroup(function(err, layergroup) {
|
|
|
|
assert.ok(!err, err);
|
|
|
|
|
|
|
|
assert.ok(layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.ok(!layergroup.hasOwnProperty('errors'));
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-05-17 20:54:44 +08:00
|
|
|
it('should work with mapnik substitution tokens', function(done) {
|
|
|
|
var cartocss = [
|
|
|
|
"#layer {",
|
|
|
|
" line-width: 2;",
|
|
|
|
" line-color: #3B3B58;",
|
|
|
|
" line-opacity: 1;",
|
|
|
|
" polygon-opacity: 0.7;",
|
|
|
|
" polygon-fill: ramp([points_count], (#E5F5F9,#99D8C9,#2CA25F))",
|
|
|
|
"}"
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
var sql = [
|
|
|
|
'WITH hgrid AS (',
|
|
|
|
' SELECT CDB_HexagonGrid(',
|
|
|
|
' ST_Expand(!bbox!, greatest(!pixel_width!,!pixel_height!) * 100),',
|
|
|
|
' greatest(!pixel_width!,!pixel_height!) * 100',
|
|
|
|
' ) as cell',
|
|
|
|
')',
|
|
|
|
'SELECT',
|
|
|
|
' hgrid.cell as the_geom_webmercator,',
|
|
|
|
' count(1) as points_count,',
|
|
|
|
' count(1)/power(100 * CDB_XYZ_Resolution(CDB_ZoomFromScale(!scale_denominator!)), 2) as points_density,',
|
|
|
|
' 1 as cartodb_id',
|
|
|
|
'FROM hgrid, (SELECT * FROM populated_places_simple_reduced) i',
|
|
|
|
'where ST_Intersects(i.the_geom_webmercator, hgrid.cell)',
|
|
|
|
'GROUP BY hgrid.cell'
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
this.testClient = new TestClient(makeMapconfig(sql, cartocss));
|
2016-05-13 06:49:09 +08:00
|
|
|
this.testClient.getLayergroup(function(err, layergroup) {
|
2016-03-09 03:06:43 +08:00
|
|
|
assert.ok(!err, err);
|
2016-05-13 06:49:09 +08:00
|
|
|
|
|
|
|
assert.ok(layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.ok(!layergroup.hasOwnProperty('errors'));
|
|
|
|
|
2016-03-09 03:06:43 +08:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2016-05-19 17:42:28 +08:00
|
|
|
|
|
|
|
it('should work with mapnik substitution tokens and analyses', function(done) {
|
|
|
|
var cartocss = [
|
|
|
|
"#layer {",
|
|
|
|
" line-width: 2;",
|
|
|
|
" line-color: #3B3B58;",
|
|
|
|
" line-opacity: 1;",
|
|
|
|
" polygon-opacity: 0.7;",
|
|
|
|
" polygon-fill: ramp([points_count], (#E5F5F9,#99D8C9,#2CA25F))",
|
|
|
|
"}"
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
var sqlWrap = [
|
|
|
|
'WITH hgrid AS (',
|
|
|
|
' SELECT CDB_HexagonGrid(',
|
|
|
|
' ST_Expand(!bbox!, greatest(!pixel_width!,!pixel_height!) * 100),',
|
|
|
|
' greatest(!pixel_width!,!pixel_height!) * 100',
|
|
|
|
' ) as cell',
|
|
|
|
')',
|
|
|
|
'SELECT',
|
|
|
|
' hgrid.cell as the_geom_webmercator,',
|
|
|
|
' count(1) as points_count,',
|
|
|
|
' count(1)/power(100 * CDB_XYZ_Resolution(CDB_ZoomFromScale(!scale_denominator!)), 2) as points_density,',
|
|
|
|
' 1 as cartodb_id',
|
|
|
|
'FROM hgrid, (<%= sql %>) i',
|
|
|
|
'where ST_Intersects(i.the_geom_webmercator, hgrid.cell)',
|
|
|
|
'GROUP BY hgrid.cell'
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
var mapConfig = {
|
|
|
|
"version": "1.5.0",
|
|
|
|
"layers": [
|
|
|
|
{
|
|
|
|
"type": 'mapnik',
|
|
|
|
"options": {
|
|
|
|
"cartocss_version": '2.3.0',
|
|
|
|
"source": {
|
|
|
|
"id": "head"
|
|
|
|
},
|
|
|
|
sql_wrap: sqlWrap,
|
|
|
|
"cartocss": cartocss
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"analyses": [
|
|
|
|
{
|
|
|
|
"id": "head",
|
|
|
|
"type": "source",
|
|
|
|
"params": {
|
2016-05-19 18:13:37 +08:00
|
|
|
"query": "SELECT * FROM populated_places_simple_reduced"
|
2016-05-19 17:42:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
|
|
|
|
this.testClient = new TestClient(mapConfig, 1234);
|
|
|
|
this.testClient.getLayergroup(function(err, layergroup) {
|
|
|
|
assert.ok(!err, err);
|
|
|
|
|
|
|
|
assert.ok(layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.ok(!layergroup.hasOwnProperty('errors'));
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2016-06-20 19:09:01 +08:00
|
|
|
|
|
|
|
describe('empty datasource results', function() {
|
|
|
|
|
|
|
|
afterEach(function (done) {
|
|
|
|
if (this.testClient) {
|
|
|
|
this.testClient.drain(done);
|
|
|
|
} else {
|
|
|
|
done();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
function emptyResultMapConfig(markerFillRule) {
|
|
|
|
var cartocss = [
|
|
|
|
"#county_points_with_population {",
|
|
|
|
" marker-placement: point;",
|
|
|
|
" marker-allow-overlap: true;",
|
|
|
|
" marker-fill-opacity: 1.0;",
|
|
|
|
" marker-fill: " + markerFillRule + ';',
|
|
|
|
" marker-line-width: 0;",
|
|
|
|
"}"
|
|
|
|
].join('\n');
|
|
|
|
|
|
|
|
return {
|
|
|
|
"version": "1.5.0",
|
|
|
|
"layers": [
|
|
|
|
{
|
|
|
|
"type": 'mapnik',
|
|
|
|
"options": {
|
|
|
|
"cartocss_version": '2.3.0',
|
|
|
|
"source": {
|
|
|
|
"id": "head"
|
|
|
|
},
|
|
|
|
"cartocss": cartocss
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"analyses": [
|
|
|
|
{
|
|
|
|
"id": "head",
|
|
|
|
"type": "source",
|
|
|
|
"params": {
|
|
|
|
"query": "SELECT * FROM populated_places_simple_reduced limit 0"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-07-04 08:11:52 +08:00
|
|
|
var methods = ['quantiles', 'equal', 'jenks', 'headtails', 'category'];
|
2016-06-20 19:09:01 +08:00
|
|
|
|
2016-07-04 08:11:52 +08:00
|
|
|
methods.forEach(function(method) {
|
|
|
|
it('should work for "' + method+ '" method', function(done) {
|
|
|
|
var makerFillRule = 'ramp([pop_max], (#E5F5F9,#99D8C9,#2CA25F), ' + method + ')';
|
2016-06-20 19:09:01 +08:00
|
|
|
|
2016-07-04 08:11:52 +08:00
|
|
|
this.testClient = new TestClient(emptyResultMapConfig(makerFillRule), 1234);
|
|
|
|
this.testClient.getLayergroup(function(err, layergroup) {
|
|
|
|
assert.ok(!err, err);
|
2016-06-20 19:09:01 +08:00
|
|
|
|
2016-07-04 08:11:52 +08:00
|
|
|
assert.ok(layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.ok(!layergroup.hasOwnProperty('errors'));
|
2016-06-20 19:09:01 +08:00
|
|
|
|
2016-07-04 08:11:52 +08:00
|
|
|
done();
|
|
|
|
});
|
2016-06-20 19:09:01 +08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-07-18 17:57:27 +08:00
|
|
|
var scenarios = [
|
|
|
|
{
|
|
|
|
desc: 'numeric datasource',
|
|
|
|
cartocss: [
|
|
|
|
"#points {",
|
|
|
|
" marker-fill: ramp([scalerank], colorbrewer(Reds), category);",
|
|
|
|
"}"
|
|
|
|
].join('\n'),
|
|
|
|
expected: [
|
|
|
|
'#points {',
|
|
|
|
' marker-fill: #fee5d9;',
|
|
|
|
' [ scalerank = 6 ] {',
|
|
|
|
' marker-fill: #fcae91',
|
|
|
|
' }',
|
|
|
|
' [ scalerank = 8 ] {',
|
|
|
|
' marker-fill: #fb6a4a',
|
|
|
|
' }',
|
|
|
|
' [ scalerank = 4 ] {',
|
|
|
|
' marker-fill: #de2d26',
|
|
|
|
' }',
|
|
|
|
' [ scalerank = 10 ] {',
|
|
|
|
' marker-fill: #a50f15',
|
|
|
|
' }',
|
|
|
|
'}',
|
|
|
|
].join('\n')
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: 'string datasource',
|
|
|
|
cartocss: [
|
|
|
|
"#points {",
|
|
|
|
" marker-fill: ramp([adm0name], colorbrewer(Reds), category);",
|
|
|
|
"}"
|
|
|
|
].join('\n'),
|
|
|
|
expected: [
|
|
|
|
'#points {',
|
|
|
|
' marker-fill: #fee5d9;',
|
|
|
|
' [ adm0name = "Russia" ] {',
|
|
|
|
' marker-fill: #fcae91',
|
|
|
|
' }',
|
|
|
|
' [ adm0name = "China" ] {',
|
|
|
|
' marker-fill: #fb6a4a',
|
|
|
|
' }',
|
|
|
|
' [ adm0name = "Brazil" ] {',
|
|
|
|
' marker-fill: #de2d26',
|
|
|
|
' }',
|
|
|
|
' [ adm0name = "Canada" ] {',
|
|
|
|
' marker-fill: #a50f15',
|
|
|
|
' }',
|
|
|
|
'}',
|
|
|
|
].join('\n')
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: 'numeric manual',
|
|
|
|
cartocss: [
|
|
|
|
"#points {",
|
|
|
|
" marker-fill: ramp([scalerank], colorbrewer(Reds), (-1, 6, 8, 4, 10), category);",
|
|
|
|
"}"
|
|
|
|
].join('\n'),
|
|
|
|
expected: [
|
|
|
|
'#points {',
|
|
|
|
' marker-fill: #fee5d9;',
|
|
|
|
' [ scalerank = 6 ] {',
|
|
|
|
' marker-fill: #fcae91',
|
|
|
|
' }',
|
|
|
|
' [ scalerank = 8 ] {',
|
|
|
|
' marker-fill: #fb6a4a',
|
|
|
|
' }',
|
|
|
|
' [ scalerank = 4 ] {',
|
|
|
|
' marker-fill: #de2d26',
|
|
|
|
' }',
|
|
|
|
' [ scalerank = 10 ] {',
|
|
|
|
' marker-fill: #a50f15',
|
|
|
|
' }',
|
|
|
|
'}',
|
|
|
|
].join('\n')
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
|
|
|
scenarios.forEach(function(scenario) {
|
|
|
|
it('category ramps should use original type: ' + scenario.desc, function(done) {
|
|
|
|
var mapConfig = makeMapconfig('SELECT * FROM populated_places_simple_reduced', scenario.cartocss);
|
|
|
|
this.testClient = new TestClient(mapConfig);
|
|
|
|
this.testClient.getLayergroup(function(err, layergroup) {
|
|
|
|
assert.ok(!err, err);
|
|
|
|
|
|
|
|
assert.ok(layergroup.hasOwnProperty('layergroupid'));
|
|
|
|
assert.deepEqual(layergroup.metadata.layers[0].meta.cartocss, scenario.expected);
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-06-29 21:03:45 +08:00
|
|
|
describe.only('Buckets calculation', function () {
|
|
|
|
afterEach(function (done) {
|
|
|
|
if (this.testClient) {
|
|
|
|
this.testClient.drain(done);
|
|
|
|
} else {
|
|
|
|
done();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const scenarios = [
|
|
|
|
{
|
|
|
|
numBuckets: 1,
|
|
|
|
bucketResponse: [
|
|
|
|
{
|
|
|
|
filter: {
|
|
|
|
type: 'range',
|
|
|
|
start: 0,
|
|
|
|
end: 8
|
|
|
|
},
|
|
|
|
value: 1
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
numBuckets: 2,
|
|
|
|
bucketResponse: [
|
|
|
|
{
|
|
|
|
filter: {
|
|
|
|
type: 'range',
|
|
|
|
start: 0,
|
|
|
|
end: 3
|
|
|
|
},
|
|
|
|
value: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
filter: {
|
|
|
|
type: 'range',
|
|
|
|
start: 3,
|
|
|
|
end: 8
|
|
|
|
},
|
|
|
|
value: 20
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
numBuckets: 3,
|
|
|
|
bucketResponse: [
|
|
|
|
{
|
|
|
|
filter: {
|
|
|
|
type: 'range',
|
|
|
|
start: 0,
|
|
|
|
end: 2
|
|
|
|
},
|
|
|
|
value: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
filter: {
|
|
|
|
type: 'range',
|
|
|
|
start: 2,
|
|
|
|
end: 5
|
|
|
|
},
|
|
|
|
value: 10.5
|
|
|
|
},
|
|
|
|
{
|
|
|
|
filter: {
|
|
|
|
type: 'range',
|
|
|
|
start: 5,
|
|
|
|
end: 8
|
|
|
|
},
|
|
|
|
value: 20
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
scenarios.forEach(function (scenario) {
|
|
|
|
it('Buckets: ' + scenario.numBuckets, function (done) {
|
|
|
|
const bucketsMapConfig = makeMapconfig({ numQuantiles: scenario.numBuckets });
|
|
|
|
|
|
|
|
this.testClient = new TestClient(bucketsMapConfig);
|
|
|
|
this.testClient.getLayergroup({ response: OK_RESPONSE }, function (err, layergroup) {
|
|
|
|
const rule = layergroup.metadata.layers[0].meta.cartocss_meta.rules[0];
|
|
|
|
|
|
|
|
assert.ok(!err, err);
|
|
|
|
assert.equal(rule.buckets.length, scenario.numBuckets);
|
|
|
|
assert.deepEqual(rule.buckets, scenario.bucketResponse);
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
function makeMapconfig({numQuantiles = 1}) {
|
|
|
|
return {
|
|
|
|
"version": "1.4.0",
|
|
|
|
"layers": [
|
|
|
|
{
|
|
|
|
"type": 'mapnik',
|
|
|
|
"options": {
|
|
|
|
"cartocss_version": '2.3.0',
|
|
|
|
"sql": 'SELECT * FROM populated_places_simple_reduced',
|
|
|
|
"cartocss": `#layer {\n
|
|
|
|
marker-width: ramp([labelrank], range(1, 20), quantiles(${numQuantiles}));\n
|
|
|
|
marker-fill: #EE4D5A;\n marker-fill-opacity: 0.9;\n
|
|
|
|
marker-allow-overlap: true;\n marker-line-width: 1;\n
|
|
|
|
marker-line-color: #FFFFFF;\n marker-line-opacity: 1;\n}`,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const OK_RESPONSE = {
|
|
|
|
status: 200,
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json; charset=utf-8'
|
|
|
|
}
|
|
|
|
};
|
|
|
|
});
|
2016-03-09 03:06:43 +08:00
|
|
|
});
|