mirror of
https://github.com/trafficlunar/tomodachi-share.git
synced 2026-03-28 11:13:16 +00:00
126 lines
6.7 KiB
TypeScript
126 lines
6.7 KiB
TypeScript
import { convertQrCode } from "./qr-codes";
|
|
import { describe, it, expect } from "vitest";
|
|
import Mii from "./mii.js/mii";
|
|
import { TomodachiLifeMii, HairDyeMode } from "./tomodachi-life-mii";
|
|
|
|
// List of encrypted QR code data to test against.
|
|
const validQrCodes = [
|
|
// Frieren (from step6.webp)
|
|
{
|
|
base64: "lymEx6TA4eLgfwTEceW+DbdYBTfBfPKM4VesJkq1qzoGGXnk/3OohvPPuGS8mmnzvpvyC36ge8+C2c4IIhJ0l7Lx9KKxnPEAuLBM1YtCnSowjVNHo3CmjE7D7lkDUBuhO3qKjAqWXfQthtqBqjTe4Hv95TKNVPimaxNXhAVZGSmOMh++0Z2N0TvIDpU/8kxLIsgntsQH4PNlaFcVF2HeOERXRdTm/TMFb6pozO57nJ9NKi5bxh1ClNArbpyTQgBe7cfvnNretFVNWzGJBWctZC7weecYIKyU3qbH5c4kogmj4WcfJhYsuOT3odvv2WBaGhchgR779Ztc0E76COxNEaJa6M7QDyHGw8XQfxCH3j4pMkhFOlPn/ObS3rNUADYUCY8d+Wt4fBbO7PTW7ppDnDaCyxwEIbglsMtkD/cIPIr4f0RPnpV7ZlOmrJ3HdIbmwXi6TqKTwqkHtBmBPvZVpXm4RJN8A6gF22Uc7NY8B3KMYY7Q",
|
|
expected: {
|
|
miiName: "Frieren",
|
|
firstName: "Frieren",
|
|
lastName: "the Slayer",
|
|
islandName: "Wuhu",
|
|
},
|
|
},
|
|
{
|
|
base64: "ky1cf9hr9xotl250Y8cDOGqPd7t51NS8tNVrJMRAI2bfXr4LNirKvqu7ZrvC00vgz70NU8kQRR6H3uAnRaHupxjLbeYfU0s6CduruAEnXP8rZanCeSePQQH0NSL3QqiilT2WEt7nCAMvHwR9fT/LE/k6NBMDHqoK3zqzemr4OlQro0YWBeRJ501EawWan/k/rq2VSGTeLO09CsQD6AFHECtxF9+sSKyJxK1aiu7VhmOZLY6L9VKrlpvahQ1/vHVyYVpFJvc3bdHE8D94bBXkZ18mnXj++ST+j1Had4aki7oqTT83fgs7asRg3DRRZArw8PiKVmZWJ4ODRWR/LNvjIxb1FQqWk9I6S3DEo6AMuBRbXgj1H4YWrRuTkWlEpP2Y/P5+Mvfv5GbNQKYSKwpTYFOCTn13yQ1wtDbF4yG+Ro4Xf45cNBT6k3yqswrKt9bkP2wiULqYZR7McaD1SJ4whFFqcadjpLvbn8zQNFY0lOUTQMGI",
|
|
expected: {
|
|
miiName: "ミク",
|
|
firstName: "ミク",
|
|
lastName: "はつね",
|
|
islandName: "LOSTの",
|
|
},
|
|
},
|
|
];
|
|
|
|
// 372 bytes of zeroes.
|
|
const zeroes372 = "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA";
|
|
|
|
// 32 bytes of zeroes (too short to be a Mii QR code).
|
|
const length32 = "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA=";
|
|
|
|
// Mii QR code encrypted correctly but has an invalid CRC-16 checksum.
|
|
const badCrc = "ULENrXILeXZooDaJvwHMcFaBjGH7k5pj+DDPDV0irq66c01Fh82TIrZerZX2xiB+TbHBzYIaDHdryA9IbR1uH/slSalGCScmjDiL9zpOXlvh8z38NGkmjtL1fVVrshCcJHKeQApdbZU4S6h+wZd0LA==";
|
|
|
|
// Hair dye should be applied to hair only.
|
|
const hairDyeHairOnly = "lHDnfRgqe2+drU303Slkj7+o4JldrKcIdOl5zgLM0LpwQKQY+i3cpt5IIg7LBNAr7TCzHOvi698oUV0QkcyNj71MgtAAaw4MvOdT4dsv0PLof6E7IcjgnCA1ZAJ2Bs5PQTnM/yuVBUIXdq6WYh+nmG3HtxV7zKbEpSy4bqVep8uuvUlfZcB+BQgucPXQLmDnS8ECKwOlANcKTI+ZjIZggVaEsyY88pjRWyXnwe1z4Favw16bIzecesehGlqXzZh9U5Vm5dZP8wmKc3G6TGylYmbnloRd99UYRNULvTQCUer8WljGuV30ftXlJOwfsnwoAiVOGoG3KvbsBpPtPLywR5DavRgQIPd0/b+XUzHQDhkyftMXeqVEalsuEmU/b/1/j4yVL+2lWgD1i2xyET65uJawAnd8jbKbG8lxPMgzIKGVqJB4QmJOl9/dTf21r9GgRFRaFEz+66bVfiYhzXKmJUQv2qx/t/V3r96QzYd08nrWSHK0";
|
|
const hairDyeHairOnlyExpectedCommonColor = 99; // Must apply to hair but not eyebrow and beard.
|
|
// Hair dye should be applied to hair, eyebrow, and beard.
|
|
const hairDyeHairEyebrowBeard = validQrCodes[1].base64; // Miku has hair dye.
|
|
const hairDyeHairEyebrowBeardExpectedCommonColor = 67; // Must apply to hair, eyebrow, and beard.
|
|
|
|
// Should not have hair dye enabled.
|
|
const hairDyeMode0 = validQrCodes[0].base64; // Frieren doesn't have hair dye
|
|
|
|
function base64ToUint8Array(base64: string): Uint8Array {
|
|
const binary = Buffer.from(base64, "base64");
|
|
return new Uint8Array(binary.buffer, binary.byteOffset, binary.byteLength);
|
|
}
|
|
|
|
describe("convertQrCode", () => {
|
|
it.each(validQrCodes.map((t, i) => [i, t]))(
|
|
"should convert valid QR #%i into Mii + TomodachiLifeMii",
|
|
(_, { base64, expected }) => {
|
|
const bytes = base64ToUint8Array(base64);
|
|
const { mii, tomodachiLifeMii } = convertQrCode(bytes);
|
|
|
|
expect(mii).toBeInstanceOf(Mii);
|
|
expect(tomodachiLifeMii).toBeInstanceOf(TomodachiLifeMii);
|
|
|
|
expect(mii.miiName).toBe(expected.miiName);
|
|
expect(tomodachiLifeMii.firstName).toBe(expected.firstName);
|
|
expect(tomodachiLifeMii.lastName).toBe(expected.lastName);
|
|
expect(tomodachiLifeMii.islandName).toBe(expected.islandName);
|
|
});
|
|
|
|
it("should throw an error on zeroed out data", () => {
|
|
const bytes = base64ToUint8Array(zeroes372);
|
|
// Will decrypt wrongly, leading to the expected stream
|
|
// of zeroes in the decrypted data not being intact.
|
|
expect(() => convertQrCode(bytes)).toThrow("QR code is not a valid Mii QR code");
|
|
});
|
|
|
|
it("should throw an error on wrong length", () => {
|
|
const bytes = base64ToUint8Array(length32);
|
|
// Thrown at the beginning of the function.
|
|
expect(() => convertQrCode(bytes)).toThrow("Mii QR code has wrong size (got 32, expected 112 or longer)");
|
|
});
|
|
|
|
it("should throw an error on data with bad CRC-16", () => {
|
|
const bytes = base64ToUint8Array(badCrc);
|
|
// Verified by new Mii() constructor from mii-js.
|
|
expect(() => convertQrCode(bytes)).toThrow("Mii data is not valid");
|
|
});
|
|
|
|
it("should apply hair dye to hair, eyebrow, and beard", () => {
|
|
const bytes = base64ToUint8Array(hairDyeHairEyebrowBeard);
|
|
const { mii, tomodachiLifeMii } = convertQrCode(bytes);
|
|
|
|
expect(tomodachiLifeMii.hairDyeMode).toBe(HairDyeMode.HairEyebrowBeard);
|
|
expect(tomodachiLifeMii.studioHairColor).toBe(hairDyeHairEyebrowBeardExpectedCommonColor);
|
|
expect(mii.hairColor).toBe(hairDyeHairEyebrowBeardExpectedCommonColor);
|
|
expect(mii.eyebrowColor).toBe(hairDyeHairEyebrowBeardExpectedCommonColor);
|
|
expect(mii.facialHairColor).toBe(hairDyeHairEyebrowBeardExpectedCommonColor);
|
|
});
|
|
|
|
it("should apply hair dye to hair only", () => {
|
|
const bytes = base64ToUint8Array(hairDyeHairOnly);
|
|
const { mii, tomodachiLifeMii } = convertQrCode(bytes);
|
|
|
|
expect(tomodachiLifeMii.hairDyeMode).toBe(HairDyeMode.Hair);
|
|
expect(tomodachiLifeMii.studioHairColor).toBe(hairDyeHairOnlyExpectedCommonColor);
|
|
expect(mii.hairColor).toBe(hairDyeHairOnlyExpectedCommonColor);
|
|
expect(mii.eyebrowColor === hairDyeHairOnlyExpectedCommonColor).toBe(false);
|
|
expect(mii.facialHairColor === hairDyeHairOnlyExpectedCommonColor).toBe(false);
|
|
});
|
|
|
|
it("should not apply hair dye if mode is 0", () => {
|
|
const bytes = base64ToUint8Array(hairDyeMode0);
|
|
const { mii, tomodachiLifeMii } = convertQrCode(bytes);
|
|
|
|
expect(tomodachiLifeMii.hairDyeMode).toBe(HairDyeMode.None);
|
|
expect(mii.hairColor === tomodachiLifeMii.studioHairColor).toBe(false);
|
|
expect(mii.hairColor === mii.facialHairColor).toBe(false);
|
|
});
|
|
|
|
/*
|
|
it('should censor bad words in names', () => {
|
|
const qrWithSwears = // TODO TODO
|
|
const { tomodachiLifeMii } = convertQrCode(qrWithSwears);
|
|
expect(tomodachiLifeMii.firstName).not.toMatch(/INSERT_SWEARS_HERE/i);
|
|
});
|
|
*/
|
|
});
|