Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
8598122
Feature: Add permissions display to Feast UI lineage visualization
devin-ai-integration[bot] May 3, 2025
8d947de
Format code with Prettier
devin-ai-integration[bot] May 4, 2025
fc1ea5c
Fix TypeScript errors in getEntityPermissions function
devin-ai-integration[bot] May 4, 2025
52a6e0c
Add permissions for zipcode_features, zipcode_source, and model_v1
devin-ai-integration[bot] May 4, 2025
15eeab0
Add permissions page and display permissions on feature service page
devin-ai-integration[bot] May 4, 2025
74f5448
Add mock permissions data for development
devin-ai-integration[bot] May 4, 2025
bf21399
Add permissions display to feature view pages
devin-ai-integration[bot] May 4, 2025
d550f4e
Add permissions display to entity pages
devin-ai-integration[bot] May 4, 2025
3211d58
Add permissions display to data source pages
devin-ai-integration[bot] May 4, 2025
5ed769f
Fix permissions implementation with separate apply_permissions.py script
devin-ai-integration[bot] May 4, 2025
f6c9624
Fix: Comment out NPM_TOKEN requirement in .npmrc for easier local dev…
devin-ai-integration[bot] May 4, 2025
f476a69
Fix: Add permissions display to features page and fix icon loading er…
devin-ai-integration[bot] May 4, 2025
2e15860
Update: EUI package to fix icon loading errors
devin-ai-integration[bot] May 4, 2025
ca8292b
Fix: Remove theme import to resolve icon loading errors
devin-ai-integration[bot] May 4, 2025
7f48e99
Fix: Update package dependencies to resolve icon loading errors
devin-ai-integration[bot] May 4, 2025
4fded55
Add Home | Lineage link and update permission utils
devin-ai-integration[bot] May 4, 2025
faed8d8
Add RAG project with feature views and permissions
devin-ai-integration[bot] May 4, 2025
8a38cfd
Update feature_store.yaml for RAG project
devin-ai-integration[bot] May 4, 2025
0c0d27d
Update: Move Home and Lineage links to sidebar navigation
devin-ai-integration[bot] May 4, 2025
6f5fd5e
Update: Remove Home | Lineage links from top of page
devin-ai-integration[bot] May 4, 2025
0b80676
Add: RAG data files for document embeddings and metadata
devin-ai-integration[bot] May 4, 2025
6d0d655
Fix: Update sidebar navigation to show Home | Lineage as a single item
devin-ai-integration[bot] May 4, 2025
2c3b195
Fix: Update Home | Lineage navigation to appear as a single line with…
devin-ai-integration[bot] May 4, 2025
e966f07
Fix: Update sidebar navigation to fix React Hook error
devin-ai-integration[bot] May 4, 2025
ccdf949
Fix: Update Lineage link to use feature service demo tab URL format
devin-ai-integration[bot] May 4, 2025
88c50de
Fix: Update Lineage link to properly redirect to the lineage tab
devin-ai-integration[bot] May 4, 2025
f211d2a
Update: Add data files to .gitignore
devin-ai-integration[bot] May 4, 2025
3c84796
Update: Move Lineage to its own page under Resources in sidebar
devin-ai-integration[bot] May 4, 2025
859b957
Update: Remove Home hyperlink and Lineage tab from home page
devin-ai-integration[bot] May 4, 2025
c7dfb24
Format: Run yarn format to ensure code follows project standards
devin-ai-integration[bot] May 4, 2025
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
Next Next commit
Feature: Add permissions display to Feast UI lineage visualization
Co-Authored-By: Francisco Javier Arceo <arceofrancisco@gmail.com>
  • Loading branch information
commit 85981225adb3bb796f19064e8ccf8ca62ea90eaf
67 changes: 66 additions & 1 deletion ui/src/components/RegistryVisualization.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -15,11 +15,12 @@ import {
} from "reactflow";
import "reactflow/dist/style.css";
import dagre from "dagre";
import { EuiPanel, EuiTitle, EuiSpacer, EuiLoadingSpinner } from "@elastic/eui";
import { EuiPanel, EuiTitle, EuiSpacer, EuiLoadingSpinner, EuiToolTip } from "@elastic/eui";
import { FEAST_FCO_TYPES } from "../parsers/types";
import { EntityRelation } from "../parsers/parseEntityRelationships";
import { feast } from "../protos";
import { useTheme } from "../contexts/ThemeContext";
import { formatPermissions, getEntityPermissions } from "../utils/permissionUtils";

const edgeAnimationStyle = `
@keyframes dashdraw {
Expand Down Expand Up @@ -53,6 +54,7 @@ interface NodeData {
label: string;
type: FEAST_FCO_TYPES;
metadata: any;
permissions?: any[]; // Add permissions field
}

const getNodeColor = (type: FEAST_FCO_TYPES) => {
Expand Down Expand Up @@ -107,6 +109,7 @@ const CustomNode = ({ data }: { data: NodeData }) => {
const lightColor = getLightNodeColor(data.type);
const icon = getNodeIcon(data.type);
const [isHovered, setIsHovered] = useState(false);
const hasPermissions = data.permissions && data.permissions.length > 0;

const handleClick = () => {
let path;
Expand All @@ -129,6 +132,10 @@ const CustomNode = ({ data }: { data: NodeData }) => {
navigate(path);
};

const permissionsTooltipContent = hasPermissions
? formatPermissions(data.permissions)
: "No permissions set";

return (
<div
style={{
Expand Down Expand Up @@ -167,6 +174,30 @@ const CustomNode = ({ data }: { data: NodeData }) => {
</div>
)}

{/* Permissions indicator */}
{hasPermissions && (
<EuiToolTip
position="top"
content={<pre style={{ margin: 0 }}>{permissionsTooltipContent}</pre>}
>
<div
style={{
position: "absolute",
top: 0,
left: 0,
backgroundColor: "#5a7be0",
color: "white",
padding: "2px 8px",
fontSize: "12px",
borderBottomRightRadius: "6px",
zIndex: 5,
}}
>
P
</div>
</EuiToolTip>
)}

<Handle
type="target"
position={Position.Left}
Expand Down Expand Up @@ -441,6 +472,7 @@ const Legend = () => {
const registryToFlow = (
objects: feast.core.Registry,
relationships: EntityRelation[],
permissions?: any[]
) => {
const nodes: Node[] = [];
const edges: Edge[] = [];
Expand All @@ -453,6 +485,11 @@ const registryToFlow = (
label: fs.spec?.name,
type: FEAST_FCO_TYPES.featureService,
metadata: fs,
permissions: permissions ? getEntityPermissions(
permissions,
FEAST_FCO_TYPES.featureService,
fs.spec?.name
) : [],
},
position: { x: 0, y: 0 },
});
Expand All @@ -466,6 +503,11 @@ const registryToFlow = (
label: fv.spec?.name,
type: FEAST_FCO_TYPES.featureView,
metadata: fv,
permissions: permissions ? getEntityPermissions(
permissions,
FEAST_FCO_TYPES.featureView,
fv.spec?.name
) : [],
},
position: { x: 0, y: 0 },
});
Expand All @@ -479,6 +521,11 @@ const registryToFlow = (
label: odfv.spec?.name,
type: FEAST_FCO_TYPES.featureView,
metadata: odfv,
permissions: permissions ? getEntityPermissions(
permissions,
FEAST_FCO_TYPES.featureView,
odfv.spec?.name
) : [],
},
position: { x: 0, y: 0 },
});
Expand All @@ -492,6 +539,11 @@ const registryToFlow = (
label: sfv.spec?.name,
type: FEAST_FCO_TYPES.featureView,
metadata: sfv,
permissions: permissions ? getEntityPermissions(
permissions,
FEAST_FCO_TYPES.featureView,
sfv.spec?.name
) : [],
},
position: { x: 0, y: 0 },
});
Expand All @@ -505,6 +557,11 @@ const registryToFlow = (
label: entity.spec?.name,
type: FEAST_FCO_TYPES.entity,
metadata: entity,
permissions: permissions ? getEntityPermissions(
permissions,
FEAST_FCO_TYPES.entity,
entity.spec?.name
) : [],
},
position: { x: 0, y: 0 },
});
Expand Down Expand Up @@ -535,6 +592,11 @@ const registryToFlow = (
label: dsName,
type: FEAST_FCO_TYPES.dataSource,
metadata: { name: dsName },
permissions: permissions ? getEntityPermissions(
permissions,
FEAST_FCO_TYPES.dataSource,
dsName
) : [],
},
position: { x: 0, y: 0 },
});
Expand Down Expand Up @@ -590,13 +652,15 @@ interface RegistryVisualizationProps {
relationships: EntityRelation[];
indirectRelationships: EntityRelation[];
filterNode?: { type: FEAST_FCO_TYPES; name: string };
permissions?: any[]; // Add permissions field
}

const RegistryVisualization: React.FC<RegistryVisualizationProps> = ({
registryData,
relationships,
indirectRelationships,
filterNode,
permissions,
}) => {
const [nodes, setNodes, onNodesChange] = useNodesState([]);
const [edges, setEdges, onEdgesChange] = useEdgesState([]);
Expand Down Expand Up @@ -668,6 +732,7 @@ const RegistryVisualization: React.FC<RegistryVisualizationProps> = ({
const { nodes: initialNodes, edges: initialEdges } = registryToFlow(
registryData,
validRelationships,
permissions
);

const { nodes: layoutedNodes, edges: layoutedEdges } =
Expand Down
27 changes: 27 additions & 0 deletions ui/src/components/RegistryVisualizationTab.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,14 @@ import useLoadRegistry from "../queries/useLoadRegistry";
import RegistryPathContext from "../contexts/RegistryPathContext";
import RegistryVisualization from "./RegistryVisualization";
import { FEAST_FCO_TYPES } from "../parsers/types";
import { filterPermissionsByAction } from "../utils/permissionUtils";

const RegistryVisualizationTab = () => {
const registryUrl = useContext(RegistryPathContext);
const { isLoading, isSuccess, isError, data } = useLoadRegistry(registryUrl);
const [selectedObjectType, setSelectedObjectType] = useState("");
const [selectedObjectName, setSelectedObjectName] = useState("");
const [selectedPermissionAction, setSelectedPermissionAction] = useState("");

const getObjectOptions = (objects: any, type: string) => {
switch (type) {
Expand Down Expand Up @@ -114,11 +116,36 @@ const RegistryVisualizationTab = () => {
/>
</EuiFormRow>
</EuiFlexItem>
<EuiFlexItem grow={false} style={{ width: 300 }}>
<EuiFormRow label="Filter by permissions">
<EuiSelect
options={[
{ value: "", text: "All" },
{ value: "CREATE", text: "CREATE" },
{ value: "DESCRIBE", text: "DESCRIBE" },
{ value: "UPDATE", text: "UPDATE" },
{ value: "DELETE", text: "DELETE" },
{ value: "READ_ONLINE", text: "READ_ONLINE" },
{ value: "READ_OFFLINE", text: "READ_OFFLINE" },
{ value: "WRITE_ONLINE", text: "WRITE_ONLINE" },
{ value: "WRITE_OFFLINE", text: "WRITE_OFFLINE" },
]}
value={selectedPermissionAction}
onChange={(e) => setSelectedPermissionAction(e.target.value)}
aria-label="Filter by permissions"
/>
</EuiFormRow>
</EuiFlexItem>
</EuiFlexGroup>
<RegistryVisualization
registryData={data.objects}
relationships={data.relationships}
indirectRelationships={data.indirectRelationships}
permissions={
selectedPermissionAction
? filterPermissionsByAction(data.permissions, selectedPermissionAction)
: data.permissions
}
filterNode={
selectedObjectType && selectedObjectName
? {
Expand Down
2 changes: 2 additions & 0 deletions ui/src/queries/useLoadRegistry.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ interface FeatureStoreAllData {
mergedFVList: genericFVType[];
indirectRelationships: EntityRelation[];
allFeatures: Feature[];
permissions?: any[]; // Add permissions field
}

interface Feature {
Expand Down Expand Up @@ -79,6 +80,7 @@ const useLoadRegistry = (url: string) => {
relationships,
indirectRelationships,
allFeatures,
permissions: objects.permissions || [], // Add permissions to the returned data
};
});
},
Expand Down
110 changes: 110 additions & 0 deletions ui/src/utils/permissionUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
import { FEAST_FCO_TYPES } from "../parsers/types";
import { feast } from "../protos";

/**
* Get permissions for a specific entity
* @param permissions List of all permissions
* @param entityType Type of the entity
* @param entityName Name of the entity
* @returns List of permissions that apply to the entity
*/
export const getEntityPermissions = (
permissions: any[] | undefined,
entityType: FEAST_FCO_TYPES,
entityName: string
): any[] => {
if (!permissions || permissions.length === 0) {
return [];
}

return permissions.filter((permission) => {
const matchesType = permission.spec?.types?.includes(
getPermissionType(entityType)
);

const matchesName =
permission.spec?.name_patterns?.length === 0 ||
permission.spec?.name_patterns?.some((pattern: string) => {
const regex = new RegExp(pattern);
return regex.test(entityName);
});

return matchesType && matchesName;
});
};

/**
* Convert FEAST_FCO_TYPES to permission type value
*/
const getPermissionType = (type: FEAST_FCO_TYPES): number => {
switch (type) {
case FEAST_FCO_TYPES.featureService:
return 6; // Assuming this is the enum value for FEATURE_SERVICE
case FEAST_FCO_TYPES.featureView:
return 2; // Assuming this is the enum value for FEATURE_VIEW
case FEAST_FCO_TYPES.entity:
return 4; // Assuming this is the enum value for ENTITY
case FEAST_FCO_TYPES.dataSource:
return 7; // Assuming this is the enum value for DATA_SOURCE
default:
return -1;
}
};

/**
* Format permissions for display
* @param permissions List of permissions
* @returns Formatted permissions string
*/
export const formatPermissions = (permissions: any[] | undefined): string => {
if (!permissions || permissions.length === 0) {
return "No permissions";
}

return permissions
.map((p) => {
const actions = p.spec?.actions
?.map((a: number) => getActionName(a))
.join(", ");
return `${p.spec?.name}: ${actions}`;
})
.join("\n");
};

/**
* Convert action number to readable name
*/
const getActionName = (action: number): string => {
const actionNames = [
"CREATE",
"DESCRIBE",
"UPDATE",
"DELETE",
"READ_ONLINE",
"READ_OFFLINE",
"WRITE_ONLINE",
"WRITE_OFFLINE",
];
return actionNames[action] || `Unknown (${action})`;
};

/**
* Filter function for permissions
* @param permissions List of all permissions
* @param action Action to filter by
* @returns Filtered permissions list
*/
export const filterPermissionsByAction = (
permissions: any[] | undefined,
action: string
): any[] => {
if (!permissions || permissions.length === 0) {
return [];
}

return permissions.filter((permission) => {
return permission.spec?.actions?.some(
(a: number) => getActionName(a) === action
);
});
};