Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
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
1 change: 1 addition & 0 deletions ui/.gitignore
Original file line number Diff line number Diff line change
@@ -1,2 +1,3 @@
src/protos.d.ts
src/protos.js
feature_repo/data/*.parquet
2 changes: 1 addition & 1 deletion ui/.npmrc
Original file line number Diff line number Diff line change
@@ -1 +1 @@
//registry.npmjs.org/:_authToken=${NPM_TOKEN}
# //registry.npmjs.org/:_authToken=${NPM_TOKEN}
25 changes: 25 additions & 0 deletions ui/feature_repo/apply_permissions.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
from feast import FeatureStore
from features import (
zipcode_features_permission,
zipcode_source_permission,
model_v1_permission,
risky_features_permission,
document_embeddings_permission,
document_metadata_permission,
rag_model_permission,
)

store = FeatureStore(repo_path=".")

store.apply([
zipcode_features_permission,
zipcode_source_permission,
model_v1_permission,
risky_features_permission,
document_embeddings_permission,
document_metadata_permission,
rag_model_permission,
])

print("Permissions applied successfully!")
print("Current permissions:", store.list_permissions())
32 changes: 32 additions & 0 deletions ui/feature_repo/apply_rag_data.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import pandas as pd
import numpy as np
from datetime import datetime, timedelta

now = datetime.now()
embeddings = []
for i in range(10):
embeddings.append({
'document_id': f'doc_{i}',
'embedding': np.random.rand(768).astype(np.float32),
'event_timestamp': now - timedelta(days=i),
'created_timestamp': now - timedelta(days=i, hours=1)
})
df_embeddings = pd.DataFrame(embeddings)
df_embeddings.to_parquet('data/document_embeddings.parquet', index=False)

metadata = []
for i in range(10):
metadata.append({
'document_id': f'doc_{i}',
'title': f'Document {i}',
'content': f'This is the content of document {i}',
'source': 'web',
'author': f'author_{i}',
'publish_date': (now - timedelta(days=i*30)).strftime('%Y-%m-%d'),
'event_timestamp': now - timedelta(days=i),
'created_timestamp': now - timedelta(days=i, hours=1)
})
df_metadata = pd.DataFrame(metadata)
df_metadata.to_parquet('data/document_metadata.parquet', index=False)

print('Created RAG data files successfully!')
Binary file not shown.
Binary file added ui/feature_repo/data/document_metadata.parquet
Binary file not shown.
1 change: 1 addition & 0 deletions ui/feature_repo/feature_store.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,4 @@ online_store:
type: sqlite
offline_store:
type: file
entity_key_serialization_version: 2
156 changes: 155 additions & 1 deletion ui/feature_repo/features.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
from datetime import timedelta

import pandas as pd
import numpy as np

from feast import Entity, FeatureService, FeatureView, Field, FileSource
from feast.data_source import RequestSource
from feast.on_demand_feature_view import on_demand_feature_view
from feast.types import Bool, Int64, String
from feast.permissions.action import AuthzedAction, READ
from feast.permissions.permission import Permission
from feast.permissions.policy import RoleBasedPolicy
from feast.types import Bool, Int64, String, Float32, Array

zipcode = Entity(
name="zipcode",
Expand Down Expand Up @@ -199,3 +203,153 @@ def transaction_gt_last_credit_card_due(inputs: pd.DataFrame) -> pd.DataFrame:
tags={"owner": "amanda@feast.ai", "stage": "dev"},
description="Location model",
)

zipcode_features_permission = Permission(
name="zipcode-features-reader",
types=[FeatureView],
name_patterns=["zipcode_features"],
policy=RoleBasedPolicy(roles=["analyst", "data_scientist"]),
actions=[AuthzedAction.DESCRIBE, *READ],
)

zipcode_source_permission = Permission(
name="zipcode-source-writer",
types=[FileSource],
name_patterns=["zipcode"],
policy=RoleBasedPolicy(roles=["admin", "data_engineer"]),
actions=[AuthzedAction.CREATE, AuthzedAction.UPDATE, AuthzedAction.WRITE_OFFLINE],
)

model_v1_permission = Permission(
name="credit-score-v1-reader",
types=[FeatureService],
name_patterns=["credit_score_v1"],
policy=RoleBasedPolicy(roles=["model_user", "data_scientist"]),
actions=[AuthzedAction.DESCRIBE, AuthzedAction.READ_ONLINE],
)

risky_features_permission = Permission(
name="risky-features-reader",
types=[FeatureView, FeatureService],
required_tags={"stage": "prod"},
policy=RoleBasedPolicy(roles=["trusted_analyst"]),
actions=[AuthzedAction.READ_OFFLINE],
)

document = Entity(
name="document_id",
description="Document identifier for RAG system",
tags={
"owner": "nlp_team@feast.ai",
"team": "rag",
},
)

document_source = FileSource(
name="document_embeddings",
path="data/document_embeddings.parquet",
timestamp_field="event_timestamp",
created_timestamp_column="created_timestamp",
)

document_metadata_source = FileSource(
name="document_metadata",
path="data/document_metadata.parquet",
timestamp_field="event_timestamp",
created_timestamp_column="created_timestamp",
)

document_embeddings_view = FeatureView(
name="document_embeddings",
entities=[document],
ttl=timedelta(days=365),
schema=[
Field(name="embedding", dtype=Array(Float32, 768)),
Field(name="document_id", dtype=String),
],
source=document_source,
tags={
"date_added": "2025-05-04",
"model": "sentence-transformer",
"access_group": "nlp-team@feast.ai",
"stage": "prod",
},
online=True,
)

document_metadata_view = FeatureView(
name="document_metadata",
entities=[document],
ttl=timedelta(days=365),
schema=[
Field(name="title", dtype=String),
Field(name="content", dtype=String),
Field(name="source", dtype=String),
Field(name="author", dtype=String),
Field(name="publish_date", dtype=String),
Field(name="document_id", dtype=String),
],
source=document_metadata_source,
tags={
"date_added": "2025-05-04",
"access_group": "nlp-team@feast.ai",
"stage": "prod",
},
online=True,
)

# Define a request data source for query embeddings
query_request = RequestSource(
name="query",
schema=[
Field(name="query_embedding", dtype=Array(Float32, 768)),
],
)

# Define an on-demand feature view for similarity calculation
@on_demand_feature_view(
sources=[document_embeddings_view, query_request],
schema=[
Field(name="similarity_score", dtype=Float32),
],
)
def document_similarity(inputs: pd.DataFrame) -> pd.DataFrame:
"""Calculate cosine similarity between query and document embeddings."""
df = pd.DataFrame()
df["similarity_score"] = 0.95 # Placeholder value
return df

rag_model = FeatureService(
name="rag_retriever",
features=[
document_embeddings_view,
document_metadata_view,
document_similarity,
],
tags={"owner": "nlp_team@feast.ai", "stage": "prod"},
description="Retrieval Augmented Generation model",
)

document_embeddings_permission = Permission(
name="document-embeddings-reader",
types=[FeatureView],
name_patterns=["document_embeddings"],
policy=RoleBasedPolicy(roles=["ml_engineer", "data_scientist"]),
actions=[AuthzedAction.DESCRIBE, *READ],
)

document_metadata_permission = Permission(
name="document-metadata-reader",
types=[FeatureView],
name_patterns=["document_metadata"],
policy=RoleBasedPolicy(roles=["ml_engineer", "content_manager"]),
actions=[AuthzedAction.DESCRIBE, *READ],
)

rag_model_permission = Permission(
name="rag-model-user",
types=[FeatureService],
name_patterns=["rag_retriever"],
policy=RoleBasedPolicy(roles=["ml_engineer", "app_developer"]),
actions=[AuthzedAction.DESCRIBE, AuthzedAction.READ_ONLINE],
)
1 change: 1 addition & 0 deletions ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
"dependencies": {
"@elastic/datemath": "^5.0.3",
"@elastic/eui": "^95.12.0",
"@elastic/eui-theme-borealis": "1.0.0",
"@emotion/css": "^11.13.0",
"@emotion/react": "^11.13.3",
"@types/dagre": "^0.7.52",
Expand Down
5 changes: 4 additions & 1 deletion ui/src/FeastUISansProviders.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import React from "react";

import "@elastic/eui/dist/eui_theme_light.css";
import "./index.css";

import { Routes, Route } from "react-router-dom";
Expand All @@ -23,6 +22,8 @@ import FeatureServiceInstance from "./pages/feature-services/FeatureServiceInsta
import DataSourceInstance from "./pages/data-sources/DataSourceInstance";
import RootProjectSelectionPage from "./pages/RootProjectSelectionPage";
import DatasetInstance from "./pages/saved-data-sets/DatasetInstance";
import PermissionsIndex from "./pages/permissions/Index";
import LineageIndex from "./pages/lineage/Index";
import NoProjectGuard from "./components/NoProjectGuard";

import TabsRegistryContext, {
Expand Down Expand Up @@ -144,6 +145,8 @@ const FeastUISansProvidersInner = ({
path="data-set/:datasetName/*"
element={<DatasetInstance />}
/>
<Route path="permissions/" element={<PermissionsIndex />} />
<Route path="lineage/" element={<LineageIndex />} />
</Route>
</Route>
<Route path="*" element={<NoMatch />} />
Expand Down
107 changes: 107 additions & 0 deletions ui/src/components/PermissionsDisplay.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
import React from "react";
import {
EuiBadge,
EuiFlexGroup,
EuiFlexItem,
EuiPanel,
EuiText,
EuiTitle,
EuiHorizontalRule,
EuiToolTip,
} from "@elastic/eui";
import { formatPermissions } from "../utils/permissionUtils";

interface PermissionsDisplayProps {
permissions: any[] | undefined;
}

const PermissionsDisplay: React.FC<PermissionsDisplayProps> = ({
permissions,
}) => {
if (!permissions || permissions.length === 0) {
return (
<EuiText>
<p>No permissions defined for this resource.</p>
</EuiText>
);
}

const getActionColor = (action: string) => {
if (action.startsWith("READ")) return "success";
if (action.startsWith("WRITE")) return "warning";
if (action === "CREATE") return "primary";
if (action === "UPDATE") return "accent";
if (action === "DELETE") return "danger";
return "default";
};

return (
<React.Fragment>
{permissions.map((permission, index) => {
const actions = permission.spec?.actions?.map((a: number) => {
const actionNames = [
"CREATE",
"DESCRIBE",
"UPDATE",
"DELETE",
"READ_ONLINE",
"READ_OFFLINE",
"WRITE_ONLINE",
"WRITE_OFFLINE",
];
return actionNames[a] || `Unknown (${a})`;
});

return (
<div key={index} style={{ marginBottom: "8px" }}>
<EuiToolTip
position="top"
content={
<div>
<p>
<strong>Name:</strong> {permission.spec?.name}
</p>
<p>
<strong>Policy:</strong>{" "}
{permission.spec?.policy?.roles
? `Roles: ${permission.spec.policy.roles.join(", ")}`
: "No policy defined"}
</p>
{permission.spec?.name_patterns && (
<p>
<strong>Name Patterns:</strong>{" "}
{Array.isArray(permission.spec.name_patterns)
? permission.spec.name_patterns.join(", ")
: permission.spec.name_patterns}
</p>
)}
{permission.spec?.required_tags && (
<p>
<strong>Required Tags:</strong>{" "}
{Object.entries(permission.spec.required_tags)
.map(([key, value]) => `${key}: ${value}`)
.join(", ")}
</p>
)}
</div>
}
>
<EuiText>
<h4>{permission.spec?.name}</h4>
</EuiText>
</EuiToolTip>
<EuiFlexGroup wrap responsive={false} gutterSize="xs">
{actions.map((action: string, actionIndex: number) => (
<EuiFlexItem grow={false} key={actionIndex}>
<EuiBadge color={getActionColor(action)}>{action}</EuiBadge>
</EuiFlexItem>
))}
</EuiFlexGroup>
</div>
);
})}
</React.Fragment>
);
};

export default PermissionsDisplay;
Loading