> ## Documentation Index
> Fetch the complete documentation index at: https://docs.livepeer.org/llms.txt
> Use this file to discover all available pages before exploring further.

# About Frameworks

> Full-stack video infrastructure: self-hosted or hosted, built on MistServer and Livepeer.

export const frameworksSocials = [{
  icon: 'globe',
  href: 'https://frameworks.network/',
  label: 'Frameworks'
}, {
  icon: 'file-code',
  href: 'https://docs.frameworks.network/',
  label: 'Frameworks Docs'
}, {
  icon: 'grid',
  href: 'https://app.frameworks.network/',
  label: 'Frameworks App'
}, {
  icon: 'github',
  href: 'https://github.com/livepeer-frameworks/monorepo',
  label: 'Frameworks GitHub'
}, {
  icon: 'envelope',
  href: 'https://frameworks.network/contact',
  label: 'Contact'
}];

export const frameworksInfra = [{
  icon: 'server',
  label: 'self-hosted'
}, {
  icon: 'cloud',
  label: 'saas'
}, {
  icon: 'torii-gate',
  label: 'gateway'
}];

export const frameworksBadges = [{
  color: 'blue',
  label: 'Video'
}];

export const StyledStep = ({title, icon, titleSize = 'h3', iconColor = null, titleColor = null, children, className = '', style = {}, ...rest}) => {
  const styledTitle = titleColor ? <span style={{
    color: titleColor
  }}>{title}</span> : title;
  return <Step title={styledTitle} icon={icon} iconColor={iconColor || undefined} titleSize={titleSize} className={className} style={style} {...rest}>
      {children}
    </Step>;
};

export const StyledSteps = ({children, iconColor, titleColor, lineColor, iconSize = '24px', className = '', style = {}, ...rest}) => {
  const resolvedIconColor = iconColor || 'var(--accent-dark, #18794E)';
  const resolvedTitleColor = titleColor || 'var(--lp-color-accent)';
  const resolvedLineColor = lineColor || 'var(--lp-color-accent)';
  return <div className={['docs-styled-steps', className].filter(Boolean).join(' ')} style={style} {...rest}>
      <style>{`
        .docs-styled-steps .steps > div > div.absolute > div {
          background-color: ${resolvedIconColor};
        }
        .docs-styled-steps .steps > div > div.w-full > p {
          color: ${resolvedTitleColor};
        }
        .docs-styled-steps .steps > div > div.absolute.w-px {
          background-color: ${resolvedLineColor};
        }
        .docs-styled-steps .steps > div:last-child > div.absolute.w-px::after {
          content: '';
          position: absolute;
          bottom: 0;
          left: 50%;
          transform: translateX(-50%);
          width: 6px;
          height: 6px;
          background-color: ${resolvedLineColor};
          transform: translateX(-50%) rotate(45deg);
        }
      `}</style>
      <div>
        <Steps>{children}</Steps>
      </div>
    </div>;
};

export const IconBadgeWrapper = ({items = [], iconColor, size = 14, gap = "var(--lp-spacing-3)", margin = '0.25rem 0 0.5rem', style = {}, className = '', ...rest}) => {
  const wrapperStyle = {
    display: 'flex',
    flexWrap: 'wrap',
    alignItems: 'center',
    gap,
    margin
  };
  const tagStyle = {
    display: 'inline-flex',
    alignItems: 'center',
    gap: "var(--lp-spacing-1)",
    fontSize: `${size}px`,
    color: 'var(--lp-color-text-primary)'
  };
  return <div className={className} style={{
    ...wrapperStyle,
    ...style
  }} {...rest}>
      {items.map((item, i) => <span key={i} style={tagStyle}>
          <Icon icon={item.icon} size={size} color={iconColor || 'currentColor'} />
          {item.label}
        </span>)}
    </div>;
};

export const BadgeWrapper = ({badges, children, gap = '0.4rem', margin = '0.5rem 0 1.5rem 0', style = {}, className = '', ...rest}) => {
  const defaultStyle = {
    display: 'flex',
    flexWrap: 'wrap',
    alignItems: 'center',
    gap,
    margin
  };
  return <div className={className} style={{
    ...defaultStyle,
    ...style
  }} {...rest}>
      {badges ? badges.map((b, i) => <Badge key={i} color={b.color}>
              {b.label}
            </Badge>) : children}
    </div>;
};

export const Video = ({src, title = "", author = "", caption, href = "", controls = true, autoPlay = false, loop = false, muted = false, children, className = "", style = {}, ...rest}) => {
  const buildCaption = () => {
    if (!author && caption) return caption;
    if (!author && !title) return null;
    return <span style={{
      display: "flex",
      flexDirection: "column",
      alignItems: "center",
      textAlign: "center",
      lineHeight: 1.2
    }}>
        <span>
          {author && (href ? <span style={{
      display: "flex",
      width: "100%",
      height: "fit-content",
      gap: "var(--lp-spacing-2)"
    }}>
                <a href={href} target="_blank" rel="noopener noreferrer">
                  <Icon icon="microphone" size={16} />
                  <span style={{
      borderBottom: "2px solid var(--lp-color-accent)",
      marginLeft: "0.2rem"
    }}>
                    <strong>{author}</strong>
                  </span>
                  <span style={{
      alignSelf: "flex-end",
      marginLeft: "var(--lp-spacing-px-6)"
    }}>
                    <Icon icon="arrow-up-right" size={12} color="var(--lp-color-accent)" />
                  </span>
                </a>
              </span> : <>
                <Icon icon="microphone" size={16} /> <strong>{author}</strong>
              </>)}
          {author && title ? ` • ${title}` : title}
        </span>
        {caption && <span style={{
      marginTop: "var(--lp-spacing-2)",
      fontStyle: "italic"
    }}>
            {caption}
          </span>}
      </span>;
  };
  const captionContent = buildCaption();
  const [linkHovered, setLinkHovered] = useState(false);
  return <div style={{
    position: "relative",
    display: "block"
  }}>
      {href && <a href={href} target="_blank" rel="noopener noreferrer" aria-label={title || author || "External link"} onMouseEnter={() => setLinkHovered(true)} onMouseLeave={() => setLinkHovered(false)} style={{
    position: "absolute",
    top: "var(--lp-spacing-2)",
    right: "var(--lp-spacing-3)",
    zIndex: "2",
    display: "inline-flex",
    alignItems: "center",
    justifyContent: "center",
    padding: "var(--lp-spacing-px-6)",
    background: linkHovered ? "var(--lp-color-border-default)" : "transparent",
    border: `1.5px solid ${linkHovered ? "var(--lp-color-accent)" : "var(--lp-color-text-primary)"}`,
    borderRadius: "6px",
    textDecoration: "none",
    lineHeight: 0,
    cursor: "pointer",
    transition: "all 0.15s ease"
  }}>
          <Icon icon="arrow-up-right" size={14} color={linkHovered ? "var(--lp-color-accent)" : "var(--lp-color-text-primary)"} />
        </a>}
      <Frame className={className} style={style} {...captionContent ? {
    caption: captionContent
  } : {}} {...rest}>
        <video controls={controls} autoPlay={autoPlay} loop={loop} muted={muted} playsInline className="w-full aspect-video rounded-xl" src={src} title={title || author || "Video"} />
        {children}
      </Frame>
    </div>;
};

export const YouTubeVideo = ({embedUrl, url, title = "", author = "", hint = "", caption, className = "", style = {}, ...rest}) => {
  const toEmbedUrl = value => {
    if (!value || typeof value !== "string") return "";
    const source = value.trim();
    if (!source) return "";
    try {
      const parsed = new URL(source);
      const host = parsed.hostname.replace(/^www\./, "");
      if (host === "youtube.com" || host === "youtube-nocookie.com") {
        if (parsed.pathname.startsWith("/embed/")) return source;
        const videoId = parsed.pathname.startsWith("/shorts/") ? parsed.pathname.split("/").filter(Boolean)[1] : parsed.searchParams.get("v");
        if (!videoId) return "";
        const params = new URLSearchParams(parsed.search);
        params.delete("v");
        const query = params.toString();
        return `https://www.youtube.com/embed/${videoId}${query ? `?${query}` : ""}`;
      }
      if (host === "youtu.be") {
        const videoId = parsed.pathname.split("/").filter(Boolean)[0];
        if (!videoId) return "";
        const query = parsed.searchParams.toString();
        return `https://www.youtube.com/embed/${videoId}${query ? `?${query}` : ""}`;
      }
    } catch (_err) {
      return "";
    }
    return "";
  };
  const resolvedEmbedUrl = toEmbedUrl(embedUrl || url);
  if (!resolvedEmbedUrl) {
    return null;
  }
  const isValidYouTubeUrl = resolvedEmbedUrl.includes("youtube.com/embed/");
  if (!isValidYouTubeUrl) {
    console.warn("Invalid YouTube embed URL:", embedUrl || url);
    return null;
  }
  const buildCaption = () => {
    if (caption) return caption;
    if (!author && !title) return null;
    return <span style={{
      display: "flex",
      flexDirection: "column",
      alignItems: "center",
      textAlign: "center",
      lineHeight: 1.2
    }}>
        <span>
          {author && <>
              <Icon icon="microphone" size={16} /> <strong>{author}</strong>
            </>}
          {author && title ? `${" "} • ${title}` : title}
        </span>
      </span>;
  };
  const captionContent = buildCaption();
  return <Frame className={className} style={style} {...hint ? {
    hint
  } : {}} {...captionContent ? {
    caption: captionContent
  } : {}} {...rest}>
      <iframe className="w-full aspect-video rounded-xl" src={resolvedEmbedUrl} title={title || author || "YouTube Video"} allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" allowFullScreen />
    </Frame>;
};

export const CenteredContainer = ({children, maxWidth = "800px", padding = "0", preset = "default", width = "", minWidth = "", marginRight = "", marginBottom = "", textAlign = "", style = {}, className = "", ...rest}) => {
  const presets = {
    default: {},
    fitContent: {
      width: "fit-content",
      minWidth: "fit-content"
    },
    readable70: {
      width: "70%",
      minWidth: "fit-content"
    },
    readable80: {
      width: "80%",
      minWidth: "fit-content"
    },
    readable90: {
      width: "90%"
    },
    wide900: {
      maxWidth: "900px"
    }
  };
  const presetStyle = presets[preset] || presets.default;
  return <div className={className} style={{
    maxWidth: presetStyle.maxWidth || maxWidth,
    margin: "0 auto",
    padding: padding,
    ...presetStyle.width ? {
      width: presetStyle.width
    } : {},
    ...presetStyle.minWidth ? {
      minWidth: presetStyle.minWidth
    } : {},
    ...width ? {
      width
    } : {},
    ...minWidth ? {
      minWidth
    } : {},
    ...marginRight ? {
      marginRight
    } : {},
    ...marginBottom ? {
      marginBottom
    } : {},
    ...textAlign ? {
      textAlign
    } : {},
    ...style
  }} {...rest}>
      {children}
    </div>;
};

export const BorderedBox = ({children, variant = "default", padding = "var(--lp-spacing-4)", borderRadius = "var(--lp-spacing-px-8)", margin = "", accentBar = "", style = {}, className = "", ...rest}) => {
  const variants = {
    default: {
      border: "1px solid var(--lp-color-border-default)",
      backgroundColor: "var(--lp-color-bg-card)"
    },
    accent: {
      border: "1px solid var(--lp-color-accent)",
      backgroundColor: "var(--lp-color-bg-card)"
    },
    muted: {
      border: "1px solid var(--lp-color-border-default)",
      backgroundColor: "transparent"
    }
  };
  const accentBarColors = {
    accent: "var(--lp-color-accent)",
    positive: "var(--green-9)"
  };
  return <div data-docs-bordered-box="" data-accent-bar={accentBarColors[accentBar] ? "" : undefined} className={className} style={{
    ...variants[variant],
    padding: padding,
    borderRadius: borderRadius,
    ...margin ? {
      margin
    } : {},
    ...accentBarColors[accentBar] ? {
      position: "relative",
      '--accent-bar-color': accentBarColors[accentBar]
    } : {},
    ...style
  }} {...rest}>
      {children}
    </div>;
};

export const SocialLinks = ({links, size = 20, gap = "var(--lp-spacing-3)", justify = "center", iconColor, color, className = "", style = {}, ...rest}) => {
  const resolvedIconColor = iconColor || color;
  const linkStyle = {
    border: "none",
    borderBottom: "none",
    textDecoration: "none",
    display: "inline-flex"
  };
  const colors = {
    discord: resolvedIconColor || "var(--lp-color-brand-discord)",
    twitter: resolvedIconColor || "var(--lp-color-text-primary)",
    github: resolvedIconColor || "var(--lp-color-brand-github)",
    forum: resolvedIconColor || "var(--lp-color-brand-forum)",
    website: resolvedIconColor || "var(--lp-color-accent)",
    blog: resolvedIconColor || "var(--lp-color-accent)",
    globe: resolvedIconColor || "var(--lp-color-brand-globe)",
    twitch: resolvedIconColor || "var(--lp-color-brand-twitch)",
    youtube: resolvedIconColor || "var(--lp-color-brand-youtube)",
    instagram: resolvedIconColor || "var(--lp-color-brand-instagram)",
    linkedin: resolvedIconColor || "var(--lp-color-brand-linkedin)"
  };
  const iconColorMap = {
    discord: "discord",
    "x-twitter": "twitter",
    github: "github",
    "comment-pen": "forum",
    "pen-line": "blog",
    "pencil-line": "blog",
    globe: "globe",
    "book-open": "website",
    twitch: "twitch",
    youtube: "youtube",
    instagram: "instagram",
    linkedin: "linkedin"
  };
  const defaultLinks = [{
    icon: "discord",
    href: "https://discord.com/invite/livepeer",
    label: "Livepeer Discord"
  }, {
    icon: "globe",
    href: "https://livepeer.org",
    label: "Livepeer Website"
  }, {
    icon: "github",
    href: "https://github.com/livepeer",
    label: "Livepeer GitHub"
  }, {
    icon: "comment-pen",
    href: "https://forum.livepeer.org",
    label: "Livepeer Forum"
  }, {
    icon: "pen-line",
    href: "https://livepeer.org/blog",
    label: "Livepeer Blog"
  }, {
    icon: "x-twitter",
    href: "https://x.com/livepeer",
    label: "Livepeer X"
  }];
  const items = links || defaultLinks;
  return <div className={className} style={style} {...rest}>
      <style>{`
        .social-links a {
          border: none;
          border-bottom: none;
        }
      `}</style>
      <span className="social-links" style={{
    display: "flex",
    justifyContent: justify,
    gap: gap,
    marginTop: "var(--lp-spacing-2)"
  }}>
        {items.map((item, i) => <a key={i} href={item.href} target="_blank" rel="noopener noreferrer" aria-label={item.label} style={linkStyle}>
            <Tooltip headline={item.label}>
              <Icon icon={item.icon} size={size} color={colors[iconColorMap[item.icon] || "website"] || "var(--lp-color-accent)"} aria-hidden="true" />
            </Tooltip>
          </a>)}
      </span>
    </div>;
};

export const CustomCardTitle = ({icon, title, variant = "card", iconSize, style = {}, className = "", ...rest}) => {
  const variants = {
    card: {
      display: 'flex',
      alignItems: 'center',
      gap: "var(--lp-spacing-2)",
      marginBottom: "var(--lp-spacing-3)",
      color: 'var(--lp-color-text-primary)',
      fontSize: '1rem',
      fontWeight: 600
    },
    accordion: {
      display: 'inline-flex',
      alignItems: 'center',
      gap: "var(--lp-spacing-2)"
    },
    tab: {
      display: 'inline-flex',
      alignItems: 'center',
      gap: '0.4rem',
      fontSize: '0.875rem'
    }
  };
  const sizes = {
    card: 20,
    accordion: 18,
    tab: 14
  };
  const size = iconSize || sizes[variant] || 20;
  const baseStyle = variants[variant] || variants.card;
  return variant === 'card' ? <div className={className} style={{
    ...baseStyle,
    ...style
  }} {...rest}>
      {typeof icon === 'string' ? <Icon icon={icon} size={size} color="var(--lp-color-accent)" /> : icon}
      {title}
    </div> : <span className={className} style={{
    ...baseStyle,
    ...style
  }} {...rest}>
      {typeof icon === 'string' ? <Icon icon={icon} size={size} color="var(--lp-color-accent)" /> : icon}
      {title}
    </span>;
};

export const CustomDivider = ({color = "var(--lp-color-border-default)", middleText = "", spacing = "default", style = {}, className = "", ...rest}) => {
  const spacingPresets = {
    default: {
      margin: "24px 0"
    },
    overlap: {
      margin: "-1rem 0 -1rem 0"
    },
    tight: {
      margin: "0 0 -1rem 0"
    },
    section: {
      margin: "0 0 -2rem 0"
    },
    sectionOverlap: {
      margin: "-1rem 0 -2rem 0"
    },
    deepOverlap: {
      margin: "-1rem 0 -1.5rem 0"
    }
  };
  const spacingStyle = spacingPresets[spacing] || spacingPresets.default;
  return <div role="separator" aria-orientation="horizontal" className={className} style={{
    display: "flex",
    alignItems: "center",
    ...spacingStyle,
    fontSize: style?.fontSize || "16px",
    height: "fit-content",
    ...style
  }} {...rest}>
      <span style={{
    marginRight: "var(--lp-spacing-px-8)",
    opacity: 0.2
  }}>
        <Icon icon="/snippets/assets/logos/Livepeer-Logo-Symbol-Theme.svg" />
      </span>
      <div style={{
    flex: 1,
    height: "1px",
    background: "var(--lp-color-border-default)",
    opacity: 0.4
  }}></div>
      {middleText && <>
          <Icon icon="circle" size={2} />
          <span style={{
    margin: "0 8px",
    fontWeight: "bold",
    color: color,
    opacity: 0.7
  }}>
            {middleText}
          </span>
          <Icon icon="circle" size={2} />
        </>}
      <div style={{
    flex: 1,
    height: "1px",
    background: "var(--lp-color-border-default)",
    opacity: 0.4
  }}></div>
      <span style={{
    marginLeft: "var(--lp-spacing-px-8)",
    opacity: 0.2
  }}>
        <span style={{
    display: "inline-block",
    transform: "scaleX(-1)"
  }}>
          <Icon icon="/snippets/assets/logos/Livepeer-Logo-Symbol-Theme.svg" />
        </span>
      </span>
    </div>;
};

<IconBadgeWrapper margin="-1rem 0 1rem 0" iconColor="var(--accent)" size={16} items={frameworksInfra} />

<BadgeWrapper margin="0 0 1rem 0" badges={frameworksBadges} />

<CenteredContainer maxWidth="fit-content">
  <BorderedBox margin="0" padding="0.5rem 1rem 1rem 1rem" variant="accent">
    <SocialLinks iconColor="var(--text)" links={frameworksSocials} />
  </BorderedBox>
</CenteredContainer>

<Info>
  This page is maintained by the Frameworks Team. Help & support available via the [Frameworks Contact](https://frameworks.network/contact).
</Info>

Frameworks is **full-stack video infrastructure** from a Livepeer Special Purpose Entity (SPE). It combines full self-hosting with optional hosted processing, powered by MistServer: run the stack yourself, use Frameworks' hosted services, or mix both.

Frameworks connects to Livepeer's decentralised GPU network for AI and transcoding workloads, giving operators a sovereign, unclouded alternative to managed video platforms.

<CustomDivider style={{margin: "-0.5rem 0 -2.5rem 0"}} />

## Key Features

* **Full self-hosting** - Run the complete video stack on your own infrastructure with no vendor lock-in.
* **Hosted processing** - Use Frameworks' hosted ingest and transcoding services where self-hosting is not practical.
* **StreamCrafter** - Browser-based live production tool: mix sources, add graphics, and go live without desktop software.
* **Skipper** - Routing and load-balancing layer for directing streams across ingest and processing nodes.
* **MistServer powered** - Built on MistServer for flexible, open-source media server capabilities.

<YouTubeVideo embedUrl="https://www.youtube.com/embed/DKBRp0U-RKw" title="Frameworks  -  sovereign video infrastructure" caption="Frameworks: full-stack, self-hostable video infrastructure on Livepeer" />

<CustomDivider style={{margin: "0 0 -2.5rem 0"}} />

## Try Frameworks

<CardGroup cols={2}>
  <Card title={<CustomCardTitle icon="bolt-lightning" title="Quickstart" />} href="/v2/solutions/frameworks/overview" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=b55f53ecf282b584dcc124de95b04c59" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=04b3d20bf5c79cdfcd2a60423bb07a2b 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=2497f4d19d8aafccbf7c61b3d9798247 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=dd5322a5a986787b314c00c77698c201 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=9440eec03765bb9b74ae1c1a6f374ce3 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=90b1009e09353d52bd02e3c1e21e90a4 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Quickstart.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=73f1cb4c2b901e54db14e1bf6769fbd7 2500w">
    Go live in minutes from the app or with the SDK.
  </Card>

  <Card title={<CustomCardTitle icon="grid" title="Frameworks app" />} href="https://app.frameworks.network/" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=a2a33d74afbe52977f54acd42fe568ad" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=4f4a4c4fad7cb05eaf25b9f92208b733 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=0b923fb2fca1d7eafba48d9bbaa696ad 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=260b207ce91b796756df5bf11367b1c5 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=3b761b6c667ec90467e30d1757f99777 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=22594f079d359eebdb0c412e9ba20ba6 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Try-App.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=241f46394587a376e0c6304539ed5e13 2500w">
    Dashboard, StreamCrafter, and Skipper.
  </Card>

  <Card title={<CustomCardTitle icon="book-open" title="Documentation" />} href="https://docs.frameworks.network/" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=64622ed8e681c86c21202fc34a80d4b0" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=0457146d4e250143604b56e702d09177 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=271e16d6d4e44aa0e8b55a9cf9e4fe6f 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=13050d18074a6bff0c8af9e9e22f694c 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=bb1d90ba0b5f826811437f0cc69aae16 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=be7e5fda0781a33653ec50980f9aedda 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Docs.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=720beebabdbba3e3c12218ea8e975e3b 2500w">
    Setup, API reference, playback, and StreamCrafter guides.
  </Card>

  <Card title={<CustomCardTitle icon="comment-question" title="SPE pilot proposal" />} href="https://forum.livepeer.org/t/pre-proposal-livepeer-frameworks-spe-pilot-phase/2773" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=6541e9f9a0b7044b61c53ff6ccbf6a52" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=9c627771849f28ccbfc3b4cea947ed45 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=fec2e8e38fe0de64208b23aa6c8ae9e0 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=2450611292c1a1166013447987798af3 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=ff806258eb4f70d70c8af2f1133a0364 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=620eab03f74ea0ffcad8d9dad7e51253 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/frameworks/Frameworks_Hero_Proposal.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=af88768a1d5e64966a62589abdb289b1 2500w">
    Community treasury proposal and Frameworks’ role in the Livepeer ecosystem.
  </Card>
</CardGroup>

<CustomDivider style={{margin: "-0.5rem 0 -2.5rem 0"}} />

## Get Started

Get started with Frameworks by using the hosted app or by following the official setup in the [Frameworks documentation](https://docs.frameworks.network/). Full API reference, playback, and self-hosting details are there.

<StyledSteps>
  <StyledStep title="Start from the app" icon="browser">
    1. Go to [app.frameworks.network](https://app.frameworks.network/) and sign up (free tier available).
    2. Create a stream in the dashboard (e.g. "Product Demo").
    3. Go live with **StreamCrafter** in the browser or connect an external encoder (e.g. OBS) using the shown RTMP server and stream key.
    4. Use the dashboard for recordings, clips, and analytics.

    Free tier includes self-hosting and access to the shared bandwidth pool with Livepeer-backed compute.
  </StyledStep>

  <StyledStep title="Use the SDK" icon="code">
    To add playback or broadcast to your own app:

    1. Read the [Frameworks docs](https://docs.frameworks.network/) for the Player and StreamCrafter SDKs.
    2. Install the packages (e.g. `@livepeer-frameworks/player-react`) and configure your stream/playback URLs.
    3. Use the GraphQL API or MCP for streams, diagnostics, and billing when you need automation.

    <Info>
      Endpoints, SDK usage, and self-hosted deployment are documented in the [Frameworks documentation](https://docs.frameworks.network/). Use it as the source for API reference and code samples.
    </Info>
  </StyledStep>

  <StyledStep title="Self-host the stack" icon="server">
    To run the full stack yourself (sovereign deployment):

    1. Follow the [Frameworks docs](https://docs.frameworks.network/) for self-hosting and deployment.
    2. Optionally use hybrid mode: self-host the edge and use Frameworks’ hosted services for burst or GPU workloads.
    3. For agent integration, use [SKILL.md](https://frameworks.network/SKILL.md) and MCP to discover, authenticate, and operate the platform.
  </StyledStep>
</StyledSteps>

<CustomDivider style={{margin: "-0.5rem 0 -2.5rem 0"}} />

## Frameworks Resources

<CenteredContainer maxWidth="fit-content">
  <BorderedBox margin="0 0 1rem 0" padding="0.5rem 1rem 1rem 1rem" variant="accent">
    <SocialLinks iconColor="var(--text)" links={frameworksSocials} />
  </BorderedBox>
</CenteredContainer>

<CardGroup cols={2}>
  <Card title={<CustomCardTitle title="Frameworks docs" icon="newspaper"/>} href="https://docs.frameworks.network/" arrow>
    Full documentation: setup, API reference, and StreamCrafter guides.
  </Card>

  <Card title={<CustomCardTitle title="GitHub" icon="github"/>} href="https://github.com/livepeer-frameworks/monorepo" arrow>
    Open-source Frameworks monorepo: server, SDK, and tooling.
  </Card>

  <Card title={<CustomCardTitle title="Frameworks app" icon="grid"/>} href="https://app.frameworks.network/" arrow>
    Dashboard, StreamCrafter live production, and Skipper routing.
  </Card>

  <Card title={<CustomCardTitle title="Contact" icon="envelope"/>} href="https://frameworks.network/contact" arrow>
    Talk to the Frameworks team about enterprise and custom deployments.
  </Card>
</CardGroup>
