Skip to content

Commit 3b734ad

Browse files
authored
More command tests (#23)
This adds testing for all commands except SSH.
1 parent 7f5e7f3 commit 3b734ad

File tree

18 files changed

+368
-32
lines changed

18 files changed

+368
-32
lines changed

__mocks__/@firebase/auth.ts

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
export const getAuth = jest.fn().mockReturnValue({});
2+
3+
export const signInWithCredential = jest.fn();
4+
5+
export const SignInMethod = {
6+
GOOGLE: "google.com",
7+
};
8+
9+
export class OAuthProvider {
10+
credential() {
11+
return "test-credential";
12+
}
13+
}

__mocks__/firebase/firestore.ts

Lines changed: 27 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,37 @@
1-
import { noop } from "lodash";
2-
31
export const doc = jest.fn();
42

3+
export const getCollection = jest.fn();
4+
5+
export const getDoc = jest.fn();
6+
57
export const getFirestore = jest.fn().mockReturnValue({});
68

79
let snapshotCallbacks: ((snapshot: object) => void)[] = [];
810

11+
/** Triggerable mock onSnapshot
12+
*
13+
* Usage:
14+
* ```
15+
* import { onSnapshot } from "firebase/firestore";
16+
*
17+
* beforeEach(() => {
18+
* (onSnapshot as any).clear();
19+
* })
20+
*
21+
* test(..., () => {
22+
* // call code under test here
23+
* (onSnapshot as any).trigger(data)
24+
* })
25+
* ```
26+
*
27+
* Note that only one `onSnapshot` may be tested at a time.
28+
*/
929
export const onSnapshot = Object.assign(
1030
jest.fn().mockImplementation((_doc, cb) => {
1131
snapshotCallbacks.push(cb);
12-
return noop;
32+
return () => {
33+
snapshotCallbacks = [];
34+
};
1335
}),
1436
{
1537
clear: (snapshotCallbacks = []),
@@ -21,4 +43,6 @@ export const onSnapshot = Object.assign(
2143
}
2244
);
2345

46+
export const query = jest.fn();
47+
2448
export const terminate = jest.fn();

jest.config.js

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
/** @type {import('ts-jest').JestConfigWithTsJest} */
22
module.exports = {
3+
testRegex: ".*\\.test\\.ts$",
4+
prettierPath: null,
35
preset: "ts-jest",
46
testEnvironment: "node",
57
modulePathIgnorePatterns: ["/build"],
Lines changed: 40 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
// Jest Snapshot v1, https://goo.gl/fbAQLP
2+
3+
exports[`login organization exists should write the user's identity to the file system 1`] = `
4+
[
5+
[
6+
"/path/to/home/.p0",
7+
"{
8+
"credential": {
9+
"access_token": "test-access-token",
10+
"id_token": "test-id-token",
11+
"token_type": "oidc",
12+
"scope": "oidc",
13+
"expires_in": 3600,
14+
"refresh_token": "test-refresh-token",
15+
"device_secret": "test-device-secret",
16+
"expires_at": 1600003599
17+
},
18+
"org": {
19+
"slug": "test-org",
20+
"tenantId": "test-tenant",
21+
"ssoProvider": "google"
22+
}
23+
}",
24+
{
25+
"mode": "600",
26+
},
27+
],
28+
]
29+
`;
30+
31+
exports[`login organization exists validates authentication 1`] = `
32+
[
33+
[
34+
{
35+
"tenantId": "test-tenant",
36+
},
37+
"test-credential",
38+
],
39+
]
40+
`;

src/commands/__tests__/login.test.ts

Lines changed: 66 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
import { pluginLoginMap } from "../../plugins/login";
2+
import { mockGetDoc } from "../../testing/firestore";
3+
import { login } from "../login";
4+
import { signInWithCredential } from "firebase/auth";
5+
import { readFile, writeFile } from "fs/promises";
6+
7+
jest.spyOn(Date, "now").mockReturnValue(1.6e12);
8+
jest.mock("fs/promises");
9+
jest.mock("../../drivers/auth", () => ({
10+
...jest.requireActual("../../drivers/auth"),
11+
IDENTITY_FILE_PATH: "/path/to/home/.p0",
12+
}));
13+
jest.mock("../../drivers/stdio");
14+
jest.mock("../../plugins/login");
15+
16+
const mockReadFile = readFile as jest.Mock;
17+
const mockWriteFile = writeFile as jest.Mock;
18+
19+
describe("login", () => {
20+
it("prints a friendly error if the org is not found", async () => {
21+
mockGetDoc(undefined);
22+
await expect(login({ org: "test-org" })).rejects.toMatchInlineSnapshot(
23+
`"Could not find organization"`
24+
);
25+
});
26+
it("should print a friendly error if unsupported login", async () => {
27+
mockGetDoc({
28+
slug: "test-org",
29+
tenantId: "test-tenant",
30+
ssoProvider: "microsoft",
31+
});
32+
await expect(login({ org: "test-org" })).rejects.toMatchInlineSnapshot(
33+
`"Unsupported login for your organization"`
34+
);
35+
});
36+
describe("organization exists", () => {
37+
let credentialData: string = "";
38+
mockReadFile.mockImplementation(async () =>
39+
Buffer.from(credentialData, "utf-8")
40+
);
41+
mockWriteFile.mockImplementation(async (_path, data) => {
42+
credentialData = data;
43+
});
44+
beforeEach(() => {
45+
credentialData = "";
46+
jest.clearAllMocks();
47+
mockGetDoc({
48+
slug: "test-org",
49+
tenantId: "test-tenant",
50+
ssoProvider: "google",
51+
});
52+
});
53+
it("should call the provider's login function", async () => {
54+
await login({ org: "test-org" });
55+
expect(pluginLoginMap.google).toHaveBeenCalled();
56+
});
57+
it("should write the user's identity to the file system", async () => {
58+
await login({ org: "test-org" });
59+
expect(mockWriteFile.mock.calls).toMatchSnapshot();
60+
});
61+
it("validates authentication", async () => {
62+
await login({ org: "test-org" });
63+
expect((signInWithCredential as jest.Mock).mock.calls).toMatchSnapshot();
64+
});
65+
});
66+
});

src/commands/__tests__/ls.test.ts

Lines changed: 3 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { fetchCommand } from "../../drivers/api";
22
import { print1, print2 } from "../../drivers/stdio";
3+
import { failure } from "../../testing/yargs";
34
import { lsCommand } from "../ls";
45
import yargs from "yargs";
56

@@ -25,7 +26,7 @@ describe("ls", () => {
2526
});
2627

2728
it("should print list response", async () => {
28-
await lsCommand(yargs).parse(command);
29+
await lsCommand(yargs()).parse(command);
2930
expect(mockPrint1.mock.calls).toMatchSnapshot();
3031
expect(mockPrint2.mock.calls).toMatchSnapshot();
3132
});
@@ -51,14 +52,7 @@ Unknown argument: foo`,
5152
});
5253

5354
it("should print error message", async () => {
54-
let error: any;
55-
try {
56-
await lsCommand(yargs)
57-
.fail((_, err) => (error = err))
58-
.parse(command);
59-
} catch (thrown: any) {
60-
error = thrown;
61-
}
55+
const error = await failure(lsCommand(yargs()), command);
6256
expect(error).toMatchSnapshot();
6357
});
6458
});

src/commands/__tests__/request.test.ts

Lines changed: 4 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { fetchCommand } from "../../drivers/api";
22
import { print1, print2 } from "../../drivers/stdio";
3+
import { failure } from "../../testing/yargs";
34
import { RequestResponse } from "../../types/request";
45
import { sleep } from "../../util";
56
import { requestCommand } from "../request";
@@ -40,7 +41,7 @@ describe("request", () => {
4041
(isPreexisting, isPersistent, should) => {
4142
it(`should${should ? "" : " not"} print request response`, async () => {
4243
mockFetch({ isPreexisting, isPersistent });
43-
await requestCommand(yargs).parse(command);
44+
await requestCommand(yargs()).parse(command);
4445
expect(mockPrint2.mock.calls).toMatchSnapshot();
4546
expect(mockPrint1).not.toHaveBeenCalled();
4647
});
@@ -49,7 +50,7 @@ describe("request", () => {
4950

5051
it("should wait for access", async () => {
5152
mockFetch();
52-
const promise = requestCommand(yargs).parse(`${command} --wait`);
53+
const promise = requestCommand(yargs()).parse(`${command} --wait`);
5354
const wait = sleep(10);
5455
await expect(wait).resolves.toBeUndefined();
5556
(onSnapshot as any).trigger({
@@ -83,14 +84,7 @@ Unknown argument: foo`,
8384
});
8485

8586
it("should print error message", async () => {
86-
let error: any;
87-
try {
88-
await requestCommand(yargs)
89-
.fail((_, err) => (error = err))
90-
.parse(command);
91-
} catch (thrown: any) {
92-
error = thrown;
93-
}
87+
const error = await failure(requestCommand(yargs()), command);
9488
expect(error).toMatchSnapshot();
9589
});
9690
});
Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
export const samlResponse = `<html>
2+
<body>
3+
<input name="SAMLResponse" type="hidden" value="<?xml version="1.0" encoding="UTF-8"?>
 <saml2p:Response xmlns:saml2p="urn:oasis:names:tc:SAML:2.0:protocol" xmlns:xs="http://www.w3.org/2001/XMLSchema" Destination="https://signin.aws.amazon.com/saml" ID="abc" IssueInstant="2024-01-01T00:00:00.000Z" Version="2.0">
  <saml2:Issuer xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion" Format="urn:oasis:names:tc:SAML:2.0:nameid-format:entity">http://www.okta.com/abcdef</saml2:Issuer>
  <ds:Signature xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
    <ds:SignedInfo>
      <ds:CanonicalizationMethod Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/>
      <ds:SignatureMethod Algorithm="http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"/>
      <ds:Reference URI="#abc">
        <ds:Transforms>
          <ds:Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/>
          <ds:Transform Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#">
            <ec:InclusiveNamespaces xmlns:ec="http://www.w3.org/2001/10/xml-exc-c14n#" PrefixList="xs"/>
          </ds:Transform>
        </ds:Transforms>
        <ds:DigestMethod Algorithm="http://www.w3.org/2001/04/xmlenc#sha256"/>
        <ds:DigestValue>digest</ds:DigestValue>
      </ds:Reference>
    </ds:SignedInfo>
    <ds:SignatureValue>signature</ds:SignatureValue>
    <ds:KeyInfo>
      <ds:X509Data>
        <ds:X509Certificate>certificate</ds:X509Certificate>
      </ds:X509Data>
    </ds:KeyInfo>
  </ds:Signature>
  <saml2p:Status xmlns:saml2p="urn:oasis:names:tc:SAML:2.0:protocol">
    <saml2p:StatusCode Value="urn:oasis:names:tc:SAML:2.0:status:Success"/>
  </saml2p:Status>
  <saml2:Assertion xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion" xmlns:xs="http://www.w3.org/2001/XMLSchema" ID="abc" IssueInstant="2024-01-01T00:00:00.000Z" Version="2.0">
    <saml2:Issuer xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion" Format="urn:oasis:names:tc:SAML:2.0:nameid-format:entity">http://www.okta.com/abcdef</saml2:Issuer>
    <ds:Signature xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
      <ds:SignedInfo>
        <ds:CanonicalizationMethod Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/>
        <ds:SignatureMethod Algorithm="http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"/>
        <ds:Reference URI="#id8477729977532301927088708">
          <ds:Transforms>
            <ds:Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/>
            <ds:Transform Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#">
              <ec:InclusiveNamespaces xmlns:ec="http://www.w3.org/2001/10/xml-exc-c14n#" PrefixList="xs"/>
            </ds:Transform>
          </ds:Transforms>
          <ds:DigestMethod Algorithm="http://www.w3.org/2001/04/xmlenc#sha256"/>
          <ds:DigestValue>digest</ds:DigestValue>
        </ds:Reference>
      </ds:SignedInfo>
      <ds:SignatureValue>signature</ds:SignatureValue>
      <ds:KeyInfo>
        <ds:X509Data>
          <ds:X509Certificate>certificate</ds:X509Certificate>
        </ds:X509Data>
      </ds:KeyInfo>
    </ds:Signature>
    <saml2:Subject xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion">
      <saml2:NameID Format="urn:oasis:names:tc:SAML:2.0:nameid-format:unspecified">test-user@test.com</saml2:NameID>
      <saml2:SubjectConfirmation Method="urn:oasis:names:tc:SAML:2.0:cm:bearer">
        <saml2:SubjectConfirmationData NotOnOrAfter="2024-01-01T00:00:00.000Z" Recipient="https://signin.aws.amazon.com/saml"/>
      </saml2:SubjectConfirmation>
    </saml2:Subject>
    <saml2:Conditions xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion" NotBefore="2024-01-01T00:00:00.000Z" NotOnOrAfter="2024-01-01T00:00:00.000Z">
      <saml2:AudienceRestriction>
        <saml2:Audience>urn:amazon:webservices</saml2:Audience>
      </saml2:AudienceRestriction>
    </saml2:Conditions>
    <saml2:AuthnStatement xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion" AuthnInstant="2024-01-01T00:00:00.000Z" SessionIndex="abc">
      <saml2:AuthnContext>
        <saml2:AuthnContextClassRef>urn:oasis:names:tc:SAML:2.0:ac:classes:PasswordProtectedTransport</saml2:AuthnContextClassRef>
      </saml2:AuthnContext>
    </saml2:AuthnStatement>
    <saml2:AttributeStatement xmlns:saml2="urn:oasis:names:tc:SAML:2.0:assertion">
      <saml2:Attribute Name="https://aws.amazon.com/SAML/Attributes/Role" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:uri">
        <saml2:AttributeValue xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="xs:string">arn:aws:iam::1:saml-provider/test_okta,arn:aws:iam::1:role/Role1</saml2:AttributeValue>
        <saml2:AttributeValue xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="xs:string">arn:aws:iam::1:saml-provider/test_okta,arn:aws:iam::1:role/Role2</saml2:AttributeValue>
      </saml2:Attribute>
      <saml2:Attribute Name="https://aws.amazon.com/SAML/Attributes/RoleSessionName" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic">
        <saml2:AttributeValue xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="xs:string">test-user@test.com</saml2:AttributeValue>
      </saml2:Attribute>
      <saml2:Attribute Name="https://aws.amazon.com/SAML/Attributes/SessionDuration" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic">
        <saml2:AttributeValue xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="xs:string">3600</saml2:AttributeValue>
      </saml2:Attribute>
      <saml2:Attribute Name="https://aws.amazon.com/SAML/Attributes/PrincipalTag:org" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic">
        <saml2:AttributeValue xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="xs:string">test</saml2:AttributeValue>
      </saml2:Attribute>
      <saml2:Attribute Name="https://aws.amazon.com/SAML/Attributes/SourceIdentity" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:uri">
        <saml2:AttributeValue xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="xs:string">test-user@test.com</saml2:AttributeValue>
      </saml2:Attribute>
    </saml2:AttributeStatement>
  </saml2:Assertion>
</saml2p:Response>" />
4+
</body>
5+
</html>`;
Lines changed: 24 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
export const stsResponse = `<AssumeRoleWithSAMLResponse xmlns="https://sts.amazonaws.com/doc/2011-06-15/">
2+
<AssumeRoleWithSAMLResult>
3+
<Audience>https://signin.aws.amazon.com/saml</Audience>
4+
<AssumedRoleUser>
5+
<AssumedRoleId>ABCDEFGHIJLMNOPQRST:test-user@test.com</AssumedRoleId>
6+
<Arn>arn:aws:sts::1:assumed-role/Role1/test-user@test.com</Arn>
7+
</AssumedRoleUser>
8+
<Credentials>
9+
<AccessKeyId>test-access-key</AccessKeyId>
10+
<SecretAccessKey>secret-access-key</SecretAccessKey>
11+
<SessionToken>session-token</SessionToken>
12+
<Expiration>2024-02-22T00:18:21Z</Expiration>
13+
</Credentials>
14+
<Subject>test-user@test.com</Subject>
15+
<NameQualifier>abcdefghijklmnop</NameQualifier>
16+
<SourceIdentity>test-user@test.com</SourceIdentity>
17+
<PackedPolicySize>2</PackedPolicySize>
18+
<SubjectType>unspecified</SubjectType>
19+
<Issuer>http://www.okta.com/abc</Issuer>
20+
</AssumeRoleWithSAMLResult>
21+
<ResponseMetadata>
22+
<RequestId>f5b94ad4-f322-4d7b-b568-84f2ec184cd7</RequestId>
23+
</ResponseMetadata>
24+
</AssumeRoleWithSAMLResponse>`;
Lines changed: 44 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
// Jest Snapshot v1, https://goo.gl/fbAQLP
2+
3+
exports[`aws role a single installed account with Okta SAML assume should assume a role: stderr 1`] = `
4+
[
5+
[
6+
"Execute the following commands:
7+
",
8+
],
9+
[
10+
"
11+
Or, populate these environment variables using BASH command substitution:
12+
13+
$(p0 aws role assume Role1)
14+
",
15+
],
16+
]
17+
`;
18+
19+
exports[`aws role a single installed account with Okta SAML assume should assume a role: stdout 1`] = `
20+
[
21+
[
22+
" export AWS_ACCESS_KEY_ID=test-access-key
23+
export AWS_SECRET_ACCESS_KEY=secret-access-key
24+
export AWS_SESSION_TOKEN=session-token",
25+
],
26+
]
27+
`;
28+
29+
exports[`aws role a single installed account with Okta SAML ls lists roles: stderr 1`] = `
30+
[
31+
[
32+
"Your available roles for account 1:",
33+
],
34+
]
35+
`;
36+
37+
exports[`aws role a single installed account with Okta SAML ls lists roles: stdout 1`] = `
38+
[
39+
[
40+
" Role1
41+
Role2",
42+
],
43+
]
44+
`;

0 commit comments

Comments
 (0)