P2P Media Loader is an open-source JavaScript library that leverages modern web browser features, such as HTML5 video and WebRTC, to enable media delivery over peer-to-peer (P2P) networks. It integrates smoothly with many popular HTML5 video players and works entirely without browser plugins or add-ons. Experience it in action with our demo.
P2P Media Loader can be bundled into your project via an npm package or loaded directly from a CDN. Below are examples of both methods.
To include P2P Media Loader in your project using npm, follow these steps:
Install the package via npm:
For Hls.js integration:
npm install p2p-media-loader-hlsjs
For Shaka Player integration:
npm install p2p-media-loader-shaka
Import and use it in your project:
Hls.js integration:
import Hls from "hls.js";
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const HlsWithP2P = HlsJsP2PEngine.injectMixin(Hls);
Shaka Player integration:
import shaka from "shaka-player/dist/shaka-player.ui";
import { ShakaP2PEngine } from "p2p-media-loader-shaka";
ShakaP2PEngine.registerPlugins(shaka);
For additional examples using npm packages, please refer to our React demo.
P2P Media Loader supports a wide variety of players that use Hls.js as their underlying media engine. Let's use the Vidstack player for a comprehensive Hls.js example:
<!doctype html>
<html>
<head>
<!-- Include the Hls.js library from a CDN -->
<script src="https://cdn.jsdelivr.net/npm/hls.js@~1/dist/hls.min.js"></script>
<!-- Import map for the P2P Media Loader modules -->
<script type="importmap">
{
"imports": {
"p2p-media-loader-core": "https://cdn.jsdelivr.net/npm/p2p-media-loader-core@^2/dist/p2p-media-loader-core.es.min.js",
"p2p-media-loader-hlsjs": "https://cdn.jsdelivr.net/npm/p2p-media-loader-hlsjs@^2/dist/p2p-media-loader-hlsjs.es.min.js"
}
}
</script>
<!-- Include Vidstack player stylesheets -->
<link rel="stylesheet" href="https://cdn.vidstack.io/player/theme.css" />
<link rel="stylesheet" href="https://cdn.vidstack.io/player/video.css" />
<!-- Include the Vidstack player library from a CDN -->
<script src="https://cdn.vidstack.io/player" type="module"></script>
<!-- Module script to initialize the Vidstack player with P2P Media Loader -->
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const player = document.querySelector("media-player");
// Inject P2P capabilities into Hls.js
const HlsWithP2P = HlsJsP2PEngine.injectMixin(window.Hls);
player.addEventListener("provider-change", (event) => {
const provider = event.detail;
// Check if the provider is HLS
if (provider?.type === "hls") {
provider.library = HlsWithP2P;
provider.config = {
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// other P2P engine configuration parameters go here
},
onHlsJsCreated: (hls) => {
hls.p2pEngine.addEventListener("onPeerConnect", (params) => {
console.log("Peer connected:", params.peerId);
});
// Subscribe to P2P engine and Hls.js events here
},
},
};
}
});
</script>
</head>
<body>
<div style="width: 800px">
<!-- Vidstack media player with an HLS stream -->
<media-player src="streamUrl">
<media-provider></media-provider>
<media-video-layout></media-video-layout>
</media-player>
</div>
</body>
</html>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const videoElement = document.querySelector("#video");
const HlsWithP2P = HlsJsP2PEngine.injectMixin(window.Hls);
const hls = new HlsWithP2P({
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
onHlsJsCreated(hls) {
hls.p2pEngine.addEventListener("onPeerConnect", (params) => {
console.log("Peer connected:", params.peerId);
});
// Subscribe to P2P engine and Hls.js events here
},
},
});
hls.attachMedia(videoElement);
hls.loadSource(streamUrl);
</script>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
window.Hls = HlsJsP2PEngine.injectMixin(window.Hls);
const player = new Playerjs({
id: "player",
hlsconfig: {
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
onHlsJsCreated: (hls) => {
// Subscribe to P2P engine and Hls.js events here
hls.p2pEngine.addEventListener("onPeerConnect", (details) => {
console.log(`Connected to peer ${details.peerId})`);
});
},
},
},
});
</script>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const videoContainer = document.querySelector("#player");
const player = new DPlayer({
container: videoContainer,
video: {
url: "",
type: "customHls",
customType: {
customHls: (video) => {
const HlsWithP2P = HlsJsP2PEngine.injectMixin(window.Hls);
const hls = new HlsWithP2P({
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
},
});
hls.attachMedia(video);
hls.loadSource(streamUrl);
},
},
},
});
player.play();
</script>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const engine = new HlsJsP2PEngine({
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
});
const player = new Clappr.Player({
source: streamUrl,
plugins: [LevelSelector], // https://cdn.jsdelivr.net/gh/clappr/clappr-level-selector-plugin@~0/dist/level-selector.min.js
height: "100%",
width: "100%",
parentId: `#player`,
playback: {
hlsjsConfig: {
...engine.getConfigForHlsJs(),
},
},
});
engine.bindHls(() => clapprPlayer.core.getCurrentPlayback()?._hls);
</script>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const videoElement = document.querySelector("#video");
window.Hls = HlsJsP2PEngine.injectMixin(window.Hls);
const player = new MediaElementPlayer(videoElement.id, {
videoHeight: "100%",
hls: {
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
onHlsJsCreated: (hls) => {
// Subscribe to P2P engine and Hls.js events here
},
},
},
});
player.setSrc(streamUrl);
player.load();
</script>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const videoElement = document.querySelector("#video");
const HlsWithP2P = HlsJsP2PEngine.injectMixin(window.Hls);
const hls = new HlsWithP2P({
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
onHlsJsCreated(hls) {
// Subscribe to P2P engine and Hls.js events here
},
},
});
hls.on(Hls.Events.MANIFEST_PARSED, () => {
const levels = hls.levels;
const quality = {
default: levels[levels.length - 1].height,
options: levels.map((level) => level.height),
forced: true,
onChange: (newQuality) => {
levels.forEach((level, levelIndex) => {
if (level.height === newQuality) {
hls.currentLevel = levelIndex;
}
});
},
};
player = new Plyr(videoElement, {
quality,
autoplay: true,
muted: true,
});
});
hls.attachMedia(videoElement);
hls.loadSource(streamUrl);
</script>
<script type="module">
import { HlsJsP2PEngine } from "p2p-media-loader-hlsjs";
const videoElement = document.querySelector("#video");
const HlsWithP2P = HlsJsP2PEngine.injectMixin(window.Hls);
const player = new OpenPlayerJS(videoElement, {
hls: {
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
onHlsJsCreated: (hls) => {
// Subscribe to P2P engine and Hls.js events here
},
},
},
controls: {
layers: {
left: ["play", "time", "volume"],
right: ["settings", "fullscreen", "levels"],
middle: ["progress"],
},
},
});
player.src = [
{
src: streamUrl,
type: "application/x-mpegURL",
},
];
player.init();
</script>
Shaka Player is used below for an extended example:
<!doctype html>
<html>
<head>
<!-- Link to Shaka Player's CSS for controls -->
<link
rel="stylesheet"
type="text/css"
href="https://unpkg.com/shaka-player/dist/controls.css"
/>
<!-- Link to Shaka Player's compiled UI script -->
<script src="https://unpkg.com/shaka-player/dist/shaka-player.ui.js"></script>
<!-- Import map for the P2P Media Loader modules -->
<script type="importmap">
{
"imports": {
"p2p-media-loader-core": "https://cdn.jsdelivr.net/npm/p2p-media-loader-core@^2/dist/p2p-media-loader-core.es.min.js",
"p2p-media-loader-shaka": "https://cdn.jsdelivr.net/npm/p2p-media-loader-shaka@^2/dist/p2p-media-loader-shaka.es.min.js"
}
}
</script>
<!-- Module script to initialize Shaka Player with P2P Media Loader -->
<script type="module">
import { ShakaP2PEngine } from "p2p-media-loader-shaka";
// Register P2P Media Loader plugins with Shaka
ShakaP2PEngine.registerPlugins();
async function init() {
// Get the video element by its ID
const video = document.getElementById("video");
// Get Shaka UI controls and player
const ui = video["ui"];
const controls = ui.getControls();
const player = controls.getPlayer();
// Initialize P2P Media Loader with a custom config
const shakaP2PEngine = new ShakaP2PEngine({
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
});
// Subscribe to P2P engine events here
shakaP2PEngine.addEventListener("onPeerConnect", (params) => {
console.log("Peer connected:", params.peerId);
});
// Configure and initialize Shaka Player with P2P Media Loader
shakaP2PEngine.bindShakaPlayer(player);
// Load the stream URL into the player
player.load(streamUrl);
}
// Add event listener for the Shaka UI loaded event to trigger initialization
document.addEventListener("shaka-ui-loaded", init);
</script>
</head>
<body>
<div data-shaka-player-container style="max-width:40em">
<!-- Video element with Shaka Player UI -->
<video
autoplay
data-shaka-player
id="video"
style="width:100%;height:100%"
></video>
</div>
</body>
</html>
<script type="module">
import { ShakaP2PEngine } from "p2p-media-loader-shaka";
const container = document.getElementById("container");
ShakaP2PEngine.registerPlugins();
const shakaP2PEngine = new ShakaP2PEngine({
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P Media Loader Core options
},
});
const player = new Clappr.Player({
parentId: `#${container.id}`,
source: streamUrl,
plugins: [window.DashShakaPlayback, window.LevelSelector],
shakaOnBeforeLoad: (shakaPlayerInstance) => {
subscribeToUiEvents({
engine: shakaP2PEngine,
onPeerConnect,
onPeerDisconnect,
onChunkDownloaded,
onChunkUploaded,
});
shakaP2PEngine.bindShakaPlayer(shakaPlayerInstance);
},
});
</script>
<script type="module">
import { ShakaP2PEngine } from "p2p-media-loader-shaka";
const container = document.getElementById("container");
ShakaP2PEngine.registerPlugins();
const shakaP2PEngine = new ShakaP2PEngine({
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P Media Loader Core options
},
});
const player = new DPlayer({
container,
video: {
url: "",
type: "customHlsOrDash",
customType: {
customHlsOrDash: (video) => {
const shakaPlayer = new shaka.Player();
void shakaPlayer.attach(video);
shakaP2PEngine.bindShakaPlayer(shakaPlayer);
void shakaPlayer.load(streamUrl);
},
},
},
});
</script>
<script type="module">
import { ShakaP2PEngine } from "p2p-media-loader-shaka";
ShakaP2PEngine.registerPlugins();
const videoElement = document.getElementById("video");
const initPlayer = () => {
const shakaP2PEngine = new ShakaP2PEngine({
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P Media Loader Core options
},
});
const shakaPlayer = new shaka.Player();
shakaPlayer.attach(videoElement);
shakaP2PEngine.bindShakaPlayer(shakaPlayer);
shakaPlayer.load(streamUrl);
const plyrPlayer = new Plyr(videoElement);
};
initPlayer();
</script>
For legacy environments that lack ES module support (such as older Smart TVs and deprecated browsers), you can utilize our IIFE builds. In these builds, the library components are exposed via global variables instead of ES module imports.
The global namespaces are window.p2pml.hlsjs and window.p2pml.shaka.
<script src="https://cdn.jsdelivr.net/npm/hls.js@~1/dist/hls.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/p2p-media-loader-hlsjs@latest/dist/p2p-media-loader-hlsjs.iife.min.js"></script>
<script>
// Wait for the DOM and scripts to load
document.addEventListener("DOMContentLoaded", function () {
var videoElement = document.getElementById("video");
var streamUrl = "https://example.com/stream.m3u8";
if (Hls.isSupported()) {
// Access the engine from the global p2pml object
var HlsJsP2PEngine = window.p2pml.hlsjs.HlsJsP2PEngine;
var HlsWithP2P = HlsJsP2PEngine.injectMixin(window.Hls);
var hls = new HlsWithP2P({
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
},
});
hls.attachMedia(videoElement);
hls.loadSource(streamUrl);
}
});
</script>
<script src="https://unpkg.com/shaka-player/dist/shaka-player.compiled.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/p2p-media-loader-shaka@latest/dist/p2p-media-loader-shaka.iife.min.js"></script>
<script>
// Wait for the DOM and scripts to load
document.addEventListener("DOMContentLoaded", async function () {
if (shaka.Player.isBrowserSupported()) {
var videoElement = document.getElementById("video");
var streamUrl = "https://example.com/stream.mpd";
// Access the engine from the global p2pml object
var ShakaP2PEngine = window.p2pml.shaka.ShakaP2PEngine;
ShakaP2PEngine.registerPlugins();
var shakaP2PEngine = new ShakaP2PEngine({
p2p: {
core: {
swarmId: "Optional custom swarm ID for stream",
// Other P2P engine configuration parameters go here
},
},
});
var player = new shaka.Player();
await player.attach(videoElement);
shakaP2PEngine.bindShakaPlayer(player);
await player.load(streamUrl);
}
});
</script>