Skip to content

Commit

Permalink
Add convertToBase64 method
Browse files Browse the repository at this point in the history
  • Loading branch information
mhegazy committed Apr 8, 2015
1 parent 863f0b6 commit 73e22ed
Show file tree
Hide file tree
Showing 3 changed files with 112 additions and 1 deletion.
3 changes: 2 additions & 1 deletion Jakefile
Original file line number Diff line number Diff line change
Expand Up @@ -145,7 +145,8 @@ var harnessSources = [
"services/documentRegistry.ts",
"services/preProcessFile.ts",
"services/patternMatcher.ts",
"versionCache.ts"
"versionCache.ts",
"convertToBase64.ts"
].map(function (f) {
return path.join(unittestsDirectory, f);
})).concat([
Expand Down
77 changes: 77 additions & 0 deletions src/compiler/utilities.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1815,4 +1815,81 @@ module ts {
export function getLocalSymbolForExportDefault(symbol: Symbol) {
return symbol && symbol.valueDeclaration && (symbol.valueDeclaration.flags & NodeFlags.Default) ? symbol.valueDeclaration.localSymbol : undefined;
}

/**
* Replace each instance of non-ascii characters by one, two, three, or four escape sequences
* representing the UTF-8 encoding of the character, and return the expanded char code list.
*/
function getExpandedCharCodes(input: string): number[] {
let output: number[] = [];
let length = input.length;
let leadSurrogate: number = undefined;

for (let i = 0; i < length; i++) {
let charCode = input.charCodeAt(i);

// handel utf8
if (charCode < 0x80) {
output.push(charCode);
}
else if (charCode < 0x800) {
output.push((charCode >> 6) | 0B11000000);
output.push((charCode & 0B00111111) | 0B10000000);
}
else if (charCode < 0x10000) {
output.push((charCode >> 12) | 0B11100000);
output.push(((charCode >> 6) & 0B00111111) | 0B10000000);
output.push((charCode & 0B00111111) | 0B10000000);
}
else if (charCode < 0x20000) {
output.push((charCode >> 18) | 0B11110000);
output.push(((charCode >> 12) & 0B00111111) | 0B10000000);
output.push(((charCode >> 6) & 0B00111111) | 0B10000000);
output.push((charCode & 0B00111111) | 0B10000000);
}
else {
Debug.assert(false, "Unexpected code point");
}
}

return output;
}

const base64Digits = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";

/**
* Converts a string to a base-64 encoded ASCII string.
*/
export function convertToBase64(input: string): string {

This comment has been minimized.

Copy link
@sheetalkamat

sheetalkamat Apr 22, 2015

Member

don't we have similar one for converting offsets into base64 in emitter.ts

This comment has been minimized.

Copy link
@mhegazy

mhegazy Apr 22, 2015

Author Contributor

this is different. this is text -> normal base64, the one in the emitter is number->base64VLQ which takes care of sign.. etc.

var result = "";
let charCodes = getExpandedCharCodes(input);
let i = 0;
let length = charCodes.length;
let byte1: number, byte2: number, byte3: number, byte4: number;

while (i < length) {
// Convert every 6-bits in the input 3 character points
// into a base64 digit
byte1 = charCodes[i] >> 2;
byte2 = (charCodes[i] & 0B00000011) << 4 | charCodes[i + 1] >> 4;
byte3 = (charCodes[i + 1] & 0B00001111) << 2 | charCodes[i + 2] >> 6;
byte4 = charCodes[i + 2] & 0B00111111;

// We are out of characters in the input, set the extra
// digits to 64 (padding character).
if (i + 1 >= length) {
byte3 = byte4 = 64;
}
else if (i + 2 >= length) {
byte4 = 64;
}

// Write to the ouput
result += base64Digits.charAt(byte1) + base64Digits.charAt(byte2) + base64Digits.charAt(byte3) + base64Digits.charAt(byte4);

i += 3;
}

return result;
}
}
33 changes: 33 additions & 0 deletions tests/cases/unittests/convertToBase64.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
/// <reference path="..\..\..\src\harness\harness.ts" />

module ts {
describe('convertToBase64', () => {
function runTest(input: string): void {
var actual = ts.convertToBase64(input);
var expected = new Buffer(input).toString("base64");
assert.equal(actual, expected, "Encoded string using convertToBase64 does not match buffer.toString('base64')");
}

it("Converts ASCII charaters correctelly", () => {
runTest(" !\"#$ %&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~");
});

it("Converts escape sequences correctelly", () => {
runTest("\t\n\r\\\"\'\u0062");
});

it("Converts simple unicode characters correctelly", () => {
runTest("ΠΣ ٵپ औठ ⺐⺠");
});

it("Converts simple code snippit correctelly", () => {
runTest(`/// <reference path="file.ts" />
var x: string = "string";
console.log(x);`);
});

it("Converts simple code snippit with unicode characters correctelly", () => {
runTest(`var Π = 3.1415; console.log(Π);`);
});
});
}

0 comments on commit 73e22ed

Please sign in to comment.