> ## 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 Stream.Place

> Open-source, decentralised video layer for social networks and Web3 apps, built on Livepeer.

export const streamplaceSocials = [{
  icon: 'globe',
  href: 'https://stream.place/',
  label: 'Streamplace'
}, {
  icon: 'file-code',
  href: 'https://stream.place/docs/',
  label: 'Streamplace Docs'
}, {
  icon: 'gitlab',
  href: 'https://git.stream.place/streamplace/streamplace',
  label: 'Streamplace GitLab'
}, {
  icon: 'github',
  href: 'https://github.com/streamplace/streamplace',
  label: 'Streamplace GitHub'
}, {
  icon: 'tower-broadcast',
  href: 'https://stream.place/live',
  label: 'Live Dashboard'
}, {
  icon: 'discord',
  href: 'https://discord.com/invite/EdtZv4UTMU',
  label: 'Streamplace Discord'
}];

export const streamplaceInfra = [{
  icon: 'toolbox',
  label: 'sdk'
}];

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

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 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 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={streamplaceInfra} />

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

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

<Info>
  This page is maintained by the Streamplace Team. Help & support available via the [Streamplace Discord]('https://discord.com/invite/EdtZv4UTMU').
</Info>

<Frame>
  <iframe src="https://stream.place" width="100%" height="700px" frameBorder="0" allowFullScreen title="Streamplace  -  live streams" />
</Frame>

Streamplace is a **public-goods, open-source decentralised video layer** for social networks, livestreaming platforms, and Web3-native apps. It is a Livepeer Special Purpose Entity (SPE).

Streamplace routes video ingest and processing through Livepeer's decentralised GPU network, providing censorship-resistant, verifiable infrastructure for any platform that embeds it.

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

## Key Features

* **Livestream and VOD pipeline** - Ingest, transcode, and serve live and on-demand video through a decentralised stack.
* **C2PA and Ethereum provenance** - Cryptographically signed metadata for every video segment: who created it, when, and on what terms.
* **Rights and distribution metadata** - A schema for content warnings, rights, and distribution policy, attached to every piece of content.
* **Node and SDK** - Run a Streamplace node or use the SDK to add decentralised video to any app.
* **AT Protocol and Fediverse ready** - Built for integration with Bluesky, Mastodon, and other open social protocols.

<YouTubeVideo embedUrl="https://www.youtube.com/embed/-xjd_iaULT0?start=63" title="Streamplace: Eli Mallon, founder" />

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

## Try Streamplace

<CardGroup cols={2}>
  <Card title={<CustomCardTitle icon="download" title="Download the node" />} href="https://stream.place/download" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=870f46af684631aff312624f650eb7bc" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=3449da052872a37ba80e4ba202f2bc7a 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=8b19156ea7b103055ccf0824066c8fac 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=1fb9d6f8a60d630f62297a7ec60751e2 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=ea7e0b539c308a05409cbda47f5cd004 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=1f4cc52bcf457797081a201c43cb0297 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Run-Node.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=7164bb43912c01b4a2fde9e9ec4481ab 2500w">
    Run a Streamplace node for self-hosted video infrastructure.
  </Card>

  <Card title={<CustomCardTitle icon="tower-broadcast" title="Live dashboard" />} href="https://stream.place/live" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=1d344ff34648791adc02274e4894503d" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=0d4411c2b39ddfbcb310e3ee7313df3e 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=5b06ddaa22dc4f56ca76ec80f9c30014 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=9a09bd7f90e90f4a39f07b710da34cc3 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=6ff5cc95e4220a5a90bc26aa57adb4b3 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=aeb0d6a88a526434e2fbbb9ea593a7c1 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Dashboard.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=4f97a1435ae60dbe4e634e1300ed56b4 2500w">
    Browse live streams running on the Streamplace network.
  </Card>

  <Card title={<CustomCardTitle icon="book-open" title="Developer docs" />} href="https://stream.place/docs/" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=8e2f4de061abaf29c16d6c3eefa21b9f" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=efb3ddaabff01e47179d563d3c59d088 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=dfda56c2734d28aca6532975d09bacea 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=738354432e5ed76ddf148b683e1db6c5 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=6fff5eef15485fb73fba8a1c5899fcf0 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=5eb05b5de18d2da0869ff98ae78acb76 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Docs.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=b6f0ed95d9bda9f84f9e0dffe2f9442c 2500w">
    Integration guides, API reference, and node setup.
  </Card>

  <Card title={<CustomCardTitle icon="diagram-project" title="Integration guide" />} href="/solutions/streamplace/introduction/streamplace-integration" img="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=2334db0d5f9c9c1fc6fad8790f5b5622" data-og-width="800" width="800" data-og-height="450" height="450" data-path="snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?w=280&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=cdc5629cc3f28093f3d1c5dc9f7ac903 280w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?w=560&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=25c7f341468d8f6544d982b4eab8f951 560w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?w=840&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=f4c299d840cf27ad6fc56489773fcdba 840w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?w=1100&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=cd9b9aa32814bac015f6e71282b37484 1100w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?w=1650&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=9f0cbcb9c3d5e3c3e3001b297f4dac56 1650w, https://mintcdn.com/na-36/O-FYjjV6ev2CVZh-/snippets/assets/media/heros/solutions/streamplace/Streamplace_Hero_Integrate.png?w=2500&fit=max&auto=format&n=O-FYjjV6ev2CVZh-&q=85&s=9dca48f2873d6939309b16c201a3cb00 2500w">
    How to embed Streamplace into decentralised social apps and Web3 platforms.
  </Card>
</CardGroup>

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

## Get Started

Get started with [Stream.place](https://stream.place) by visiting the website and starting a live stream!

<StyledSteps>
  <StyledStep title="Install the Streamplace node">
    ```bash theme={"theme":{"light":"github-light","dark":"dark-plus"}}
    curl -s https://get.stream.place/install | bash
    ```

    Installs the Streamplace node binary and CLI. The node handles ingest, segmentation, provenance signing, and transcoding dispatch to the Livepeer Network.
  </StyledStep>

  <StyledStep title="Start a livestream">
    ```bash theme={"theme":{"light":"github-light","dark":"dark-plus"}}
    streamplace stream start --source webcam
    ```

    The node accepts WHIP/WHEP, RTMP, or browser WebRTC as input. Swap `--source webcam` for `--source rtmp` to ingest from an existing RTMP feed.
  </StyledStep>

  <StyledStep title="Configure metadata and rights">
    ```json theme={"theme":{"light":"github-light","dark":"dark-plus"}}
    {
      "contentRights": "CC-BY-4.0",
      "contentWarnings": ["flashing lights"],
      "distributionPolicy": {
        "retainSegments": true,
        "allowSyndication": true
      }
    }
    ```

    Metadata is embedded into every 1-second MP4 segment alongside a C2PA manifest and Ethereum signature. Rights and distribution policy travel with the content through transcoding and playback.
  </StyledStep>

  <StyledStep title="Integrate with the SDK">
    ```ts theme={"theme":{"light":"github-light","dark":"dark-plus"}}
    import { StreamplaceClient } from '@streamplace/sdk'

    const sp = new StreamplaceClient()
    const stream = await sp.createStream()
    ```

    Use `StreamplacePlayer` to surface provenance in your UI:

    ```jsx theme={"theme":{"light":"github-light","dark":"dark-plus"}}
    <StreamplacePlayer src={stream.manifestUrl} showProvenance={true} />
    ```

    The player exposes C2PA manifests, Ethereum signatures, and rights metadata to viewers automatically.
  </StyledStep>
</StyledSteps>

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

## Streamplace Resources

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

<CardGroup cols={2}>
  <Card title={<CustomCardTitle title="Streamplace docs" icon="newspaper"/>} href="https://stream.place/docs/" arrow>
    Full documentation: node setup, API, and integration guides.
  </Card>

  <Card title={<CustomCardTitle title="GitLab" icon="gitlab"/>} href="https://git.stream.place/streamplace/streamplace" arrow>
    Primary source: open-source Streamplace node and SDK.
  </Card>

  <Card title={<CustomCardTitle title="GitHub" icon="github"/>} href="https://github.com/streamplace/streamplace" arrow>
    GitHub mirror of the Streamplace repository.
  </Card>

  <Card title={<CustomCardTitle title="Architecture" icon="sitemap"/>} href="/solutions/streamplace/introduction/streamplace-architecture" arrow>
    How Streamplace's decentralised video stack is structured.
  </Card>

  <Card title={<CustomCardTitle title="Provenance and metadata" icon="shield-check"/>} href="/solutions/streamplace/introduction/streamplace-provenance" arrow>
    C2PA signing, Ethereum attestation, and content metadata schema.
  </Card>
</CardGroup>
