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

feat(pci-load-balancer): Load Balancer 3AZ beta #14703

Open
wants to merge 16 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
16 commits
Select commit Hold shift + click to select a range
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
Expand Up @@ -41,5 +41,9 @@
"octavia_load_balancer_create_name_field_label": "Nom du Load Balancer",
"octavia_load_balancer_create_name_field_help": "Doit uniquement contenir des nombres, lettres, underscores, tirets ou points.",
"octavia_load_balancer_create_submit": "Créer un Load Balancer",
"octavia_load_balancer_create_banner": "Votre Load Balancer est en cours de création. Cela ne prendra que quelques minutes."
"octavia_load_balancer_create_banner": "Votre Load Balancer est en cours de création. Cela ne prendra que quelques minutes.",
"octavia_load_balancer_create_private_network": "Il n'existe aucun réseau privé dans la région sélectionnée. Assurez-vous de <link /> avant la création du Load Balancer.",
Copy link
Contributor

@SimonChaumet SimonChaumet Jan 14, 2025

Choose a reason for hiding this comment

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

is there a word missing in <link /> ?

"octavia_load_balancer_create_private_network_label": "créer un réseau privé",
"octavia_load_balancer_create_region_3az_price": "Load Balancer est actuellement en phase Beta dans la zone Paris (eu-west-par) et est donc gratuite durant cette période.",
"octavia_load_balancer_create_region_1az": "Avec l'arrivée prochaine des régions 3-AZ, le type de déploiement 'Régions' change de nom ! Une région 1-AZ (zone de disponibilité) regroupe un ou plusieurs datacenters localisés sur un même site. "
}
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,6 @@ export const useGetPrivateNetworkSubnets = (
],
queryFn: () => getPrivateNetworkSubnets(projectId, region, networkId),
enabled: !!projectId && !!region && !!networkId,
throwOnError: true,
});

const list = useMemo(() => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ export type TRegion = {
macroName: string;
microName: string;
continent: string;
type: string;
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 you can import from pci-common or improve this typing with the available types

};

// @TODO this part looks similar to what have been created in the region selector
Expand Down Expand Up @@ -61,6 +62,7 @@ export const useGetRegions = (
region.name,
)}`,
),
type: region.type,
});
});
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@ import { useTranslation } from 'react-i18next';
import { ODS_TEXT_LEVEL, ODS_TEXT_SIZE } from '@ovhcloud/ods-components';
import { ODS_THEME_COLOR_INTENT } from '@ovhcloud/ods-common-theming';
import { useEffect } from 'react';
import { useMedia } from 'react-use';
import { useCreateStore } from './store';
import {
useGetPrivateNetworkSubnets,
Expand All @@ -32,7 +31,6 @@ import { useGetFloatingIps } from '@/api/hook/useFloatingIps';
export default function CreatePage(): JSX.Element {
const { me } = useMe();
const { data: addons, isPending: isAddonsPending } = useGetAddons();
const isMobile = useMedia(`(max-width: 768px)`);

const projectHref = useProjectUrl('public-cloud');

Expand Down Expand Up @@ -161,7 +159,6 @@ export default function CreatePage(): JSX.Element {
isLoading={isRegionsPending}
regions={regions}
ovhSubsidiary={me?.ovhSubsidiary}
isMobile={isMobile}
/>
<IpStep
floatingIps={filteredFloatingIps}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,20 +3,15 @@ import { StepComponent, TStepProps } from '@ovh-ux/manager-react-components';
import { render, renderHook } from '@testing-library/react';
import { act } from 'react-dom/test-utils';
import { OsdsLink, OsdsSpinner } from '@ovhcloud/ods-components/react';
import {
ShapesInputComponent,
TShapesInputProps,
} from '@ovh-ux/manager-pci-common';
import { wrapper } from '@/wrapperRenders';
import { RegionStep, TRegionStepProps } from './RegionStep';
import { StepsEnum, TAddon, useCreateStore } from '@/pages/create/store';
import { StepsEnum, useCreateStore } from '@/pages/create/store';
import { REGION_AVAILABILITY_LINK } from '@/constants';
import { TRegion } from '@/api/hook/useRegions';
import { useTracking } from '../../hooks/useTracking';
import { LabelComponent } from '@/pages/create/steps/region/components/Label.component';
import { StackLabelComponent } from '@/pages/create/steps/region/components/StackLabel.component';
import { StackTitleComponent } from '@/pages/create/steps/region/components/StackTitle.component';
import { GroupLabelComponent } from '@/pages/create/steps/region/components/GroupLabel.component';

vi.mock('@ovh-ux/manager-pci-common', () => ({
useProject: vi.fn().mockReturnValue({ data: { project_id: 'project_id' } }),
RegionSelector: () => <div />,
}));

vi.mock('react-i18next', async () => {
const { ...rest } = await vi.importActual('react-i18next');
Expand Down Expand Up @@ -63,43 +58,27 @@ vi.mock('@ovh-ux/manager-react-components', async () => {
);
return {
...rest,
useProjectUrl: vi.fn(),
StepComponent: vi
.fn()
.mockImplementation(ActualStepComponent as typeof StepComponent),
};
});
vi.mock('@ovh-ux/manager-pci-common', async () => {
const {
ShapesInputComponent: ActualShapesInputComponent,
...rest
} = await vi.importActual('@ovh-ux/manager-pci-common');
return {
...rest,
ShapesInputComponent: vi
.fn()
.mockImplementation(
ActualShapesInputComponent as typeof ShapesInputComponent,
),
};
});

const renderStep = (
{
isLoading = false,
isMobile = false,
regions = undefined,
ovhSubsidiary = undefined,
}: Partial<TRegionStepProps> = {
isLoading: false,
isMobile: false,
regions: undefined,
ovhSubsidiary: undefined,
},
) =>
render(
<RegionStep
isLoading={isLoading}
isMobile={isMobile}
regions={regions}
ovhSubsidiary={ovhSubsidiary}
/>,
Expand Down Expand Up @@ -128,7 +107,7 @@ describe('RegionStep', () => {
const call = calls[calls.length - 1][0] as TStepProps;

expect(call.title).toBe(
'load-balancer/create | octavia_load_balancer_create_region_title',
'load-balancer/create,pci-common | octavia_load_balancer_create_region_title',
);

expect(call.isOpen).toBe(true);
Expand All @@ -140,25 +119,25 @@ describe('RegionStep', () => {
expect(call.order).toBe(2);

expect(call.next.label).toBe(
'pci-common | common_stepper_next_button_label',
'load-balancer/create,pci-common | pci-common:common_stepper_next_button_label',
);
expect(call.next.isDisabled).toBe(true);

expect(call.edit.label).toBe(
'pci-common | common_stepper_modify_this_step',
'load-balancer/create,pci-common | pci-common:common_stepper_modify_this_step',
);
});

it('should render intro', () => {
const { getByText } = renderStep();
expect(
getByText(
'load-balancer/create | octavia_load_balancer_create_region_intro',
'load-balancer/create,pci-common | octavia_load_balancer_create_region_intro',
),
).toBeInTheDocument();
expect(
getByText(
'load-balancer/create | octavia_load_balancer_create_region_link',
'load-balancer/create,pci-common | octavia_load_balancer_create_region_link',
),
).toBeInTheDocument();
});
Expand Down Expand Up @@ -199,225 +178,6 @@ describe('RegionStep', () => {

expect(getByTestId('spinner')).toBeInTheDocument();
});

describe('isLoading is false', () => {
beforeAll(() => {
((ShapesInputComponent as unknown) as Mock).mockImplementationOnce(
() => <div data-testid="input"></div>,
);
});
it('should show input if regions are not pending and data is defined', () => {
const regions: TRegion[] = [
{ name: 'test', isEnabled: true, continent: 'Europe' },
{ name: 'test2', isEnabled: true, continent: 'Asia' },
] as Array<TRegion>;

const { result } = renderStore();

act(() => result.current.set.addon({ code: 's' } as TAddon));

renderStep({ regions: new Map([['s', regions]]) });

const { calls } = (ShapesInputComponent as Mock).mock;
const call = calls[calls.length - 1][0] as TShapesInputProps<TRegion>;

expect(call.items).toEqual(regions);
expect(call.value).toBe(null);

expect(call.item.LabelComponent).toEqual(LabelComponent);
expect(call.item.getId({ name: 'test' } as TRegion)).toBe('test');
expect(call.item.isDisabled({ isEnabled: true } as TRegion)).toBe(
false,
);

expect(call.stack.by({ macroName: 'test' } as TRegion)).toBe('test');
expect(call.stack.by({} as TRegion)).toBe('');
expect(call.stack.LabelComponent).toEqual(StackLabelComponent);
expect(call.stack.TitleComponent).toEqual(StackTitleComponent);

expect(call.group.by({ continent: 'Europe' } as TRegion)).toBe(
'Europe',
);
expect(call.group.LabelComponent).toEqual(GroupLabelComponent);
});
});
});
});

describe('Actions', () => {
beforeAll(() => {
(ShapesInputComponent as Mock).mockImplementationOnce(() => (
<div data-testid="input"></div>
));
});

describe('next', () => {
describe('render', () => {
test('Next button should be disabled if region is not set', () => {
const { getByText } = renderStep();

const nextButton = getByText(
'pci-common | common_stepper_next_button_label',
);
expect(
nextButton.attributes.getNamedItem('disabled').value,
).toBeTruthy();
});

test('Next button should be enabled if region is set', () => {
(ShapesInputComponent as Mock).mockImplementationOnce(() => (
<div data-testid="input"></div>
));

const { result } = renderStore();
act(() => result.current.set.region({} as TRegion));

const { getByText } = renderStep();
const nextButton = getByText(
'pci-common | common_stepper_next_button_label',
);
expect(
nextButton.attributes.getNamedItem('disabled')?.value,
).toBeFalsy();
});
});
describe('click', () => {
beforeEach(() => {
const { result } = renderStore();
act(() => {
result.current.set.region({} as TRegion);
});
});
it('Should track on next click', async () => {
(ShapesInputComponent as Mock).mockImplementationOnce(() => (
<div data-testid="input"></div>
));
const trackStepSpy = vi.fn();

(useTracking as Mock).mockImplementationOnce(() => ({
trackStep: trackStepSpy,
}));

const { result } = renderStore();

act(() => {
result.current.set.region({} as TRegion);
});

const { getByText } = renderStep();

const nextButton = getByText(
'pci-common | common_stepper_next_button_label',
);

act(() => nextButton.click());

expect(trackStepSpy).toHaveBeenCalledWith(2);
});

it('Should prepare next step on next click', async () => {
(ShapesInputComponent as Mock).mockImplementationOnce(() => (
<div data-testid="input"></div>
));

const { result } = renderStore();

act(() => {
result.current.set.region({} as TRegion);
});

const { getByText } = renderStep();

const nextButton = getByText(
'pci-common | common_stepper_next_button_label',
);

const { check, lock, open } = { ...result.current };

result.current.check = vi.fn();
result.current.lock = vi.fn();
result.current.open = vi.fn();

act(() => nextButton.click());

expect(result.current.check).toHaveBeenCalledWith(StepsEnum.REGION);
expect(result.current.lock).toHaveBeenCalledWith(StepsEnum.REGION);
expect(result.current.open).toHaveBeenCalledWith(StepsEnum.IP);

result.current.check = check;
result.current.lock = lock;
result.current.open = open;
});
});
});

describe('edit', () => {
describe('render', () => {
it('should not display if step is not checked', () => {
const { queryByText } = renderStep();

const editButton = queryByText(
'pci-common | common_stepper_modify_this_step',
);
expect(editButton).not.toBeInTheDocument();
});
it('should have the right label', () => {
const { result } = renderStore();
act(() => result.current.lock(StepsEnum.REGION));

((OsdsLink as unknown) as Mock).mockImplementation(
({ href, ...props }) => (
<a href={href} data-testid="link" {...props}></a>
),
);

const { queryByText } = renderStep();

const editButton = queryByText(
'pci-common | common_stepper_modify_this_step',
);

expect(editButton).toBeInTheDocument();
});
});

describe('click', () => {
it('should prepare steps on click', () => {
const { result } = renderStore();

const { unlock, uncheck, open, reset } = { ...result.current };

act(() => result.current.lock(StepsEnum.REGION));

result.current.unlock = vi.fn();
result.current.uncheck = vi.fn();
result.current.open = vi.fn();
result.current.reset = vi.fn();

const { getByText } = renderStep();

const editButton = getByText(
'pci-common | common_stepper_modify_this_step',
);

act(() => editButton.click());

expect(result.current.unlock).toHaveBeenCalledWith(StepsEnum.REGION);
expect(result.current.uncheck).toHaveBeenCalledWith(StepsEnum.REGION);
expect(result.current.open).toHaveBeenCalledWith(StepsEnum.REGION);
expect(result.current.reset).toHaveBeenCalledWith(
StepsEnum.IP,
StepsEnum.NETWORK,
StepsEnum.INSTANCE,
StepsEnum.NAME,
);

result.current.unlock = unlock;
result.current.uncheck = uncheck;
result.current.open = open;
result.current.reset = reset;
});
});
});
});
});
Loading
Loading