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

[GEN-1977]: improve SSE, and add SSE to UI E2E tests #1991

Merged
merged 4 commits into from
Dec 15, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
3 changes: 3 additions & 0 deletions frontend/webapp/app/(overview)/overview/page.tsx
Original file line number Diff line number Diff line change
@@ -1,13 +1,16 @@
'use client';
import React from 'react';
import dynamic from 'next/dynamic';
import { useSSE } from '@/hooks';

const ToastList = dynamic(() => import('@/components/notification/toast-list'), { ssr: false });
const AllDrawers = dynamic(() => import('@/components/overview/all-drawers'), { ssr: false });
const AllModals = dynamic(() => import('@/components/overview/all-modals'), { ssr: false });
const OverviewDataFlowContainer = dynamic(() => import('@/containers/main/overview/overview-data-flow'), { ssr: false });

export default function MainPage() {
useSSE();

return (
<>
<ToastList />
Expand Down
3 changes: 0 additions & 3 deletions frontend/webapp/app/layout.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
'use client';
import './globals.css';
import React from 'react';
import { useSSE } from '@/hooks';
import { METADATA } from '@/utils';
import { ApolloWrapper } from '@/lib';
import { ThemeProviderWrapper } from '@/styles';
Expand All @@ -15,8 +14,6 @@ const LAYOUT_STYLE: React.CSSProperties = {
};

export default function RootLayout({ children }: { children: React.ReactNode }) {
useSSE();

return (
<html lang='en'>
<head>
Expand Down
14 changes: 8 additions & 6 deletions frontend/webapp/app/page.tsx
Original file line number Diff line number Diff line change
@@ -1,20 +1,22 @@
'use client';
import { useEffect } from 'react';
import { useConfig } from '@/hooks';
import { CenterThis } from '@/styles';
import { ROUTES, CONFIG } from '@/utils';
import { NOTIFICATION_TYPE } from '@/types';
import { useRouter } from 'next/navigation';
import { useNotify, useConfig } from '@/hooks';
import { useNotificationStore } from '@/store';
import { FadeLoader } from '@/reuseable-components';
import { ROUTES, CONFIG, NOTIFICATION } from '@/utils';
import { CenterThis } from '@/styles';

export default function App() {
const router = useRouter();
const notify = useNotify();
const { data, error } = useConfig();
const { addNotification } = useNotificationStore();

useEffect(() => {
if (error) {
notify({
type: NOTIFICATION.ERROR,
addNotification({
type: NOTIFICATION_TYPE.ERROR,
title: error.name,
message: error.message,
});
Expand Down
3 changes: 2 additions & 1 deletion frontend/webapp/components/modals/cancel-warning/index.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import React from 'react';
import { NOTIFICATION_TYPE } from '@/types';
import { WarningModal } from '@/reuseable-components';

interface Props {
Expand All @@ -18,7 +19,7 @@ const CancelWarning: React.FC<Props> = ({ isOpen, noOverlay, name, onApprove, on
description='Are you sure you want to cancel?'
approveButton={{
text: 'Confirm',
variant: 'warning',
variant: NOTIFICATION_TYPE.WARNING,
onClick: onApprove,
}}
denyButton={{
Expand Down
4 changes: 2 additions & 2 deletions frontend/webapp/components/modals/delete-warning/index.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import React from 'react';
import { OVERVIEW_ENTITY_TYPES } from '@/types';
import { WarningModal } from '@/reuseable-components';
import { NOTIFICATION_TYPE, OVERVIEW_ENTITY_TYPES } from '@/types';

interface Props {
isOpen: boolean;
Expand All @@ -24,7 +24,7 @@ const DeleteWarning: React.FC<Props> = ({ isOpen, noOverlay, name, type, isLastI
note={
isLastItem
? {
type: 'warning',
type: NOTIFICATION_TYPE.WARNING,
title: `You're about to ${actionText} the last ${type}`,
message: 'This will break your pipeline!',
}
Expand Down
12 changes: 6 additions & 6 deletions frontend/webapp/components/notification/notification-manager.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { useNotificationStore } from '@/store';
import { ACTION, getStatusIcon } from '@/utils';
import { useOnClickOutside, useTimeAgo } from '@/hooks';
import theme, { hexPercentValues } from '@/styles/theme';
import type { Notification, NotificationType } from '@/types';
import { NOTIFICATION_TYPE, type Notification } from '@/types';
import { IconButton, NoDataFound, Text } from '@/reuseable-components';

const RelativeContainer = styled.div`
Expand Down Expand Up @@ -82,14 +82,14 @@ export const NotificationManager = () => {

return (
<RelativeContainer ref={containerRef}>
<IconButton onClick={toggleOpen} withPing={!!unseenCount} pingColor={theme.colors.orange_og}>
<IconButton data-id='notif-manager-button' onClick={toggleOpen} withPing={!!unseenCount} pingColor={theme.colors.orange_og}>
<Image src='/icons/common/notification.svg' alt='logo' width={16} height={16} />
</IconButton>

{isOpen && (
<AbsoluteContainer>
<AbsoluteContainer data-id='notif-manager-content'>
<PopupHeader>
<Text size={20}>Notifications</Text>{' '}
<Text size={20}>Notifications</Text>
{!!unseenCount && (
<NewCount size={12} family='secondary'>
{unseenCount} new
Expand Down Expand Up @@ -126,7 +126,7 @@ const NotifCard = styled.div`
}
`;

const StatusIcon = styled.div<{ $type: NotificationType }>`
const StatusIcon = styled.div<{ $type: NOTIFICATION_TYPE }>`
background-color: ${({ $type, theme }) => theme.text[$type] + hexPercentValues['012']};
border-radius: 8px;
width: 36px;
Expand Down Expand Up @@ -175,7 +175,7 @@ const NotificationListItem: React.FC<Notification & { onClick: () => void }> = (
}
}}
>
<StatusIcon $type={isDeleted ? 'error' : type}>
<StatusIcon $type={isDeleted ? NOTIFICATION_TYPE.ERROR : type}>
<Image src={isDeleted ? '/icons/common/trash.svg' : getStatusIcon(type)} alt='status' width={16} height={16} />
</StatusIcon>

Expand Down
16 changes: 8 additions & 8 deletions frontend/webapp/components/overview/add-entity/index.tsx
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
import React, { useState, useRef } from 'react';
import Image from 'next/image';
import theme from '@/styles/theme';
import { useOnClickOutside } from '@/hooks';
import React, { useState, useRef } from 'react';
import { useModalStore } from '@/store';
import styled, { css } from 'styled-components';
import { useBooleanStore, useModalStore } from '@/store';
import { DropdownOption, OVERVIEW_ENTITY_TYPES } from '@/types';
import { useComputePlatform, useOnClickOutside } from '@/hooks';
import { Button, FadeLoader, Text } from '@/reuseable-components';
import { type DropdownOption, OVERVIEW_ENTITY_TYPES } from '@/types';

// Styled components for the dropdown UI
const Container = styled.div`
Expand Down Expand Up @@ -65,13 +65,13 @@ const DEFAULT_OPTIONS: DropdownOption[] = [
{ id: OVERVIEW_ENTITY_TYPES.DESTINATION, value: 'Destination' },
];

interface AddEntityButtonDropdownProps {
interface Props {
options?: DropdownOption[];
placeholder?: string;
}

const AddEntity: React.FC<AddEntityButtonDropdownProps> = ({ options = DEFAULT_OPTIONS, placeholder = 'ADD...' }) => {
const { isPolling } = useBooleanStore();
const AddEntity: React.FC<Props> = ({ options = DEFAULT_OPTIONS, placeholder = 'ADD...' }) => {
const { loading } = useComputePlatform();
const { currentModal, setCurrentModal } = useModalStore();

const [isDropdownOpen, setIsDropdownOpen] = useState(false);
Expand All @@ -91,7 +91,7 @@ const AddEntity: React.FC<AddEntityButtonDropdownProps> = ({ options = DEFAULT_O
return (
<Container ref={dropdownRef}>
<StyledButton data-id='add-entity' onClick={handleToggle}>
{isPolling ? <FadeLoader color={theme.colors.primary} /> : <Image src='/icons/common/plus-black.svg' width={16} height={16} alt='Add' />}
{loading ? <FadeLoader color={theme.colors.primary} /> : <Image src='/icons/common/plus-black.svg' width={16} height={16} alt='Add' />}
<ButtonText size={14}>{placeholder}</ButtonText>
</StyledButton>

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,16 @@ import React, { useState } from 'react';
import Image from 'next/image';
import { ROUTES } from '@/utils';
import theme from '@/styles/theme';
import { CenterThis } from '@/styles';
import { useAppStore } from '@/store';
import styled from 'styled-components';
import { SetupHeader } from '@/components';
import { useRouter } from 'next/navigation';
import { useDestinationCRUD, useSourceCRUD } from '@/hooks';
import { NOTIFICATION_TYPE } from '@/types';
import { DestinationModal } from '../destination-modal';
import { useDestinationCRUD, useSourceCRUD } from '@/hooks';
import { ConfiguredDestinationsList } from './configured-destinations-list';
import { Button, FadeLoader, NotificationNote, SectionTitle, Text } from '@/reuseable-components';
import { CenterThis } from '@/styles';

const ContentWrapper = styled.div`
width: 640px;
Expand Down Expand Up @@ -93,7 +94,7 @@ export function AddDestinationContainer() {
{!isLoading && isSourcesListEmpty() && (
<NotificationNoteWrapper>
<NotificationNote
type='warning'
type={NOTIFICATION_TYPE.WARNING}
message='No sources selected. Please go back to select sources.'
action={{
label: 'Select sources',
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
import React, { Dispatch, SetStateAction, useMemo, useState } from 'react';
import styled from 'styled-components';
import { SignalUppercase } from '@/utils';
import { TestConnection } from './test-connection';
import { type ConnectionStatus, TestConnection } from './test-connection';
import { DestinationDynamicFields } from './dynamic-fields';
import type { DestinationInput, DestinationTypeItem, DynamicField } from '@/types';
import { Divider, Input, MonitoringCheckboxes, NotificationNote, SectionTitle } from '@/reuseable-components';
import { NOTIFICATION_TYPE, type DestinationInput, type DestinationTypeItem, type DynamicField } from '@/types';

interface Props {
isUpdate?: boolean;
Expand Down Expand Up @@ -34,7 +34,7 @@ export function DestinationFormBody({ isUpdate, destination, formData, formError
const { supportedSignals, testConnectionSupported, displayName } = destination || {};

const [isFormDirty, setIsFormDirty] = useState(false);
const [connectionStatus, setConnectionStatus] = useState<'success' | 'error'>();
const [connectionStatus, setConnectionStatus] = useState<ConnectionStatus>();

const dirtyForm = () => {
setIsFormDirty(true);
Expand Down Expand Up @@ -87,11 +87,11 @@ export function DestinationFormBody({ isUpdate, destination, formData, formError
status={connectionStatus}
onError={() => {
setIsFormDirty(false);
setConnectionStatus('error');
setConnectionStatus(NOTIFICATION_TYPE.ERROR);
}}
onSuccess={() => {
setIsFormDirty(false);
setConnectionStatus('success');
setConnectionStatus(NOTIFICATION_TYPE.SUCCESS);
}}
validateForm={validateForm}
/>
Expand All @@ -101,9 +101,9 @@ export function DestinationFormBody({ isUpdate, destination, formData, formError

{testConnectionSupported && (
<NotesWrapper>
{connectionStatus === 'error' && <NotificationNote type='error' message='Connection failed. Please check your input and try again.' />}
{connectionStatus === 'success' && <NotificationNote type='success' message='Connection succeeded.' />}
{!connectionStatus && <NotificationNote type='default' message={`Odigos autocompleted ${displayName} connection details.`} />}
{connectionStatus === NOTIFICATION_TYPE.ERROR && <NotificationNote type={NOTIFICATION_TYPE.ERROR} message='Connection failed. Please check your input and try again.' />}
{connectionStatus === NOTIFICATION_TYPE.SUCCESS && <NotificationNote type={NOTIFICATION_TYPE.SUCCESS} message='Connection succeeded.' />}
{!connectionStatus && <NotificationNote type={NOTIFICATION_TYPE.DEFAULT} message={`Odigos autocompleted ${displayName} connection details.`} />}
</NotesWrapper>
)}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,22 +3,22 @@ import Image from 'next/image';
import theme from '@/styles/theme';
import { getStatusIcon } from '@/utils';
import { useTestConnection } from '@/hooks';
import type { DestinationInput } from '@/types';
import styled, { css } from 'styled-components';
import { Button, FadeLoader, Text } from '@/reuseable-components';
import { type DestinationInput, NOTIFICATION_TYPE } from '@/types';

type Status = 'success' | 'error';
export type ConnectionStatus = NOTIFICATION_TYPE.SUCCESS | NOTIFICATION_TYPE.ERROR;

interface Props {
destination: DestinationInput;
disabled: boolean;
status?: Status;
status?: ConnectionStatus;
onError: () => void;
onSuccess: () => void;
validateForm: () => boolean;
}

const ActionButton = styled(Button)<{ $status?: Status }>`
const ActionButton = styled(Button)<{ $status?: ConnectionStatus }>`
display: flex;
align-items: center;
gap: 8px;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,14 @@ import React, { useMemo, useState } from 'react';
import buildCard from './build-card';
import { RuleFormBody } from '../';
import styled from 'styled-components';
import { useDrawerStore } from '@/store';
import { DataCard } from '@/reuseable-components';
import buildDrawerItem from './build-drawer-item';
import { RULE_OPTIONS } from '../rule-modal/rule-options';
import OverviewDrawer from '../../overview/overview-drawer';
import { ACTION, DATA_CARDS, FORM_ALERTS, getRuleIcon, NOTIFICATION } from '@/utils';
import { useInstrumentationRuleCRUD, useInstrumentationRuleFormData, useNotify } from '@/hooks';
import { InstrumentationRuleType, OVERVIEW_ENTITY_TYPES, type InstrumentationRuleSpec } from '@/types';
import { useDrawerStore, useNotificationStore } from '@/store';
import { ACTION, DATA_CARDS, FORM_ALERTS, getRuleIcon } from '@/utils';
import { useInstrumentationRuleCRUD, useInstrumentationRuleFormData } from '@/hooks';
import { InstrumentationRuleType, NOTIFICATION_TYPE, OVERVIEW_ENTITY_TYPES, type InstrumentationRuleSpec } from '@/types';

interface Props {}

Expand All @@ -22,7 +22,7 @@ const FormContainer = styled.div`
`;

export const RuleDrawer: React.FC<Props> = () => {
const notify = useNotify();
const { addNotification } = useNotificationStore();
const { selectedItem, setSelectedItem } = useDrawerStore();
const { formData, formErrors, handleFormChange, resetFormData, validateForm, loadFormWithDrawerItem } = useInstrumentationRuleFormData();

Expand Down Expand Up @@ -74,7 +74,7 @@ export const RuleDrawer: React.FC<Props> = () => {

const handleEdit = (bool?: boolean) => {
if (item.type === InstrumentationRuleType.UNKNOWN_TYPE) {
notify({ type: NOTIFICATION.WARNING, title: FORM_ALERTS.FORBIDDEN, message: FORM_ALERTS.CANNOT_EDIT_RULE, crdType: OVERVIEW_ENTITY_TYPES.RULE, target: id });
addNotification({ type: NOTIFICATION_TYPE.WARNING, title: FORM_ALERTS.FORBIDDEN, message: FORM_ALERTS.CANNOT_EDIT_RULE, crdType: OVERVIEW_ENTITY_TYPES.RULE, target: id });
} else {
setIsEditing(typeof bool === 'boolean' ? bool : true);
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import React, { useState } from 'react';
import { ACTION } from '@/utils';
import { RuleFormBody } from '../';
import { NOTIFICATION_TYPE } from '@/types';
import { CenterThis, ModalBody } from '@/styles';
import { RULE_OPTIONS, RuleOption } from './rule-options';
import { useInstrumentationRuleCRUD, useInstrumentationRuleFormData, useKeyDown } from '@/hooks';
Expand Down Expand Up @@ -58,7 +59,7 @@ export const RuleModal: React.FC<Props> = ({ isOpen, onClose }) => {
>
<ModalBody>
<SectionTitle title='Define Instrumentation Rule' description='Define how telemetry is recorded from your application. Choose a rule type and configure the details.' />
<NotificationNote type='info' message='We currently support one rule. We’ll be adding new rule types in the near future.' style={{ marginTop: '24px' }} />
<NotificationNote type={NOTIFICATION_TYPE.INFO} message='We currently support one rule. We’ll be adding new rule types in the near future.' style={{ marginTop: '24px' }} />
<AutocompleteInput disabled options={RULE_OPTIONS} selectedOption={selectedItem} onOptionSelect={handleSelect} style={{ marginTop: '12px' }} />

{!!selectedItem?.type ? (
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,15 +35,9 @@ export default function OverviewDataFlowContainer() {
const positions = useMemo(() => getNodePositions({ containerWidth }), [containerWidth]);

const { metrics } = useMetrics();
const { data, filteredData, startPolling } = useComputePlatform();
const { data, filteredData } = useComputePlatform();
const unfilteredCounts = useMemo(() => getEntityCounts({ computePlatform: data?.computePlatform }), [data]);

useEffect(() => {
// this is to start polling on component mount in an attempt to fix any initial errors with sources/destinations
if (!!data?.computePlatform.k8sActualSources.length || !!data?.computePlatform.destinations.length) startPolling();
// only on-mount, if we include "data" this will trigger on every refetch, causing an infinite loop
}, []);

const ruleNodes = useMemo(
() => buildRuleNodes({ entities: filteredData?.computePlatform.instrumentationRules || [], positions, unfilteredCounts }),
[filteredData?.computePlatform.instrumentationRules, positions, unfilteredCounts],
Expand Down
8 changes: 7 additions & 1 deletion frontend/webapp/cypress/constants/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,9 @@ export const DATA_IDS = {
TITLE: '[data-id=title]',
SOURCE_TITLE: '[data-id=sourceName]',
CHECKBOX: '[data-id=checkbox]',

NOTIF_MANAGER_BUTTON: '[data-id=notif-manager-button]',
NOTIF_MANAGER_CONTENR: '[data-id=notif-manager-content]',
};

export const BUTTONS = {
Expand All @@ -88,12 +91,15 @@ const CYPRESS_TEST = 'Cypress Test';
export const TEXTS = {
UPDATED_NAME: CYPRESS_TEST,

NO_RESOURCES: (namespace: string) => `No resources found in ${namespace} namespace.`,

SOURCE_WARN_MODAL_TITLE: 'Uninstrument 5 sources',
SOURCE_WARN_MODAL_NOTE: "You're about to uninstrument the last source",
DESTINATION_WARN_MODAL_TITLE: `Delete destination (${CYPRESS_TEST})`,
DESTINATION_WARN_MODAL_NOTE: "You're about to delete the last destination",
ACTION_WARN_MODAL_TITLE: `Delete action (${CYPRESS_TEST})`,
INSTRUMENTATION_RULE_WARN_MODAL_TITLE: `Delete rule (${CYPRESS_TEST})`,

NO_RESOURCES: (namespace: string) => `No resources found in ${namespace} namespace.`,
NOTIF_SOURCES_CREATED: 'successfully added 5 sources',
NOTIF_SOURCES_DELETED: 'successfully deleted 5 sources',
};
Loading
Loading