StackOne HUB is a React-based integration component library that provides a web component wrapper for seamless integration into any web application. It enables developers to easily embed StackOne's integrations hub.
- StackOne HUB
# Clone and setup
git clone <repository-url>
cd hub
npm install
npm run build
# Start development
npm run dev-
Clone the repository:
git clone <repository-url> cd hub
-
Install dependencies:
npm install
-
Build the project:
npm run build
-
Create environment file:
cp .env.example .env
-
Configure your environment variables (see Environment Variables section)
-
Start the development server:
npm run dev
The Vite dev server starts at http://localhost:3001.
A second sandbox lives in dev/nextjs/ and runs the hub inside a Next.js 15 + React 19 App Router app. Use it to verify server-side rendering behaviour.
From the repo root:
# First time (builds the hub and installs sandbox deps)
npm run dev:nextjs:setup
# Subsequent runs
npm run dev:nextjsThe Next.js sandbox runs at http://localhost:3002. Set STACKONE_API_KEY in dev/nextjs/.env (see dev/nextjs/.env.example) to have the page fetch a connect-session token server-side, or paste one into the input on the page.
After editing hub source, rerun npm run build from the repo root β the sandbox is linked via file:../.. so it picks up the new dist/ automatically.
To build the project for production:
npm run buildThe build generates multiple bundles in the dist/ directory:
| File | Description | Use Case |
|---|---|---|
dist/index.esm.js |
ES module bundle (with 'use client' banner) |
Modern React apps, Next.js, Vite |
dist/index.js |
CommonJS module (with 'use client' banner) |
Node.js / legacy environments |
dist/index.d.ts |
TypeScript declarations | Type-checking |
dist/webcomponent.js |
Web component bundle (IIFE, React inlined) | Vanilla HTML/JS integration |
For vanilla HTML/JavaScript applications:
<!DOCTYPE html>
<html>
<head>
<title>StackOne HUB Integration</title>
</head>
<body>
<script src="<TBD>/webcomponent.js"></script>
<stackone-hub token="..."></stackone-hub>
</body>
</html>For React applications (CSR β Vite, CRA, etc.):
import { StackOneHub } from "@stackone/hub";
function App() {
return (
<div className="app">
<h1>My Application</h1>
<StackOneHub token={token} />
</div>
);
}
export default App;StackOneHub is a client-side component β it ships with a 'use client' directive and is safe to import directly in any framework that supports server-side rendering.
StackOneHub is annotated with 'use client' so you can import it directly from any Server Component. The token can be created server-side (recommended β keeps your API key off the client and avoids the CORS-protected /connect_sessions endpoint), and passed as a prop to a small Client Component that renders the hub.
Important: Add suppressHydrationWarning to the <html> tag in your root layout. The hub applies its theme CSS custom properties to document.documentElement after hydration, which would otherwise trigger a hydration warning on the <html> element (the warning only suppresses the <html> tag itself, not its children):
// app/layout.tsx
export default function RootLayout({ children }) {
return (
<html lang="en" suppressHydrationWarning>
<body>{children}</body>
</html>
);
}app/page.tsx (Server Component):
import HubWrapper from "./HubWrapper";
export default async function Page() {
const res = await fetch("https://api.stackone.com/connect_sessions", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Basic ${Buffer.from(process.env.STACKONE_API_KEY!).toString("base64")}`,
},
body: JSON.stringify({
origin_owner_id: "your_customer_id",
origin_owner_name: "Your Customer",
origin_username: "your_username",
}),
cache: "no-store",
});
const { token } = await res.json();
return <HubWrapper token={token} />;
}app/HubWrapper.tsx (Client Component):
"use client";
import { StackOneHub } from "@stackone/hub";
export default function HubWrapper({ token }: { token: string }) {
return (
<StackOneHub
token={token}
mode="integration-picker"
onSuccess={(account) => console.log("connected", account)}
/>
);
}If you prefer to opt the hub out of SSR entirely (Pages Router, or to skip the server pre-render):
import dynamic from "next/dynamic";
const StackOneHub = dynamic(
() => import("@stackone/hub").then((m) => m.StackOneHub),
{ ssr: false },
);A working example lives in dev/nextjs/.
@stackone/hub declares react and react-dom as peer dependencies and the bundle imports them at runtime β your app's copy must be the only copy that ends up loaded. In a standard npm install your bundler will hoist React and you won't see this. But the following setups can leave you with two copies of React and trip the "Invalid hook call" error:
- Monorepos (npm workspaces, Yarn workspaces, Turborepo) where multiple packages each have their own
node_modules/react. - pnpm with strict isolation β a transitive copy can shadow the root copy.
file:/link:dependencies pointing at a directory that has its ownnode_modules/react(this is what bit our local Vite sandbox).
Fixes by bundler:
Vite β add resolve.dedupe to your config:
// vite.config.ts
export default defineConfig({
resolve: {
dedupe: ['react', 'react-dom', 'react-hook-form'],
},
});Webpack / Next.js β usually handled automatically. If not, alias react and react-dom to a single absolute path:
// next.config.mjs
import path from "node:path";
export default {
webpack: (config) => {
config.resolve.alias["react"] = path.resolve("./node_modules/react");
config.resolve.alias["react-dom"] = path.resolve("./node_modules/react-dom");
return config;
},
};pnpm β set public-hoist-pattern[]=react* in .npmrc, or shamefully-hoist=true.
To diagnose, run npm ls react (or pnpm why react) at your app's root β if you see more than one entry resolved to a different path, that's the cause.
<script src="dist/webcomponent.js"></script>
<stackone-hub token="..."></stackone-hub>import { StackOneHub } from "../dist/index.esm.js";
function App() {
return <StackOneHub token={token} />;
}Create a .env file in the dev directory with the following variables:
| Variable | Description | Required |
|---|---|---|
STACKONE_API_KEY |
Your StackOne API key | β |
ORIGIN_OWNER_ID |
The origin owner identifier | β |
ORIGIN_OWNER_NAME |
Display name for the owner | β |
ORIGIN_USERNAME |
Username for authentication | β |
API_URL |
Backend API endpoint URL | β |
DASHBOARD_URL |
Dashboard application URL | β |
STACKONE_API_KEY=your_api_key_here
ORIGIN_OWNER_ID=your_owner_id
ORIGIN_OWNER_NAME=Your Company Name
ORIGIN_USERNAME=your_username
API_URL=https://api.stackone.com
DASHBOARD_URL=https://dashboard.stackone.comWe welcome contributions and feedback! Please keep in mind:
- π No formal process yet: Contribution guidelines are still being established
- π¬ Communication is key: Please open an issue before submitting large changes
- Fork the repository
- Create a feature branch
- Make your changes
- Test thoroughly
- Submit a pull request
This project is licensed under the MIT License. See the LICENSE file for details.