Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Connector Builder] Add paginator #20698

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
130 commits
Select commit Hold shift + click to select a range
699470e
move connector builder components into the same shared components/con…
lmossman Dec 1, 2022
827532b
move diff over from poc branch
lmossman Dec 1, 2022
4899673
save current progress
lmossman Dec 3, 2022
a1b62f1
add modal for adding streams
lmossman Dec 3, 2022
39e8ba8
focus stream after adding and reset button style
lmossman Dec 5, 2022
2352ef9
add reset confirm modal and select view on add
lmossman Dec 6, 2022
df492fa
style global config and streams buttons
lmossman Dec 6, 2022
5a99c7a
styling improvements
lmossman Dec 6, 2022
62fc08c
handle long stream names better
lmossman Dec 6, 2022
7b26b78
pull in connector manifest schema directly
lmossman Dec 6, 2022
71a88f1
add box shadows to resizable panels
lmossman Dec 6, 2022
d7bee54
upgrade orval and use connector manifest schema directly
lmossman Dec 6, 2022
162ecf2
remove airbyte protocol from connector builder api spec
lmossman Dec 6, 2022
def60b0
Merge branch 'master' into lmossman/use-connector-manifest-schema-dir…
lmossman Dec 6, 2022
ff9aa76
generate python models from openapi change
lmossman Dec 6, 2022
3678956
merge with lmossman/use-connector-manifest-schema-directly
lmossman Dec 6, 2022
b16d4a8
fix position of yaml toggle
lmossman Dec 6, 2022
67fb9dd
handle no stream case with better looking message
lmossman Dec 7, 2022
bb42007
group global fields into single object and fix console error
lmossman Dec 7, 2022
9bb1812
confirmation modal on toggling dirty form + cleanup
lmossman Dec 7, 2022
587ca2f
merge with master
lmossman Dec 7, 2022
81cf108
fix connector name display
lmossman Dec 7, 2022
094a045
undo change to manifest schema
lmossman Dec 7, 2022
d1c8c80
remove commented code
lmossman Dec 7, 2022
aebac20
remove unnecessary change
lmossman Dec 7, 2022
1e39be4
fix spacing
lmossman Dec 7, 2022
09cf875
use shadow mixin for connector img
lmossman Dec 7, 2022
ec65d67
add comment about connector img
lmossman Dec 7, 2022
f9a3fa1
Merge remote-tracking branch 'origin/master' into lmossman/connector-…
Dec 8, 2022
603820a
change onSubmit to no-op
lmossman Dec 8, 2022
cb2f299
remove console log
lmossman Dec 8, 2022
d1672a0
clean up styling
lmossman Dec 8, 2022
7255784
simplify sidebar to remove StreamSelectButton component
lmossman Dec 8, 2022
65cd42b
swap colors of toggle
lmossman Dec 8, 2022
10d7893
move FormikPatch to src/core/form
lmossman Dec 8, 2022
fc89001
move types up to connectorBuilder/ level
lmossman Dec 8, 2022
24eaaff
use grid display for ui yaml toggle button
lmossman Dec 8, 2022
de604f6
use spread instead of setting array index directly
lmossman Dec 8, 2022
e696e57
add intl in missing places
lmossman Dec 8, 2022
5952f70
merge with master
lmossman Dec 8, 2022
b7c08f6
pull connector manifest schema in through separate openapi spec
lmossman Dec 8, 2022
dd04e61
erge branch 'lmossman/connector-builder-config-ui' of github.com:airb…
Dec 9, 2022
d6f10b4
use correct intl string id
lmossman Dec 9, 2022
5ffe20e
throttle setting json manifest in yaml editor
lmossman Dec 9, 2022
b82ba2a
use button prop instead of manually styling
lmossman Dec 9, 2022
b9580b4
consolidate AddStreamButton styles
lmossman Dec 9, 2022
1587243
fix sidebar flex styles
lmossman Dec 9, 2022
62a60e5
use specific flex properties instead of flex
lmossman Dec 9, 2022
092821d
clean up download and reset button styles
lmossman Dec 10, 2022
7b8dec8
use row-reverse for yaml editor download button
lmossman Dec 10, 2022
6d68db9
fix stream selector styles to remove margins
lmossman Dec 10, 2022
91bc357
give connector setup guide panel same corner and shadow styles
lmossman Dec 10, 2022
4ffb969
remove blur from page display
lmossman Dec 10, 2022
70afdf8
set view to stream when selected in test panel
lmossman Dec 10, 2022
f656506
add placeholder when stream name is empty
lmossman Dec 10, 2022
c437d30
switch to index-based stream selection to preserve testing panel sele…
lmossman Dec 10, 2022
3cdbc70
handle empty name in stream selector
lmossman Dec 10, 2022
831f574
Merge branch 'lmossman/connector-builder-config-ui' of github.com:air…
Dec 12, 2022
e2730fe
Merge branch 'lmossman/connector-builder-config-ui' into flash1293/in…
Dec 12, 2022
269ae82
make connector form work in connector builder
Dec 12, 2022
901fc63
wip
Dec 12, 2022
4944d3f
Merge branch 'flash1293/connector-form-remove-ui-widget-state' into f…
Dec 13, 2022
3255298
fix small stuff
Dec 13, 2022
333fb0b
Merge branch 'flash1293/connector-form-remove-ui-widget-state' into f…
Dec 13, 2022
cec206f
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 13, 2022
ba28014
add basic input UI
Dec 13, 2022
53ad55f
Merge branch 'flash1293/connector-form-remove-ui-widget-state' into f…
Dec 14, 2022
5300933
Merge remote-tracking branch 'origin/master' into flash1293/integrate…
Dec 14, 2022
935188b
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 14, 2022
b694dc5
user inputs
Dec 14, 2022
e7d81ea
make most of inputs configuration work
Dec 14, 2022
3de021a
Merge branch 'flash1293/connector-form-remove-ui-widget-state' into f…
Dec 15, 2022
552fc65
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 15, 2022
ace1299
fix a bunch of stuff
Dec 15, 2022
7fa1752
handle unknown config types
Dec 15, 2022
7196409
Merge branch 'flash1293/connector-form-remove-ui-widget-state' into f…
Dec 16, 2022
4a37a67
add warning label
Dec 16, 2022
10fa73e
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 16, 2022
7e48ba5
fix label
Dec 16, 2022
75db697
fix some styling
Dec 16, 2022
b340ef4
Merge branch 'flash1293/connector-form-remove-ui-widget-state' into f…
Dec 16, 2022
98c146c
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 16, 2022
9835a5c
review comments
Dec 18, 2022
c352df0
Merge branch 'feature/connector-builder' into flash1293/integrate-con…
Dec 19, 2022
ed43eab
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 19, 2022
7958ef1
improve state management and error handling
Dec 19, 2022
beb4968
Merge branch 'feature/connector-builder' into flash1293/integrate-con…
Dec 19, 2022
b3cfb8f
Merge branch 'flash1293/integrate-connector-form' into flash1293/allo…
Dec 19, 2022
49f55a1
Merge branch 'flash1293/allow-configuring-inputs' of github.com:airby…
Dec 19, 2022
04d87cf
allow auth configuration
Dec 19, 2022
b796201
check for conflicts with the inferred inputs
Dec 19, 2022
18498c7
fix invisible inputs
Dec 19, 2022
9ffa34e
handle stored form values that don't contain new fields properly
lmossman Dec 19, 2022
8c4b6c7
Merge branch 'flash1293/allow-configuring-inputs' into flash1293/conn…
lmossman Dec 19, 2022
c0350d3
Merge branch 'flash1293/allow-configuring-inputs' of github.com:airby…
Dec 20, 2022
e98ebc1
Merge branch 'feature/connector-builder' into flash1293/allow-configu…
Dec 20, 2022
fb1ec21
Merge branch 'flash1293/allow-configuring-inputs' into flash1293/conn…
Dec 20, 2022
a17ab05
session token and oauth authentication
Dec 20, 2022
7b6470e
fill in session token variable
Dec 20, 2022
23f4fbe
fix merge of default values
lmossman Dec 20, 2022
d4975bf
add primaryKey and cursorField to builder types, and consolidate defa…
lmossman Dec 20, 2022
32213f8
add cursor and primary key fields to ui
lmossman Dec 20, 2022
3e20324
save
lmossman Dec 20, 2022
3ee0ffb
add page size and token option inputs
lmossman Dec 21, 2022
0576478
fixes after rebase
lmossman Dec 21, 2022
5e45f59
add pagination
lmossman Dec 21, 2022
f101602
fix pagination types
lmossman Dec 21, 2022
5fb62e2
handle empty field_name better
lmossman Dec 21, 2022
7cce350
Update airbyte-webapp/src/locales/en.json
Dec 21, 2022
20e8044
Update airbyte-webapp/src/components/connectorBuilder/Builder/InputsV…
Dec 21, 2022
50bddcb
inputs editing weirdness
Dec 21, 2022
0ca37ad
input form reset
Dec 21, 2022
8095fda
using the Label component
Dec 21, 2022
fbeb2e1
Merge branch 'flash1293/allow-configuring-inputs' into flash1293/conn…
Dec 21, 2022
1bccbbd
reduce redundancy and hide advanced input options for inferred inputs
Dec 21, 2022
eaec279
unnecessary validation
Dec 21, 2022
a338590
typo
Dec 21, 2022
2429d29
unnecessary effect hook
Dec 21, 2022
6dd02d9
build spec even for invalid forms but do not update stream list
Dec 21, 2022
84803ce
Merge branch 'flash1293/connector-builder-authentication' into flash1…
Dec 21, 2022
4bd0bdc
typos
Dec 21, 2022
ca0219b
make sure validation error does not go away
Dec 21, 2022
134fc17
Merge branch 'flash1293/oauth-session-auth' into lmossman/connector-b…
Dec 21, 2022
ed9860e
make primary key and cursor optional, and reorder
lmossman Dec 21, 2022
e029279
save toggle group progress
lmossman Dec 21, 2022
e964eb8
fix style of toggle label
lmossman Dec 21, 2022
a23ffb8
handle empty values better
lmossman Dec 21, 2022
e3c578d
fix page size/token option field validation and rendering
lmossman Dec 21, 2022
24bfc32
handle cursor pagination page size option correctly
lmossman Dec 21, 2022
e2450c4
Merge branch 'feature/connector-builder' into lmossman/connector-buil…
Dec 22, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { Form, Formik, useField } from "formik";
import merge from "lodash/merge";
import { useState } from "react";
import React from "react";
import { FormattedMessage, useIntl } from "react-intl";
Expand All @@ -9,7 +10,7 @@ import { Modal, ModalBody, ModalFooter } from "components/ui/Modal";
import { FormikPatch } from "core/form/FormikPatch";

import { ReactComponent as PlusIcon } from "../../connection/ConnectionOnboarding/plusIcon.svg";
import { BuilderStream } from "../types";
import { BuilderStream, DEFAULT_BUILDER_STREAM_VALUES } from "../types";
import styles from "./AddStreamButton.module.scss";
import { BuilderField } from "./BuilderField";

Expand Down Expand Up @@ -49,18 +50,11 @@ export const AddStreamButton: React.FC<AddStreamButtonProps> = ({ onAddStream, b
onSubmit={(values: AddStreamValues) => {
helpers.setValue([
...streamsField.value,
{
fieldPointer: [],
httpMethod: "GET",
requestOptions: {
requestParameters: [],
requestHeaders: [],
requestBody: [],
},
merge({}, DEFAULT_BUILDER_STREAM_VALUES, {
...initialValues,
name: values.streamName,
urlPath: values.urlPath,
},
}),
]);
setIsOpen(false);
onAddStream(numStreams);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,4 +18,5 @@
.form {
flex: 1;
padding: variables.$spacing-xl;
overflow: auto;
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,3 +6,8 @@
flex-direction: column;
gap: variables.$spacing-xl;
}

.toggleContainer {
display: flex;
gap: variables.$spacing-md;
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,38 @@ import classNames from "classnames";
import React from "react";

import { Card } from "components/ui/Card";
import { CheckBox } from "components/ui/CheckBox";

import styles from "./BuilderCard.module.scss";

interface BuilderCardProps {
className?: string;
toggleConfig?: {
label: React.ReactNode;
toggledOn: boolean;
onToggle: (newToggleValue: boolean) => void;
};
}

export const BuilderCard: React.FC<React.PropsWithChildren<BuilderCardProps>> = ({ children, className }) => {
return <Card className={classNames(className, styles.card)}>{children}</Card>;
export const BuilderCard: React.FC<React.PropsWithChildren<BuilderCardProps>> = ({
children,
className,
toggleConfig,
}) => {
return (
<Card className={classNames(className, styles.card)}>
{toggleConfig && (
<div className={styles.toggleContainer}>
<CheckBox
checked={toggleConfig.toggledOn}
onChange={(event) => {
toggleConfig.onToggle(event.target.checked);
}}
/>
{toggleConfig.label}
</div>
)}
{(!toggleConfig || toggleConfig.toggledOn) && children}
</Card>
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,9 @@ export const BuilderField: React.FC<BuilderFieldProps> = ({
{...field}
onChange={(e) => {
field.onChange(e);
if (e.target.value === "") {
helpers.setValue(undefined);
}
props.onChange?.(e.target.value);
}}
type={props.type}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,14 +10,10 @@ import { Heading } from "components/ui/Heading";
import { Text } from "components/ui/Text";

import { useConfirmationModalService } from "hooks/services/ConfirmationModal";
import {
BuilderView,
DEFAULT_BUILDER_FORM_VALUES,
useConnectorBuilderState,
} from "services/connectorBuilder/ConnectorBuilderStateService";
import { BuilderView, useConnectorBuilderState } from "services/connectorBuilder/ConnectorBuilderStateService";

import { DownloadYamlButton } from "../DownloadYamlButton";
import { BuilderFormValues, getInferredInputs } from "../types";
import { BuilderFormValues, DEFAULT_BUILDER_FORM_VALUES, getInferredInputs } from "../types";
import { useBuilderErrors } from "../useBuilderErrors";
import { AddStreamButton } from "./AddStreamButton";
import styles from "./BuilderSidebar.module.scss";
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import { useField } from "formik";

import { RequestOption } from "core/request/ConnectorManifest";

import { injectIntoValues } from "../types";
import { BuilderField } from "./BuilderField";

interface InjectRequestOptionFieldsProps {
path: string;
descriptor: string;
excludeInjectIntoValues?: string[];
}

export const InjectRequestOptionFields: React.FC<InjectRequestOptionFieldsProps> = ({
path,
descriptor,
excludeInjectIntoValues,
}) => {
const [field, , helpers] = useField<RequestOption>(path);

return (
<>
<BuilderField
type="enum"
path={`${path}.inject_into`}
options={
excludeInjectIntoValues
? injectIntoValues.filter((val) => !excludeInjectIntoValues.includes(val))
: injectIntoValues
}
onChange={(newValue) => {
if (newValue === "path") {
helpers.setValue({ inject_into: newValue, field_name: undefined });
}
}}
label="Inject into"
tooltip={`Configures where the ${descriptor} should be set on the HTTP requests`}
/>
{field.value.inject_into !== "path" && (
<BuilderField
type="string"
path={`${path}.field_name`}
label="Field name"
tooltip={`Configures which key should be used in the location that the ${descriptor} is being injected into`}
/>
)}
</>
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -334,7 +334,7 @@ const InputItem = ({
isInferredInput: boolean;
}): JSX.Element => {
return (
<li className={styles.listItem}>
<li className={styles.listItem} key={input.key}>
<div className={styles.itemLabel}>{input.definition.title || input.key}</div>
<Button
className={styles.itemButton}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
import { useField } from "formik";

import GroupControls from "components/GroupControls";
import { ControlLabels } from "components/LabeledControl";

import { BuilderCard } from "./BuilderCard";
import { BuilderField } from "./BuilderField";
import { BuilderOneOf } from "./BuilderOneOf";
import { InjectRequestOptionFields } from "./InjectRequestOptionFields";
import { ToggleGroupField } from "./ToggleGroupField";

interface PaginationSectionProps {
streamFieldPath: (fieldPath: string) => string;
}

export const PaginationSection: React.FC<PaginationSectionProps> = ({ streamFieldPath }) => {
const [field, , helpers] = useField(streamFieldPath("paginator"));
const [pageSizeField] = useField(streamFieldPath("paginator.strategy.page_size"));
const [, , pageSizeOptionHelpers] = useField(streamFieldPath("paginator.pageSizeOption"));

const handleToggle = (newToggleValue: boolean) => {
if (newToggleValue) {
helpers.setValue({
strategy: {
type: "OffsetIncrement",
},
pageTokenOption: {
inject_into: "request_parameter",
},
});
} else {
helpers.setValue(undefined);
}
};
const toggledOn = field.value !== undefined;

const pageTokenOption = (
<GroupControls
label={
<ControlLabels
label="Page token option"
infoTooltipContent="Configures how the page token will be sent in requests to the source API"
/>
}
>
<InjectRequestOptionFields path={streamFieldPath("paginator.pageTokenOption")} descriptor="page token" />
</GroupControls>
);

const pageSizeOption = (
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A similar situation here - you need to set the page size if yopu want to use the page size option:
Screenshot 2022-12-21 at 14 44 47

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think putting the request options behind a checkbox like you suggested and looking for those to make page size optional or required would work.

<ToggleGroupField
label="Page size option"
tooltip="Configures how the page size will be sent in requests to the source API"
fieldPath={streamFieldPath("paginator.pageSizeOption")}
initialValues={{
inject_into: "request_parameter",
field_name: "",
}}
>
<InjectRequestOptionFields
path={streamFieldPath("paginator.pageSizeOption")}
descriptor="page size"
excludeInjectIntoValues={["path"]}
/>
</ToggleGroupField>
);

return (
<BuilderCard
toggleConfig={{
label: (
<ControlLabels
label="Pagination"
infoTooltipContent="Configure how pagination is handled by your connector"
/>
),
toggledOn,
onToggle: handleToggle,
}}
>
<BuilderOneOf
path={streamFieldPath("paginator.strategy")}
label="Mode"
tooltip="Pagination method to use for requests sent to the API"
options={[
{
label: "Offset Increment",
typeValue: "OffsetIncrement",
children: (
<>
<BuilderField
type="number"
path={streamFieldPath("paginator.strategy.page_size")}
label="Page size"
tooltip="Set the size of each page"
/>
{pageSizeOption}
{pageTokenOption}
</>
),
},
{
label: "Page Increment",
typeValue: "PageIncrement",
children: (
<>
<BuilderField
type="number"
path={streamFieldPath("paginator.strategy.page_size")}
label="Page size"
tooltip="Set the size of each page"
/>
<BuilderField
type="number"
path={streamFieldPath("paginator.strategy.start_from_page")}
label="Start from page"
tooltip="Page number to start requesting pages from"
optional
/>
{pageSizeOption}
{pageTokenOption}
</>
),
},
{
label: "Cursor Pagination",
typeValue: "CursorPagination",
children: (
<>
<BuilderField
type="string"
path={streamFieldPath("paginator.strategy.cursor_value")}
label="Cursor value"
tooltip="Value of the cursor to send in requests to the API"
/>
<BuilderField
type="string"
path={streamFieldPath("paginator.strategy.stop_condition")}
label="Stop condition"
tooltip="Condition that determines when to stop requesting further pages"
optional
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems like we have a problem with optional number fields - clearing them out will turn them into an empty string which will fail the validation.

/>
<BuilderField
type="number"
path={streamFieldPath("paginator.strategy.page_size")}
onChange={(newValue) => {
if (newValue === undefined || newValue === "") {
pageSizeOptionHelpers.setValue(undefined);
}
}}
label="Page size"
tooltip="Set the size of each page"
optional
/>
Comment on lines +143 to +154
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I decided to keep the page size field here outside of the pageSizeOption group, because it is technically allowed by the low code framework to set a page size here, but not set a page_size_option

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's slightly weird that starting to type into the page size input suddenly pops in the page size request options, but I think we can start like this.

{pageSizeField.value && pageSizeField.value !== "" && pageSizeOption}
{pageTokenOption}
</>
),
},
]}
/>
</BuilderCard>
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import { BuilderConfigView } from "./BuilderConfigView";
import { BuilderField } from "./BuilderField";
import { BuilderTitle } from "./BuilderTitle";
import { KeyValueListField } from "./KeyValueListField";
import { PaginationSection } from "./PaginationSection";
import styles from "./StreamConfigView.module.scss";

interface StreamConfigViewProps {
Expand Down Expand Up @@ -84,10 +85,18 @@ export const StreamConfigView: React.FC<StreamConfigViewProps> = ({ streamNum })
<BuilderField
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Seems like these are not required, should they be marked as optional in the UI?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes they should, good catch

type="array"
path={streamFieldPath("fieldPointer")}
label="Field Pointer"
label="Record selector"
tooltip="Pointer into the response that should be extracted as the final record"
/>
<BuilderField
type="array"
path={streamFieldPath("primaryKey")}
label="Primary key"
tooltip="Pointer into the response that should be used as the primary key when deduplicating records in the destination"
optional
/>
</BuilderCard>
<PaginationSection streamFieldPath={streamFieldPath} />
<BuilderCard>
<KeyValueListField
path={streamFieldPath("requestOptions.requestParameters")}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
@use "scss/variables";

.label {
display: flex;
align-items: center;
gap: variables.$spacing-md;
height: 34px;

label {
padding-bottom: 0;
}
}
Loading