The Publisher SDK (@malloy-publisher/sdk) is a comprehensive React component library for building data applications that interact with Publisher's REST API. It provides everything you need to browse semantic models, execute queries, visualize results, and build interactive data experiences.
- Installation
- Quick Start
- Core Concepts
- ServerProvider
- Page Components
- Query & Results Components
- Dimensional Filters
- Hooks
- Utilities
- Workbook Storage
- Styling
- Building a Custom Data App
- API Reference
# Using bun
bun add @malloy-publisher/sdk
# Using npm
npm install @malloy-publisher/sdk
# Using yarn
yarn add @malloy-publisher/sdkimport { ServerProvider, Home } from "@malloy-publisher/sdk";
import "@malloy-publisher/sdk/styles.css";
function App() {
return (
<ServerProvider baseURL="http://localhost:4000/api/v0">
<Home onClickEnvironment={(path) => console.log("Navigate to:", path)} />
</ServerProvider>
);
}import {
BrowserRouter,
Routes,
Route,
useNavigate,
useParams,
} from "react-router-dom";
import {
ServerProvider,
Home,
Environment,
Package,
Model,
Notebook,
encodeResourceUri,
useRouterClickHandler,
} from "@malloy-publisher/sdk";
import "@malloy-publisher/sdk/styles.css";
function App() {
return (
<ServerProvider>
<BrowserRouter>
<Routes>
<Route path="/" element={<HomePage />} />
<Route path="/:environmentName" element={<EnvironmentPage />} />
<Route
path="/:environmentName/:packageName"
element={<PackagePage />}
/>
<Route
path="/:environmentName/:packageName/*"
element={<ModelPage />}
/>
</Routes>
</BrowserRouter>
</ServerProvider>
);
}
function HomePage() {
const navigate = useRouterClickHandler();
return <Home onClickEnvironment={navigate} />;
}
function EnvironmentPage() {
const navigate = useRouterClickHandler();
const { environmentName } = useParams();
const resourceUri = encodeResourceUri({ environmentName });
return <Environment onSelectPackage={navigate} resourceUri={resourceUri} />;
}
function PackagePage() {
const navigate = useRouterClickHandler();
const { environmentName, packageName } = useParams();
const resourceUri = encodeResourceUri({ environmentName, packageName });
return <Package onClickPackageFile={navigate} resourceUri={resourceUri} />;
}
function ModelPage() {
const params = useParams();
const modelPath = params["*"];
const resourceUri = encodeResourceUri({
environmentName: params.environmentName,
packageName: params.packageName,
modelPath,
});
if (modelPath?.endsWith(".malloy")) {
return <Model resourceUri={resourceUri} />;
}
if (modelPath?.endsWith(".malloynb")) {
return <Notebook resourceUri={resourceUri} />;
}
return <div>Unknown file type</div>;
}The SDK uses a standardized URI format to identify resources:
publisher://environments/{environmentName}/packages/{packageName}/models/{modelPath}?versionId={version}
Examples:
- Environment:
publisher://environments/my-environment - Package:
publisher://environments/my-environment/packages/analytics - Model:
publisher://environments/my-environment/packages/analytics/models/orders.malloy
Use the encodeResourceUri() and parseResourceUri() utilities to work with these URIs.
The SDK components follow a natural hierarchy:
ServerProvider (required wrapper)
├── Home (list all environments)
│ └── Environment (show packages in an environment)
│ └── Package (show models, notebooks, connections)
│ ├── Model (visual query builder + named queries)
│ └── Notebook (read-only notebook viewer)
└── Workbook (interactive analysis workbook)
Components accept callback functions for navigation rather than handling routing directly. This allows you to integrate with any routing solution:
// With React Router
const navigate = useRouterClickHandler();
<Home onClickEnvironment={navigate} />
// Custom navigation
<Home onClickEnvironment={(path) => window.location.href = path} />
// SPA with history
<Home onClickEnvironment={(path) => history.push(path)} />The ServerProvider is the required context provider that wraps your application. It initializes API clients and passes auth headers (if required by the backend server).
| Prop | Type | Default | Description |
|---|---|---|---|
baseURL |
string |
Auto-detected | Base URL of the Publisher API (e.g., http://localhost:4000/api/v0) |
getAccessToken |
() => Promise<string> |
undefined |
Async function returning auth token |
mutable |
boolean |
true |
Enable/disable environment/package management UI |
<ServerProvider>{/* Your app */}</ServerProvider>async function getAccessToken() {
const response = await fetch("/auth/token");
const { token } = await response.json();
return `Bearer ${token}`;
}
<ServerProvider getAccessToken={getAccessToken}>
{/* Your app */}
</ServerProvider>;// Disable add/edit/delete UI for production deployments
<ServerProvider mutable={false}>{/* Your app */}</ServerProvider><ServerProvider baseURL="https://publisher.example.com/api/v0">
{/* Your app */}
</ServerProvider>Displays a landing page with feature cards and a list of all available environments.
import { Home } from "@malloy-publisher/sdk";
interface HomeProps {
onClickEnvironment?: (path: string, event?: React.MouseEvent) => void;
}
// Usage
<Home
onClickEnvironment={(path, event) => {
// path is like "/my-environment/"
navigate(path);
}}
/>;Features:
- Hero section with Publisher branding
- Feature cards (Ad Hoc Analysis, Notebook Dashboards, AI Agents)
- Environment listing with descriptions
- Add/Edit/Delete environment dialogs (when
mutable=true)
Shows all packages within an environment.
import { Environment, encodeResourceUri } from "@malloy-publisher/sdk";
interface EnvironmentProps {
onSelectPackage: (path: string, event?: React.MouseEvent) => void;
resourceUri: string;
}
// Usage
const resourceUri = encodeResourceUri({ environmentName: "my-environment" });
<Environment
onSelectPackage={(path) => navigate(path)}
resourceUri={resourceUri}
/>;Features:
- Package listing with version info
- Add/Edit/Delete package dialogs (when
mutable=true) - Environment README display
Displays package details including models, notebooks, databases, and connections.
import { Package, encodeResourceUri } from "@malloy-publisher/sdk";
interface PackageProps {
onClickPackageFile?: (path: string, event?: React.MouseEvent) => void;
resourceUri: string;
}
// Usage
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
});
<Package
onClickPackageFile={(path) => navigate(path)}
resourceUri={resourceUri}
/>;Features:
- Models list (
.malloyfiles) - Notebooks list (
.malloynbfiles) - Embedded databases (
.parquetfiles) - Connection configuration
- Package README
The visual query builder and model explorer. This is the primary component for ad-hoc data analysis.
import { Model, encodeResourceUri } from "@malloy-publisher/sdk";
interface ModelProps {
resourceUri: string;
onChange?: (query: QueryExplorerResult) => void;
runOnDemand?: boolean; // Default: false
maxResultSize?: number; // Default: 0 (no limit)
}
interface QueryExplorerResult {
query: string | undefined;
malloyQuery: Malloy.Query | string | undefined;
malloyResult: Malloy.Result | undefined;
}
// Usage
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/orders.malloy",
});
<Model
resourceUri={resourceUri}
runOnDemand={true}
maxResultSize={512 * 1024}
onChange={(result) => {
console.log("Query:", result.query);
console.log("Result:", result.malloyResult);
}}
/>;Features:
- Source selector (dropdown for models with multiple sources)
- Visual query builder (Malloy Explorer integration)
- Named queries display
- Full-screen dialog mode
- Copy link to current view
A lower-level component for embedding the query builder without the full Model chrome.
import {
ModelExplorer,
useModelData,
encodeResourceUri,
} from "@malloy-publisher/sdk";
interface ModelExplorerProps {
data?: CompiledModel; // Pre-loaded model data
onChange?: (query: QueryExplorerResult) => void;
existingQuery?: QueryExplorerResult; // Initialize with existing query
initialSelectedSourceIndex?: number; // Default: 0
onSourceChange?: (index: number) => void;
resourceUri: string;
}
// Usage with automatic data loading
<ModelExplorer
resourceUri={resourceUri}
onChange={(result) => console.log(result)}
/>;
// Usage with pre-loaded data
const { data } = useModelData(resourceUri);
<ModelExplorer
data={data}
resourceUri={resourceUri}
onChange={(result) => console.log(result)}
/>;Read-only notebook viewer that executes cells and displays results.
import { Notebook, encodeResourceUri } from "@malloy-publisher/sdk";
interface NotebookProps {
resourceUri: string;
maxResultSize?: number; // Default: 0 (no limit)
}
// Usage
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "notebooks/sales-dashboard.malloynb",
});
<Notebook resourceUri={resourceUri} maxResultSize={1024 * 1024} />;Features:
- Sequential cell execution
- Markdown rendering
- Code cell execution with results
- Error handling per cell
Interactive workbook editor for creating and saving custom analyses.
import {
Workbook,
WorkbookStorageProvider,
BrowserWorkbookStorage,
encodeResourceUri,
} from "@malloy-publisher/sdk";
interface WorkbookProps {
workbookPath?: WorkbookLocator; // { path: string, workspace: string }
resourceUri: string;
}
// Usage
const workbookStorage = new BrowserWorkbookStorage();
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
});
<WorkbookStorageProvider workbookStorage={workbookStorage}>
<Workbook
workbookPath={{ path: "my-analysis", workspace: "Local" }}
resourceUri={resourceUri}
/>
</WorkbookStorageProvider>;Features:
- Add/remove Markdown and Malloy cells
- Model picker for source selection
- Auto-save to storage backend
- Export to Malloy format
- Delete workbook
Executes a query and displays the visualization.
import { QueryResult, encodeResourceUri } from "@malloy-publisher/sdk";
interface QueryResultProps {
query?: string; // Raw Malloy query
sourceName?: string; // Source name for named query
queryName?: string; // Named query to execute
resourceUri?: string; // Resource URI for model
}
// Execute a named query
<QueryResult
sourceName="orders"
queryName="by_region"
resourceUri={encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/orders.malloy",
})}
/>
// Execute a raw query
<QueryResult
query="run: orders -> { group_by: status; aggregate: order_count }"
resourceUri={encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/orders.malloy",
})}
/>Low-level component for rendering Malloy result JSON as a visualization.
import RenderedResult from "@malloy-publisher/sdk";
interface RenderedResultProps {
result: string; // JSON result string
height?: number; // Fixed height in pixels
onSizeChange?: (height: number) => void; // Callback when size changes
onDrill?: (element: unknown) => void; // Drill-down callback
}
// Usage (result is the JSON string from query execution)
<RenderedResult
result={queryResultJson}
onDrill={(element) => {
console.log("Drilled into:", element);
}}
/>;Helper for embedding query results as serialized JSON (useful for storage/transfer).
import {
EmbeddedQueryResult,
createEmbeddedQueryResult,
} from "@malloy-publisher/sdk";
// Create embedded query config
const embedded = createEmbeddedQueryResult({
queryName: "by_region",
sourceName: "orders",
resourceUri: encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/orders.malloy",
}),
});
// Later, render it
<EmbeddedQueryResult embeddedQueryResult={embedded} />;The SDK supports interactive dimensional filtering for notebooks and embedded data apps. Filters are configured through annotations in Malloy source files and notebooks.
| Type | UI Component | Use Case |
|---|---|---|
Star |
Multi-select dropdown | String fields with discrete values |
MinMax |
Range slider | Numeric fields |
DateMinMax |
Date range picker | Date/timestamp fields |
Retrieval |
Semantic search input | Free-text semantic search |
Boolean |
Toggle switch | Boolean fields |
Add filter annotations to dimensions in your Malloy source files using the #(filter) tag:
source: flights is duckdb.table('data/flights.parquet') extend {
dimension:
// Multi-select dropdown for string values
#(filter) {"type": "Star"}
origin_code is origin
// Range slider for numeric values
#(filter) {"type": "MinMax"}
distance_miles is distance
// Date range picker
#(filter) {"type": "DateMinMax"}
flight_departure is dep_time
join_one: carriers with carrier
}
source: carriers is duckdb.table('data/carriers.parquet') extend {
dimension:
#(filter) {"type": "Star"}
nickname is nickname_old
// Semantic search for text fields (requires embedding index)
#(index_values) n=-1
#(filter) {"type": "Retrieval"}
name is name_old
}
source: recalls is duckdb.table('data/recalls.csv') extend {
dimension:
// Boolean toggle filter
#(filter) {"type": "Boolean"}
is_major_recall is potentially_affected > 100000
}
By default, filters display the dimension field name in the UI. You can customize the display label using the # label="..." annotation:
source: recalls is duckdb.table('data/recalls.csv') extend {
dimension:
#(filter) {"type": "Star"}
# label="Vehicle Manufacturer"
Manufacturer is Manufacturer_old
#(filter) {"type": "Retrieval"}
# label="Recall Subject"
Subject is Subject_old
#(filter) {"type": "MinMax"}
# label="Number of Affected Vehicles"
potentially_affected is affected_count
}
The # label="..." annotation can be placed before or after the #(filter) annotation. When present, the label value will be displayed in the filter UI instead of the raw field name.
Enable filters in a notebook by adding a ##(filters) annotation in a Malloy code cell. This annotation specifies which dimensions should appear as filters using source.dimension format:
Simple array format:
##(filters) ["flights.origin_code", "carriers.name", "flights.flight_departure"]
import {flights, carriers} from 'flights.malloy'
The filter type for each dimension is determined by the #(filter) annotation on that dimension in the source file. If no source annotation exists, the dimension is ignored.
Note: Semantic search is not supported by the Publisher. When using the Notebook component, you supply an async function which implements the search for that column+query. If no search function is supplied, the filter is ignored.
For custom data apps, use the SDK's React hooks:
import {
useDimensionFiltersFromSpec,
DimensionFiltersConfig
} from '@malloy-publisher/sdk';
const config: DimensionFiltersConfig = {
environment: "malloy-samples",
package: "faa",
indexLimit: 1000,
dimensionSpecs: [
{ dimensionName: "origin_code", filterType: "Star", source: "flights", model: "flights.malloy", label: "Origin Airport" },
{ dimensionName: "distance", filterType: "MinMax", source: "flights", model: "flights.malloy", label: "Distance (miles)" },
{ dimensionName: "dep_time", filterType: "DateMinMax", source: "flights", model: "flights.malloy", label: "Departure Time" },
],
};
function FilteredDashboard() {
const {
filterStates, // Current filter values
updateFilter, // Update a single filter
clearAllFilters, // Reset all filters
activeFilters, // Array of active filter selections
data, // Dimension values for dropdowns/sliders
isLoading, // Loading state
executeQuery, // Run query with current filters
queryString, // Generated Malloy query
} = useDimensionFiltersFromSpec(config);
// Render filter UI and results...
}Filters support different match types depending on the filter type:
| Match Type | Description | Applicable To |
|---|---|---|
Equals |
Exact match (multi-select supported) | Star, Retrieval |
Contains |
Substring match | Star |
Greater Than / Less Than |
Comparison | MinMax |
Between |
Range (inclusive) | MinMax, DateMinMax |
After / Before |
Date comparison | DateMinMax |
Semantic Search |
Semantic similarity | Retrieval |
Access the server context (API clients, configuration).
import { useServer } from "@malloy-publisher/sdk";
function MyComponent() {
const {
server, // Base URL string
apiClients, // API client instances
mutable, // Whether mutations are allowed
getAccessToken, // Auth token function
} = useServer();
// Use API clients directly
const environments = await apiClients.environments.listEnvironments();
const model = await apiClients.models.getModel(
environmentName,
packageName,
modelPath,
versionId,
);
}interface ApiClients {
models: ModelsApi; // Get/execute models
environments: EnvironmentsApi; // CRUD environments
packages: PackagesApi; // CRUD packages
notebooks: NotebooksApi; // Get/execute notebooks
connections: ConnectionsApi; // CRUD connections
databases: DatabasesApi; // Access embedded databases
watchMode: WatchModeApi; // File watching for dev
}React Query wrapper with standardized error handling.
import { useQueryWithApiError } from "@malloy-publisher/sdk";
function MyComponent() {
const { data, isLoading, isError, error } = useQueryWithApiError({
queryKey: ["my-data", someParam],
queryFn: async () => {
const response = await apiClients.environments.listEnvironments();
return response.data;
},
});
if (isLoading) return <Loading />;
if (isError) return <ApiErrorDisplay error={error} context="Loading data" />;
return <div>{JSON.stringify(data)}</div>;
}Features:
- Automatic server-based cache key namespacing
- Standardized axios error transformation
- No automatic retries (explicit control)
Mutation wrapper with standardized error handling.
import { useMutationWithApiError } from "@malloy-publisher/sdk";
function MyComponent() {
const mutation = useMutationWithApiError({
mutationFn: async (newEnvironment) => {
const response = await apiClients.environments.createEnvironment(newEnvironment);
return response.data;
},
onSuccess: () => {
queryClient.invalidateQueries(["environments"]);
},
});
return (
<button onClick={() => mutation.mutate({ name: "new-environment" })}>
Create Environment
</button>
);
}Fetch compiled model data for a resource URI.
import { useModelData } from "@malloy-publisher/sdk";
function MyComponent({ resourceUri }) {
const {
data, // CompiledModel
isLoading,
isError,
error,
} = useModelData(resourceUri);
if (isLoading) return <Loading text="Loading model..." />;
if (isError) return <ApiErrorDisplay error={error} />;
// Access model data
console.log("Sources:", data.sourceInfos);
console.log("Queries:", data.queries);
}Execute a query and get raw data (array of rows) instead of visualization.
import { useRawQueryData } from "@malloy-publisher/sdk";
function MyComponent({ resourceUri }) {
const {
data, // Array of row objects
isLoading,
isSuccess,
isError,
error,
} = useRawQueryData({
resourceUri,
modelPath: "models/orders.malloy",
queryName: "by_region",
sourceName: "orders",
enabled: true,
});
if (isSuccess) {
// data is an array of row objects
data.forEach((row) => {
console.log(row.region, row.total_sales);
});
}
}Smart navigation hook that supports modifier keys (Cmd/Ctrl+click for new tab).
import { useRouterClickHandler } from "@malloy-publisher/sdk";
function MyComponent() {
const navigate = useRouterClickHandler();
return (
<button onClick={(e) => navigate("/environments/analytics", e)}>
Go to Analytics
</button>
);
}Behavior:
- Normal click: In-app navigation
- Cmd/Ctrl+click: Open in new tab
- Middle-click: Open in new tab
- Shift+click: Open in new window
Create a resource URI from components.
import { encodeResourceUri } from "@malloy-publisher/sdk";
// Environment only
const environmentUri = encodeResourceUri({
environmentName: "my-environment",
});
// Result: "publisher://environments/my-environment"
// Package
const packageUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
});
// Result: "publisher://environments/my-environment/packages/analytics"
// Model with version
const modelUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/orders.malloy",
versionId: "abc123",
});
// Result: "publisher://environments/my-environment/packages/analytics/models/models/orders.malloy?versionId=abc123"Parse a resource URI back to components.
import { parseResourceUri } from "@malloy-publisher/sdk";
const uri =
"publisher://environments/my-environment/packages/analytics/models/orders.malloy?versionId=abc123";
const parsed = parseResourceUri(uri);
// Result:
// {
// environmentName: "my-environment",
// packageName: "analytics",
// modelPath: "orders.malloy",
// versionId: "abc123"
// }type ParsedResource = {
environmentName: string;
packageName?: string;
connectionName?: string;
versionId?: string;
modelPath?: string;
};Workbooks are interactive analysis documents that can be saved and loaded. The SDK provides a storage abstraction that you can implement for different backends.
interface Workspace {
name: string;
writeable: boolean;
description: string;
}
interface WorkbookLocator {
path: string;
workspace: string;
}
interface WorkbookStorage {
listWorkspaces(writeableOnly: boolean): Promise<Workspace[]>;
listWorkbooks(workspace: Workspace): Promise<WorkbookLocator[]>;
getWorkbook(path: WorkbookLocator): Promise<string>;
deleteWorkbook(path: WorkbookLocator): Promise<void>;
saveWorkbook(path: WorkbookLocator, workbook: string): Promise<void>;
moveWorkbook(from: WorkbookLocator, to: WorkbookLocator): Promise<void>;
}Built-in implementation using browser localStorage.
import {
BrowserWorkbookStorage,
WorkbookStorageProvider,
} from "@malloy-publisher/sdk";
const storage = new BrowserWorkbookStorage();
<WorkbookStorageProvider workbookStorage={storage}>
<App />
</WorkbookStorageProvider>;class S3WorkbookStorage implements WorkbookStorage {
private s3Client: S3Client;
private bucket: string;
constructor(s3Client: S3Client, bucket: string) {
this.s3Client = s3Client;
this.bucket = bucket;
}
async listWorkspaces(writeableOnly: boolean): Promise<Workspace[]> {
return [
{
name: this.bucket,
writeable: true,
description: "S3 bucket storage",
},
];
}
async listWorkbooks(workspace: Workspace): Promise<WorkbookLocator[]> {
const objects = await this.s3Client.listObjects(
this.bucket,
"workbooks/",
);
return objects.map((obj) => ({
path: obj.key,
workspace: workspace.name,
}));
}
async getWorkbook(path: WorkbookLocator): Promise<string> {
const data = await this.s3Client.getObject(this.bucket, path.path);
return data.toString();
}
async saveWorkbook(path: WorkbookLocator, workbook: string): Promise<void> {
await this.s3Client.putObject(this.bucket, path.path, workbook);
}
async deleteWorkbook(path: WorkbookLocator): Promise<void> {
await this.s3Client.deleteObject(this.bucket, path.path);
}
async moveWorkbook(
from: WorkbookLocator,
to: WorkbookLocator,
): Promise<void> {
const content = await this.getWorkbook(from);
await this.saveWorkbook(to, content);
await this.deleteWorkbook(from);
}
}
// Usage
const storage = new S3WorkbookStorage(s3Client, "my-workbooks-bucket");
<WorkbookStorageProvider workbookStorage={storage}>
<App />
</WorkbookStorageProvider>;Context provider for workbook storage.
import {
WorkbookStorageProvider,
useWorkbookStorage,
} from "@malloy-publisher/sdk";
// Provider setup
<WorkbookStorageProvider workbookStorage={myStorage}>
<App />
</WorkbookStorageProvider>;
// Access in components
function MyComponent() {
const { workbookStorage } = useWorkbookStorage();
const workbooks = await workbookStorage.listWorkbooks({
name: "Local",
writeable: true,
description: "",
});
}Import the SDK styles in your app entry point:
// Main SDK styles (required)
import "@malloy-publisher/sdk/styles.css";
// If using Model/ModelExplorer outside of Publisher
import "@malloy-publisher/sdk/malloy-explorer.css";
// If using Workbook markdown editor
import "@malloy-publisher/sdk/markdown-editor.css";The SDK uses Material-UI (MUI) v7. You can customize the theme:
import { createTheme, ThemeProvider, CssBaseline } from "@mui/material";
import { ServerProvider } from "@malloy-publisher/sdk";
const theme = createTheme({
palette: {
primary: {
main: "#14b3cb", // Malloy teal
},
secondary: {
main: "#fbbb04", // Malloy yellow
},
},
typography: {
fontFamily: '"Inter", "Roboto", sans-serif',
},
});
function App() {
return (
<ServerProvider>
<ThemeProvider theme={theme}>
<CssBaseline />
{/* Your app */}
</ThemeProvider>
</ServerProvider>
);
}The SDK exports several pre-styled components for consistent UI:
import {
StyledCard,
StyledCardContent,
StyledCardMedia,
PackageCard,
PackageCardContent,
PackageSectionTitle,
CleanNotebookContainer,
CleanNotebookSection,
} from "@malloy-publisher/sdk";import {
ServerProvider,
QueryResult,
useModelData,
encodeResourceUri,
ApiErrorDisplay,
Loading,
} from "@malloy-publisher/sdk";
import "@malloy-publisher/sdk/styles.css";
import { Grid, Typography, Paper } from "@mui/material";
function Dashboard() {
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/sales.malloy",
});
const { data, isLoading, isError, error } = useModelData(resourceUri);
if (isLoading) return <Loading text="Loading dashboard..." />;
if (isError) return <ApiErrorDisplay error={error} context="Dashboard" />;
return (
<Grid container spacing={3}>
<Grid item xs={12}>
<Typography variant="h4">Sales Dashboard</Typography>
</Grid>
<Grid item xs={12} md={6}>
<Paper sx={{ p: 2 }}>
<Typography variant="h6">Sales by Region</Typography>
<QueryResult
sourceName="orders"
queryName="by_region"
resourceUri={resourceUri}
/>
</Paper>
</Grid>
<Grid item xs={12} md={6}>
<Paper sx={{ p: 2 }}>
<Typography variant="h6">Monthly Trends</Typography>
<QueryResult
sourceName="orders"
queryName="monthly_trends"
resourceUri={resourceUri}
/>
</Paper>
</Grid>
<Grid item xs={12}>
<Paper sx={{ p: 2 }}>
<Typography variant="h6">Custom Query</Typography>
<QueryResult
query="run: orders -> {
group_by: product_category
aggregate:
total_revenue is sum(revenue)
avg_order_value is avg(order_value)
}"
resourceUri={resourceUri}
/>
</Paper>
</Grid>
</Grid>
);
}
function App() {
return (
<ServerProvider baseURL="http://localhost:4000/api/v0">
<Dashboard />
</ServerProvider>
);
}import {
ServerProvider,
useRawQueryData,
encodeResourceUri,
Loading,
ApiErrorDisplay,
} from "@malloy-publisher/sdk";
import { DataGrid } from "@mui/x-data-grid";
function DataTable() {
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/customers.malloy",
});
const { data, isLoading, isError, error } = useRawQueryData({
resourceUri,
modelPath: "models/customers.malloy",
sourceName: "customers",
queryName: "all_customers",
});
if (isLoading) return <Loading text="Loading data..." />;
if (isError) return <ApiErrorDisplay error={error} />;
const columns =
data.length > 0
? Object.keys(data[0]).map((key) => ({
field: key,
headerName: key,
width: 150,
}))
: [];
return (
<DataGrid
rows={data.map((row, i) => ({ id: i, ...row }))}
columns={columns}
pageSize={10}
autoHeight
/>
);
}import {
ServerProvider,
ModelExplorer,
encodeResourceUri,
} from "@malloy-publisher/sdk";
import "@malloy-publisher/sdk/styles.css";
import "@malloy-publisher/sdk/malloy-explorer.css";
import { useState } from "react";
function Explorer() {
const [selectedQuery, setSelectedQuery] = useState(null);
const resourceUri = encodeResourceUri({
environmentName: "my-environment",
packageName: "analytics",
modelPath: "models/orders.malloy",
});
return (
<div style={{ display: "flex", gap: "20px" }}>
<div style={{ flex: 1 }}>
<h2>Build Your Query</h2>
<ModelExplorer
resourceUri={resourceUri}
onChange={(result) => {
setSelectedQuery(result);
console.log("Generated Query:", result.query);
}}
/>
</div>
{selectedQuery && (
<div style={{ flex: 1 }}>
<h2>Query Preview</h2>
<pre>{selectedQuery.query}</pre>
</div>
)}
</div>
);
}For minimal bundle size when you only need API access:
// Use the client entry point
import { ServerProvider, useServer } from "@malloy-publisher/sdk/client";
function MyApp() {
return (
<ServerProvider baseURL="http://localhost:4000/api/v0">
<EnvironmentList />
</ServerProvider>
);
}
function EnvironmentList() {
const { apiClients } = useServer();
const [environments, setEnvironments] = useState([]);
useEffect(() => {
apiClients.environments
.listEnvironments()
.then((response) => setEnvironments(response.data));
}, []);
return (
<ul>
{environments.map((p) => (
<li key={p.name}>{p.name}</li>
))}
</ul>
);
}| Component | Description |
|---|---|
ServerProvider |
Required context provider for API access |
Home |
Environment listing landing page |
Environment |
Package listing for an environment |
Package |
Package detail (models, notebooks, connections) |
Model |
Full model explorer with visual query builder |
ModelExplorer |
Lower-level query builder component |
ModelExplorerDialog |
Model explorer in a modal dialog |
Notebook |
Read-only notebook viewer |
Workbook |
Interactive workbook editor |
WorkbookList |
List workbooks from storage |
WorkbookManager |
Workbook state management class |
WorkbookStorageProvider |
Context for workbook storage |
QueryResult |
Execute and display query |
RenderedResult |
Render Malloy result JSON |
EmbeddedQueryResult |
Render serialized query config |
Loading |
Loading spinner with text |
ApiErrorDisplay |
Error display component |
AnalyzePackageButton |
Create/manage workbooks |
SourcesExplorer |
Source schema browser |
ConnectionExplorer |
Connection management UI |
| Hook | Description |
|---|---|
useServer |
Access ServerProvider context |
useQueryWithApiError |
React Query with error handling |
useMutationWithApiError |
Mutations with error handling |
useModelData |
Fetch compiled model |
useRawQueryData |
Execute query, get raw data |
useRouterClickHandler |
Smart navigation with modifier keys |
useWorkbookStorage |
Access workbook storage context |
useDimensionFiltersFromSpec |
Programmatic dimensional filtering |
| Utility | Description |
|---|---|
encodeResourceUri |
Create resource URI from components |
parseResourceUri |
Parse resource URI to components |
createEmbeddedQueryResult |
Serialize query config |
BrowserWorkbookStorage |
localStorage-based workbook storage |
globalQueryClient |
Shared React Query client |
| Type | Description |
|---|---|
ParsedResource |
Parsed resource URI components |
ServerContextValue |
Server context interface |
ServerProviderProps |
ServerProvider props |
QueryExplorerResult |
Query builder result |
SourceAndPath |
Source info with model path |
WorkbookStorage |
Workbook storage interface |
WorkbookLocator |
Workbook path + workspace |
Workspace |
Workspace metadata |
ApiError |
Standardized API error |
ModelExplorerProps |
ModelExplorer props |
DimensionFiltersConfig |
Dimensional filter configuration |