Windows: mark downloads as "from the internet"

This commit is contained in:
Evan Hahn 2020-09-11 16:17:07 -05:00 committed by Josh Perez
parent 50378ed9bc
commit 6d2e994f9f
5 changed files with 152 additions and 0 deletions

View file

@ -12,6 +12,10 @@ const normalizePath = require('normalize-path');
const sanitizeFilename = require('sanitize-filename');
const getGuid = require('uuid/v4');
const { isPathInside } = require('../ts/util/isPathInside');
const { isWindows } = require('../ts/OS');
const {
writeWindowsZoneIdentifier,
} = require('../ts/util/windowsZoneIdentifier');
let xattr;
try {
@ -229,6 +233,13 @@ async function writeWithAttributes(target, data) {
const attrValue = `${type};${timestamp};${appName};${guid}`;
await xattr.set(target, 'com.apple.quarantine', attrValue);
} else if (isWindows()) {
// This operation may fail (see the function's comments), which is not a show-stopper.
try {
await writeWindowsZoneIdentifier(target);
} catch (err) {
console.warn('Failed to write Windows Zone.Identifier file; continuing');
}
}
}

14
ts/test/helpers.ts Normal file
View file

@ -0,0 +1,14 @@
import { assert } from 'chai';
export async function assertRejects(fn: () => Promise<unknown>): Promise<void> {
let err: unknown;
try {
await fn();
} catch (e) {
err = e;
}
assert(
err instanceof Error,
'Expected promise to reject with an Error, but it resolved'
);
}

View file

@ -0,0 +1,64 @@
import { assert } from 'chai';
import * as path from 'path';
import * as os from 'os';
import * as fs from 'fs';
import * as fse from 'fs-extra';
import * as Sinon from 'sinon';
import { assertRejects } from '../helpers';
import { writeWindowsZoneIdentifier } from '../../util/windowsZoneIdentifier';
describe('writeWindowsZoneIdentifier', () => {
before(function thisNeeded() {
if (process.platform !== 'win32') {
this.skip();
}
});
beforeEach(async function thisNeeded() {
this.sandbox = Sinon.createSandbox();
this.tmpdir = await fs.promises.mkdtemp(
path.join(os.tmpdir(), 'signal-test-')
);
});
afterEach(async function thisNeeded() {
this.sandbox.restore();
await fse.remove(this.tmpdir);
});
it('writes zone transfer ID 3 (internet) to the Zone.Identifier file', async function thisNeeded() {
const file = path.join(this.tmpdir, 'file.txt');
await fse.outputFile(file, 'hello');
await writeWindowsZoneIdentifier(file);
assert.strictEqual(
await fs.promises.readFile(`${file}:Zone.Identifier`, 'utf8'),
'[ZoneTransfer]\r\nZoneId=3'
);
});
it('fails if there is an existing Zone.Identifier file', async function thisNeeded() {
const file = path.join(this.tmpdir, 'file.txt');
await fse.outputFile(file, 'hello');
await fs.promises.writeFile(`${file}:Zone.Identifier`, '# already here');
await assertRejects(() => writeWindowsZoneIdentifier(file));
});
it('fails if the original file does not exist', async function thisNeeded() {
const file = path.join(this.tmpdir, 'file-never-created.txt');
await assertRejects(() => writeWindowsZoneIdentifier(file));
});
it('fails if not on Windows', async function thisNeeded() {
this.sandbox.stub(process, 'platform').get(() => 'darwin');
const file = path.join(this.tmpdir, 'file.txt');
await fse.outputFile(file, 'hello');
await assertRejects(() => writeWindowsZoneIdentifier(file));
});
});

View file

@ -13151,6 +13151,24 @@
"reasonCategory": "falseMatch",
"updated": "2020-02-07T19:52:28.522Z"
},
{
"rule": "jQuery-before(",
"path": "ts/test/util/windowsZoneIdentifier_test.js",
"line": " before(function thisNeeded() {",
"lineNumber": 19,
"reasonCategory": "testCode",
"updated": "2020-09-02T18:59:59.432Z",
"reasonDetail": "This is test code (and isn't jQuery code)."
},
{
"rule": "jQuery-before(",
"path": "ts/test/util/windowsZoneIdentifier_test.ts",
"line": " before(function thisNeeded() {",
"lineNumber": 12,
"reasonCategory": "testCode",
"updated": "2020-09-02T18:59:59.432Z",
"reasonDetail": "This is test code (and isn't jQuery code)."
},
{
"rule": "jQuery-append(",
"path": "ts/textsecure/ContactsParser.js",

View file

@ -0,0 +1,45 @@
import * as fs from 'fs';
import { isWindows } from '../OS';
const ZONE_IDENTIFIER_CONTENTS = Buffer.from('[ZoneTransfer]\r\nZoneId=3');
/**
* Internet Explorer introduced the concept of "Security Zones". For our purposes, we
* just need to set the security zone to the "Internet" zone, which Windows will use to
* offer some protections. This is customizable by the user (or, more likely, by IT).
*
* To do this, we write the "Zone.Identifier" for the NTFS alternative data stream.
*
* This can fail in a bunch of sitations:
*
* - The OS is not Windows.
* - The filesystem is not NTFS.
* - Writing the metadata file fails for some reason (permissions, for example).
* - The metadata file already exists. (We could choose to overwrite it.)
* - The original file is deleted between the time that we check for its existence and
* when we write the metadata. This is a rare race condition, but is possible.
*
* Consumers of this module should probably tolerate failures.
*/
export async function writeWindowsZoneIdentifier(
filePath: string
): Promise<void> {
if (!isWindows()) {
throw new Error('writeWindowsZoneIdentifier should only run on Windows');
}
// tslint:disable-next-line non-literal-fs-path
if (!fs.existsSync(filePath)) {
throw new Error(
'writeWindowsZoneIdentifier could not find the original file'
);
}
await fs.promises.writeFile(
`${filePath}:Zone.Identifier`,
ZONE_IDENTIFIER_CONTENTS,
{
flag: 'wx',
}
);
}