Windows compatibility #19

Hide WebP format and normalise option

Separate test runners for node and iojs
This commit is contained in:
Lovell Fuller 2015-04-21 12:13:19 +01:00
parent 8926ebc56c
commit 1e52c2dbe6
13 changed files with 231 additions and 206 deletions

View File

@ -88,9 +88,12 @@ The _gettext_ dependency of _libvips_ [can lead](https://github.com/lovell/sharp
Requires x86 32-bit Node.js or io.js (use `iojs.exe` rather than `node.exe`). Requires x86 32-bit Node.js or io.js (use `iojs.exe` rather than `node.exe`).
The WebP format is currently unsupported. The WebP format is currently unsupported.
1. [Download](http://www.vips.ecs.soton.ac.uk/supported/current/win32/) and unzip `vips-dev.x.y.z.zip`. 1. Ensure the [node-gyp prerequisites](https://github.com/TooTallNate/node-gyp#installation) are met.
2. Set the `VIPS_HOME` environment variable to the full path of the `vips-dev-x.y.z` directory. 2. [Download](http://www.vips.ecs.soton.ac.uk/supported/current/win32/) and unzip `vips-dev.x.y.z.zip`.
3. Add `vips-dev-x.y.z\bin` to `PATH`. 3. Set the `VIPS_HOME` environment variable to the full path of the `vips-dev-x.y.z` directory.
4. Add `vips-dev-x.y.z\bin` to `PATH`.
Versions of MSVC more recent than 2013 may require the use of `npm install --arch=ia32 --msvs_version=2013`.
### Heroku ### Heroku
@ -678,6 +681,10 @@ A [guide for contributors](https://github.com/lovell/sharp/blob/master/CONTRIBUT
[![Centos 6.5 Build Status](https://snap-ci.com/lovell/sharp/branch/master/build_image)](https://snap-ci.com/lovell/sharp/branch/master) [![Centos 6.5 Build Status](https://snap-ci.com/lovell/sharp/branch/master/build_image)](https://snap-ci.com/lovell/sharp/branch/master)
#### Windows Server 2012
[![Windows Server 2012 Build Status](https://ci.appveyor.com/api/projects/status/pgtul704nkhhg6sg)](https://ci.appveyor.com/project/lovell/sharp)
### Benchmark tests ### Benchmark tests
``` ```

View File

@ -1,17 +1,17 @@
os: Visual Studio 2014 CTP4 os: Visual Studio 2014 CTP4
platform: x86
environment: environment:
VIPS_VERSION_MAJOR_MINOR: 7.42 VIPS_VERSION_MAJOR_MINOR: 7.42
VIPS_VERSION_PATCH: 3 VIPS_VERSION_PATCH: 3
VIPS_WARNING: 0
install: install:
- ps: $env:VIPS_VERSION = "$env:VIPS_VERSION_MAJOR_MINOR.$env:VIPS_VERSION_PATCH" - ps: $env:VIPS_VERSION = "$env:VIPS_VERSION_MAJOR_MINOR.$env:VIPS_VERSION_PATCH"
- ps: Write-Output "VIPS_VERSION=$env:VIPS_VERSION"
- ps: Write-Output "Fetching http://www.vips.ecs.soton.ac.uk/supported/$env:VIPS_VERSION_MAJOR_MINOR/win32/vips-dev-$env:VIPS_VERSION.zip" - ps: Write-Output "Fetching http://www.vips.ecs.soton.ac.uk/supported/$env:VIPS_VERSION_MAJOR_MINOR/win32/vips-dev-$env:VIPS_VERSION.zip"
- ps: Start-FileDownload http://www.vips.ecs.soton.ac.uk/supported/$env:VIPS_VERSION_MAJOR_MINOR/win32/vips-dev-$env:VIPS_VERSION.zip -FileName c:\vips-dev-$env:VIPS_VERSION.zip - ps: Start-FileDownload http://www.vips.ecs.soton.ac.uk/supported/$env:VIPS_VERSION_MAJOR_MINOR/win32/vips-dev-$env:VIPS_VERSION.zip -FileName c:\vips-dev-$env:VIPS_VERSION.zip
- ps: Write-Output "Extracting c:\vips-dev-$env:VIPS_VERSION.zip"
- ps: Invoke-Expression "& 7z -y x c:\vips-dev-$env:VIPS_VERSION.zip -oc:\ | FIND /V `"ing `"" - ps: Invoke-Expression "& 7z -y x c:\vips-dev-$env:VIPS_VERSION.zip -oc:\ | FIND /V `"ing `""
- ps: $env:VIPS_HOME = "c:\vips-dev-$env:VIPS_VERSION" - ps: $env:VIPS_HOME = "c:\vips-dev-$env:VIPS_VERSION"
- ps: $env:PATH = "$env:VIPS_HOME\bin;$env:PATH" - ps: $env:PATH = "$env:VIPS_HOME\bin;$env:PATH"
- ps: Install-Product node 0 x86 - ps: Install-Product node 0 x86
- npm install --msvs_version=2013 - npm install --msvs_version=2013
test_script: test_script:
- npm test - npm run-script test-win32-node

View File

@ -349,10 +349,14 @@ Sharp.prototype.gamma = function(gamma) {
}; };
/* /*
Normalize histogram Enhance output image contrast by stretching its luminance to cover the full dynamic range
*/ */
Sharp.prototype.normalize = function(normalize) { Sharp.prototype.normalize = function(normalize) {
if (process.platform !== 'win32') {
this.options.normalize = (typeof normalize === 'boolean') ? normalize : true; this.options.normalize = (typeof normalize === 'boolean') ? normalize : true;
} else {
console.error('normalize unavailable on win32 platform');
}
return this; return this;
}; };
Sharp.prototype.normalise = Sharp.prototype.normalize; Sharp.prototype.normalise = Sharp.prototype.normalize;

View File

@ -20,7 +20,9 @@
], ],
"description": "High performance Node.js module to resize JPEG, PNG, WebP and TIFF images using the libvips library", "description": "High performance Node.js module to resize JPEG, PNG, WebP and TIFF images using the libvips library",
"scripts": { "scripts": {
"test": "VIPS_WARNING=0 node ./node_modules/istanbul/lib/cli.js cover ./node_modules/mocha/bin/_mocha -- --slow=5000 --timeout=10000 ./test/unit/*.js" "test": "VIPS_WARNING=0 node ./node_modules/istanbul/lib/cli.js cover ./node_modules/mocha/bin/_mocha -- --slow=5000 --timeout=15000 ./test/unit/*.js",
"test-win32-node": "node ./node_modules/mocha/bin/mocha --slow=5000 --timeout=15000 ./test/unit/*.js",
"test-win32-iojs": "iojs ./node_modules/mocha/bin/mocha --slow=5000 --timeout=15000 ./test/unit/*.js"
}, },
"main": "index.js", "main": "index.js",
"repository": { "repository": {

View File

@ -708,6 +708,7 @@ class ResizeWorker : public NanAsyncWorker {
image = gammaDecoded; image = gammaDecoded;
} }
#ifndef _WIN32
// Apply normalization // Apply normalization
if (baton->normalize) { if (baton->normalize) {
VipsInterpretation typeBeforeNormalize = image->Type; VipsInterpretation typeBeforeNormalize = image->Type;
@ -787,6 +788,7 @@ class ResizeWorker : public NanAsyncWorker {
image = normalized; image = normalized;
} }
} }
#endif
// Convert image to sRGB, if not already // Convert image to sRGB, if not already
if (image->Type != VIPS_INTERPRETATION_sRGB) { if (image->Type != VIPS_INTERPRETATION_sRGB) {

View File

@ -8,6 +8,10 @@
#include "resize.h" #include "resize.h"
#include "utilities.h" #include "utilities.h"
#ifdef _WIN64
#error Windows 64-bit currently unsupported - see https://github.com/lovell/sharp#windows
#endif
extern "C" void init(v8::Handle<v8::Object> target) { extern "C" void init(v8::Handle<v8::Object> target) {
NanScope(); NanScope();
vips_init("sharp"); vips_init("sharp");

View File

@ -31,6 +31,7 @@ describe('Colour space conversion', function() {
.toFile(fixtures.path('output.greyscale-not.jpg'), done); .toFile(fixtures.path('output.greyscale-not.jpg'), done);
}); });
if (sharp.format.webp.output.buffer) {
it('From 1-bit TIFF to sRGB WebP [slow]', function(done) { it('From 1-bit TIFF to sRGB WebP [slow]', function(done) {
sharp(fixtures.inputTiff) sharp(fixtures.inputTiff)
.webp() .webp()
@ -41,6 +42,7 @@ describe('Colour space conversion', function() {
done(); done();
}); });
}); });
}
it('From CMYK to sRGB', function(done) { it('From CMYK to sRGB', function(done) {
sharp(fixtures.inputJpgWithCmykProfile) sharp(fixtures.inputJpgWithCmykProfile)

View File

@ -8,6 +8,8 @@ var cpplint = require('node-cpplint/lib/');
describe('cpplint', function() { describe('cpplint', function() {
// Ignore cpplint failures, possibly newline-related, on Windows
if (process.platform !== 'win32') {
// List C++ source files // List C++ source files
fs.readdirSync(path.join(__dirname, '..', '..', 'src')).forEach(function (source) { fs.readdirSync(path.join(__dirname, '..', '..', 'src')).forEach(function (source) {
var file = path.join('src', source); var file = path.join('src', source);
@ -42,5 +44,6 @@ describe('cpplint', function() {
}); });
}); });
}
}); });

View File

@ -28,6 +28,7 @@ describe('Embed', function() {
}); });
}); });
if (sharp.format.webp.output.buffer) {
it('JPEG within WebP, to include alpha channel', function(done) { it('JPEG within WebP, to include alpha channel', function(done) {
sharp(fixtures.inputJpg) sharp(fixtures.inputJpg)
.resize(320, 240) .resize(320, 240)
@ -47,5 +48,6 @@ describe('Embed', function() {
}); });
}); });
}); });
}
}); });

View File

@ -31,6 +31,7 @@ describe('Partial image extraction', function() {
}); });
}); });
if (sharp.format.webp.output.file) {
it('WebP', function(done) { it('WebP', function(done) {
sharp(fixtures.inputWebP) sharp(fixtures.inputWebP)
.extract(50, 100, 125, 200) .extract(50, 100, 125, 200)
@ -41,6 +42,7 @@ describe('Partial image extraction', function() {
done(); done();
}); });
}); });
}
it('TIFF', function(done) { it('TIFF', function(done) {
sharp(fixtures.inputTiff) sharp(fixtures.inputTiff)

View File

@ -806,9 +806,11 @@ describe('Input/output', function() {
.toBuffer(function(err) { .toBuffer(function(err) {
sharp.queue.removeListener('change', queueListener); sharp.queue.removeListener('change', queueListener);
if (err) throw err; if (err) throw err;
process.nextTick(function() {
assert.strictEqual(2, eventCounter); assert.strictEqual(2, eventCounter);
done(); done();
}); });
}); });
});
}); });

View File

@ -82,6 +82,7 @@ describe('Image metadata', function() {
}); });
}); });
if (sharp.format.webp.input.file) {
it('WebP', function(done) { it('WebP', function(done) {
sharp(fixtures.inputWebP).metadata(function(err, metadata) { sharp(fixtures.inputWebP).metadata(function(err, metadata) {
if (err) throw err; if (err) throw err;
@ -95,6 +96,7 @@ describe('Image metadata', function() {
done(); done();
}); });
}); });
}
it('GIF via libmagick', function(done) { it('GIF via libmagick', function(done) {
sharp(fixtures.inputGif).metadata(function(err, metadata) { sharp(fixtures.inputGif).metadata(function(err, metadata) {

View File

@ -13,6 +13,9 @@ describe('Normalization', function () {
assert.strictEqual(sharp.prototype.normalize, sharp.prototype.normalise); assert.strictEqual(sharp.prototype.normalize, sharp.prototype.normalise);
}); });
// Normalize is currently unavailable on Windows
if (process.platform !== 'win32') {
it('spreads rgb image values between 0 and 255', function(done) { it('spreads rgb image values between 0 and 255', function(done) {
sharp(fixtures.inputJpgWithLowContrast) sharp(fixtures.inputJpgWithLowContrast)
.normalize() .normalize()
@ -54,7 +57,6 @@ describe('Normalization', function () {
.normalize() .normalize()
.raw() .raw()
.toBuffer(function (err, data, info) { .toBuffer(function (err, data, info) {
// raw toBuffer does not return the alpha channel (yet?)
var min = 255, max = 0, i; var min = 255, max = 0, i;
for (i = 0; i < data.length; i++) { for (i = 0; i < data.length; i++) {
min = Math.min(min, data[i]); min = Math.min(min, data[i]);
@ -89,11 +91,6 @@ describe('Normalization', function () {
.metadata() .metadata()
.then(function (metadata) { .then(function (metadata) {
assert.strictEqual(true, metadata.hasAlpha); assert.strictEqual(true, metadata.hasAlpha);
// because of complications with greyscale
// we return everything in srgb for now.
//
// assert.strictEqual(2, metadata.channels);
// assert.strictEqual('b-w', metadata.space);
assert.strictEqual(4, metadata.channels); assert.strictEqual(4, metadata.channels);
assert.strictEqual('srgb', metadata.space); assert.strictEqual('srgb', metadata.space);
}) })
@ -113,11 +110,6 @@ describe('Normalization', function () {
}) })
.then(function (metadata) { .then(function (metadata) {
assert.strictEqual(false, metadata.hasAlpha); assert.strictEqual(false, metadata.hasAlpha);
// because of complications with greyscale
// we return everything in srgb for now.
//
// assert.strictEqual(1, metadata.channels);
// assert.strictEqual('b-w', metadata.space);
assert.strictEqual(3, metadata.channels); assert.strictEqual(3, metadata.channels);
assert.strictEqual('srgb', metadata.space); assert.strictEqual('srgb', metadata.space);
}) })
@ -127,10 +119,11 @@ describe('Normalization', function () {
.toBuffer(); .toBuffer();
}) })
.then(function (rawData) { .then(function (rawData) {
// var blackBuffer = new Buffer([0,0,0,0]);
var blackBuffer = new Buffer([0,0,0, 0,0,0, 0,0,0, 0,0,0]); var blackBuffer = new Buffer([0,0,0, 0,0,0, 0,0,0, 0,0,0]);
assert.strictEqual(blackBuffer.toString(), rawData.toString()); assert.strictEqual(blackBuffer.toString(), rawData.toString());
}) })
.finally(done); .finally(done);
}); });
}
}); });