Skip to content

Commit cffb7e6

Browse files
committed
WIP autocomponents
1 parent 371d684 commit cffb7e6

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

57 files changed

+10307
-373
lines changed

.github/workflows/test.yml

+9
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,15 @@ jobs:
1414
run: pnpm test
1515
- name: Build example package
1616
run: pnpm --filter=blog-example build-vite
17+
test-cypress:
18+
runs-on: ubuntu-latest
19+
steps:
20+
- uses: actions/checkout@v3
21+
- uses: ./.github/actions/setup-test-env
22+
- name: Build all packages (so they can require each other)
23+
run: pnpm build
24+
- name: Run cypress tests
25+
run: pnpm --filter=react exec cypress run
1726
lint:
1827
runs-on: ubuntu-latest
1928
steps:

.gitignore

+4-1
Original file line numberDiff line numberDiff line change
@@ -13,4 +13,7 @@ packages/*/dist
1313
.idea
1414
packages/sample-app
1515
*.orig
16-
cypress.env.json
16+
# cypress environment variables for each developer
17+
cypress.env.json
18+
# generated graphql queries
19+
packages/react/src/internal/gql

Contributing.md

+37-7
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,43 @@
1-
# Development environment
1+
# Contributing
22

3-
We require `node` and `pnpm` to exist. If you're a nix user, we have a flake.nix present that installs the same version of the development tools we use for everyone.
3+
## Development environment
44

5-
# Building TypeScript
5+
We require `node` and `pnpm` to exist. If you're a nix user, we have a `flake.nix` present that installs the same version of the development tools we use for everyone.
6+
7+
## Building TypeScript
68

79
- You can run `pnpm build` to build all the projects in the repo
810
- You can run `pnpm watch` to start the TypeScript watcher process for all the projects in the repo which will recompile files as you change them
911

10-
# Prereleasing
12+
## Running Cypress tests
13+
14+
For running the Cypress tests, you need an API key for communicating with a Gadget app this repo uses for testing. You can ask a Gadget staff member for an existing key, or fork the test app and use your own API key for [this app](https://app.gadget.dev/auth/fork?domain=js-clients-test--development.gadget.app).
15+
16+
To open Cypress and execute tests interactively, run:
17+
18+
```react
19+
pnpm -F=react exec cypress open
20+
```
21+
22+
To run Cypress on the command line, run:
23+
24+
```react
25+
pnpm -F=react exec cypress run
26+
```
27+
28+
## Regenerating GraphQL queries
29+
30+
Most of the GraphQL queries this library makes are assembled dynamically at runtime based on the user's application and the metadata that comes from their API client. For queries we issue against our static part of the GraphQL schema that each application shares though (under `gadgetMeta { ... }`), we have an automatic type-safe GraphQL query generator in place using `graphql-codegen`.
31+
32+
You can author queries using the `graphql` helper, and then generate types for their return values with
33+
34+
```
35+
pnpm -F=react gql-gen
36+
```
37+
38+
See the [`graphql-codegen`](https://the-guild.dev/graphql/codegen/docs/guides/react-vue#writing-graphql-queries) docs for more info.
39+
40+
## Prereleasing
1141

1242
It can be annoying to work with these packages via `pnpm link` sometimes, so we also support building and releasing the package to a git SHA which can then be installed conventionally in another repo. To push a prerelease, run `pnpm --filter=@gadgetinc/api-client-core prerelease`. This will:
1343

@@ -16,17 +46,17 @@ It can be annoying to work with these packages via `pnpm link` sometimes, so we
1646
- push that to the remote git repo
1747
- and log out a version you can then refer to from other repos
1848

19-
# Checking test bundle sizes
49+
## Checking test bundle sizes
2050

2151
We have a small project setup for evaluating what the bundled size of these dependencies might be together. Run:
2252

2353
```shell
2454
pnpm -F=test-bundles test-build
2555
```
2656

27-
to build the test bundles
57+
to build the test bundles.
2858

29-
# Releasing
59+
## Releasing
3060

3161
Releasing is done automatically via [our release workflow](.github/workflows/release.yml). Any commits to the main branch that changes one of our `packages/**/package.json` versions will automatically be published.
3262

packages/api-client-core/src/GadgetConnection.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,7 @@ export class GadgetConnection {
7070
static version = "<prerelease>" as const;
7171

7272
// Options used when generating new GraphQL clients for the base connection and for for transactions
73-
private endpoint: string;
73+
readonly endpoint: string;
7474
private subscriptionClientOptions?: SubscriptionClientOptions;
7575
private websocketsEndpoint: string;
7676
private websocketImplementation?: WebSocket;

packages/api-client-core/src/GadgetRecordList.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import type { GadgetRecord, RecordShape } from "./GadgetRecord.js";
55
import type { InternalModelManager } from "./InternalModelManager.js";
66
import type { AnyModelManager } from "./ModelManager.js";
77
import { GadgetClientError, GadgetOperationError } from "./support.js";
8-
import { PaginateOptions } from "./types.js";
8+
import type { PaginateOptions } from "./types.js";
99

1010
type PaginationConfig = {
1111
pageInfo: { hasNextPage: boolean; hasPreviousPage: boolean; startCursor: string; endCursor: string };

packages/react/codegen.ts

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import type { CodegenConfig } from "@graphql-codegen/cli";
2+
3+
const config: CodegenConfig = {
4+
schema: "https://js-clients-test--development.gadget.app/api/graphql",
5+
documents: ["src/**/*.tsx"],
6+
ignoreNoDocuments: true, // for better experience with the watcher
7+
generates: {
8+
"./src/internal/gql/": {
9+
preset: "client",
10+
presetConfig: {
11+
fragmentMasking: false,
12+
},
13+
config: {
14+
useTypeImports: true,
15+
},
16+
},
17+
},
18+
};
19+
20+
export default config;

packages/react/cypress.config.ts

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
import { defineConfig } from "cypress";
2+
3+
export default defineConfig({
4+
component: {
5+
devServer: {
6+
framework: "react",
7+
bundler: "vite",
8+
viteConfig: {},
9+
},
10+
},
11+
});
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"JS_CLIENTS_TEST_API_KEY": "gsk-<some secret key>"
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
import React from "react";
2+
import { api } from "../../../support/api.js";
3+
import { describeForEachAutoAdapter } from "../../../support/auto.js";
4+
5+
describeForEachAutoAdapter("AutoForm", ({ name, adapter: { AutoForm }, wrapper }) => {
6+
beforeEach(() => {
7+
cy.viewport("macbook-13");
8+
});
9+
10+
it("renders an error if the backend returns one when fetching the model data", () => {
11+
cy.intercept("POST", `${api.connection.endpoint}?operation=ModelActionMetadata`, { forceNetworkError: true });
12+
13+
cy.mountWithWrapper(<AutoForm action={api.widget.create} />, wrapper);
14+
cy.contains("error");
15+
});
16+
17+
it("can customize the submit label", () => {
18+
cy.mountWithWrapper(<AutoForm action={api.widget.create} submitLabel="Save doodad" />, wrapper);
19+
cy.contains("Save doodad");
20+
});
21+
22+
it("can render a form to create model and submit it", () => {
23+
cy.mountWithWrapper(<AutoForm action={api.widget.create} />, wrapper);
24+
25+
// TODO(airhorns): export nice human names in the gadget metadata api
26+
cy.contains("name");
27+
cy.contains("inventoryCount");
28+
cy.contains("anything");
29+
30+
cy.get(`input[name="widget.name"]`).type("test record");
31+
cy.get(`input[name="widget.inventoryCount"]`).type("42");
32+
cy.get("form [type=submit][aria-hidden!=true]").click();
33+
cy.contains("Saved Widget successfully");
34+
});
35+
36+
it("can show invalid field errors from the server and recover from them", () => {
37+
cy.mountWithWrapper(<AutoForm action={api.widget.create} />, wrapper);
38+
39+
// TODO(airhorns): export nice human names in the gadget metadata api
40+
cy.contains("name");
41+
cy.contains("inventoryCount");
42+
cy.contains("anything");
43+
44+
// fill in name but not inventoryCount
45+
cy.get(`input[name="widget.name"]`).type("test record");
46+
47+
cy.get("form [type=submit][aria-hidden!=true]").click();
48+
cy.contains("is a required");
49+
50+
cy.get(`input[name="widget.inventoryCount"]`).type("42");
51+
cy.get("form [type=submit][aria-hidden!=true]").click();
52+
cy.contains("Saved Widget successfully");
53+
});
54+
55+
it("can render a form to update a model without making changes and submit it", async () => {
56+
const name = `test record ${new Date()}`;
57+
58+
cy.wrap(null)
59+
.then(async () => await api.widget.create({ name, inventoryCount: 42, anything: "hello" }))
60+
.then((record) => {
61+
cy.mountWithWrapper(<AutoForm action={api.widget.update} record={record.id} />, wrapper);
62+
cy.get(`input[name="widget.name"]`).should("have.value", name);
63+
cy.get(`input[name="widget.inventoryCount"]`).should("have.value", 42);
64+
cy.get("form [type=submit][aria-hidden!=true]").click();
65+
cy.contains("Saved Widget successfully");
66+
});
67+
});
68+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,108 @@
1+
import { format } from "date-fns";
2+
import { utcToZonedTime } from "date-fns-tz";
3+
import React from "react";
4+
import { PolarisDateTimePicker } from "../../../../src/auto/polaris/PolarisDateTimePicker.js";
5+
import { PolarisWrapper } from "../../../support/auto.js";
6+
7+
const baseDate = new Date("2021-03-05T11:23:10.000Z");
8+
const localTz = Intl.DateTimeFormat().resolvedOptions().timeZone;
9+
const dateInLocalTZ = utcToZonedTime(baseDate, localTz);
10+
11+
describe("PolarisDateTimePicker", () => {
12+
beforeEach(() => {
13+
cy.viewport("macbook-13");
14+
});
15+
16+
describe("date only", () => {
17+
it("can show with a blank current value", () => {
18+
const onChangeSpy = cy.spy().as("onChangeSpy");
19+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" onChange={onChangeSpy} />, PolarisWrapper);
20+
cy.get("#test-date").should("have.value", "");
21+
});
22+
23+
it("can show the current value", () => {
24+
const onChangeSpy = cy.spy().as("onChangeSpy");
25+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" includeTime value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
26+
cy.get("#test-date").should("have.value", format(dateInLocalTZ, "yyyy-MM-dd"));
27+
});
28+
29+
it("can change the date", async () => {
30+
const onChangeSpy = cy.spy().as("onChangeSpy");
31+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
32+
cy.get("#test-date").click();
33+
cy.get(`[aria-label='Thursday March 4 2021']`).click();
34+
// eslint-disable-next-line jest/valid-expect-in-promise
35+
cy.get("@onChangeSpy")
36+
.should("have.been.called")
37+
.then(() => {
38+
expect(onChangeSpy.getCalls()[0].args[0].toISOString()).equal(new Date("2021-03-04T11:23:10.000Z").toISOString());
39+
});
40+
});
41+
42+
it("can change the date across a DST boundary", async () => {
43+
const onChangeSpy = cy.spy().as("onChangeSpy");
44+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
45+
cy.get("#test-date").click();
46+
cy.get(`[aria-label='Wednesday March 17 2021']`).click();
47+
// eslint-disable-next-line jest/valid-expect-in-promise
48+
cy.get("@onChangeSpy")
49+
.should("have.been.called")
50+
.then(() => {
51+
expect(onChangeSpy.getCalls()[0].args[0].toISOString()).equal(new Date("2021-03-17T10:23:10.000Z").toISOString());
52+
});
53+
});
54+
});
55+
56+
describe("date and time", () => {
57+
it("can show with a blank current value", () => {
58+
const onChangeSpy = cy.spy().as("onChangeSpy");
59+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" includeTime onChange={onChangeSpy} />, PolarisWrapper);
60+
cy.get("#test-date").should("have.value", "");
61+
cy.get("#test-time").should("have.value", "");
62+
});
63+
64+
it("can show the current value", () => {
65+
const onChangeSpy = cy.spy().as("onChangeSpy");
66+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" includeTime value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
67+
cy.get("#test-date").should("have.value", format(dateInLocalTZ, "yyyy-MM-dd"));
68+
cy.get("#test-time").should("have.value", format(dateInLocalTZ, "K:m aa"));
69+
});
70+
71+
it("can change the date", async () => {
72+
const onChangeSpy = cy.spy().as("onChangeSpy");
73+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" includeTime value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
74+
cy.get("#test-date").click();
75+
cy.get(`[aria-label='Thursday March 4 2021']`).click();
76+
// eslint-disable-next-line jest/valid-expect-in-promise
77+
cy.get("@onChangeSpy")
78+
.should("have.been.called")
79+
.then(() => {
80+
expect(onChangeSpy.getCalls()[0].args[0].toISOString()).equal(new Date("2021-03-04T11:23:10.000Z").toISOString());
81+
});
82+
});
83+
84+
it("can change the time", () => {
85+
const onChangeSpy = cy.spy().as("onChangeSpy");
86+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" includeTime value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
87+
cy.get("#test-time").clear().type("11:00 AM", { parseSpecialCharSequences: false });
88+
// eslint-disable-next-line jest/valid-expect-in-promise
89+
cy.get("@onChangeSpy")
90+
.should("have.been.called")
91+
.then(() => {
92+
const calls = onChangeSpy.getCalls();
93+
94+
expect(calls[calls.length - 1].args[0].toISOString()).equal(new Date("2021-03-05T16:00:00.000Z").toISOString());
95+
});
96+
});
97+
98+
it("can enter an invalid time and show an error", () => {
99+
const onChangeSpy = cy.spy().as("onChangeSpy");
100+
cy.mountWithWrapper(<PolarisDateTimePicker id="test" includeTime value={baseDate} onChange={onChangeSpy} />, PolarisWrapper);
101+
cy.get("#test-time").clear().type("foo");
102+
cy.get("body").click();
103+
cy.contains("Invalid time");
104+
cy.get("#test-time").clear().type("12:21 AM");
105+
cy.contains("Invalid time").should("not.exist");
106+
});
107+
});
108+
});

0 commit comments

Comments
 (0)