Браузер Fork - Wiki:Fork Browser - Forkplayer.tv
ForkPlayer — это прикладное программное обеспечение для просмотра fxml(Fork eXtensible Markup Language)-страниц в глобальной сети. ForkPlayer используют для запроса, обработки, манипулирования и отображения содержания fxml-сайтов а также для непосредственного просмотра содержания файлов плейлистов (m3u,xml,xspf), изображений (gif, jpeg, png), аудио-видео форматов (mp3, mpeg, mkv), потокового видео (udp, hls).
В отличии от веб-сайтов, FXML-порталам не нужна для навигации мышь или сенсорный экран, страницы просты в отображении и серфинге по ним с помощью кнопочного пульта, не требовательны к ресурсам и удобные для просмотра на телевизорах.
Создаются и размещаются FXML-сайты аналогично веб-сайтам на своем сервере или хостинге со своим уникальным доменным именем, доступ происходит по протоколу HTTP (Files Headers For ForkPlayer).
С помощью FXML CMS создать свой fxml-портал может даже начинающий вебмастер.
Спецификация разметки FXML для написания кода вручную.
Содержание
История технических изменений ForkPlayer
Пользователям ForkPlayer
Инструкции по установке ForkPlayer
Хранилище закладок, загрузить плейлист в свой аккаунт в ForkPlayer
Подключение архива iptv
RemoteFork (Windows, Android)
TODO - ошибки в ForkPlayer
Авторам / издателям
FXML CMS - Готовый портал под ForkPlayer на вашем хостинге
DLE FXML - Модуль для популярной CMS Data Life Engine
Files Headers For ForkPlayer - Обязательно настройте нужные заголовки ваших страниц
PHP JSON генерация страниц под ForkPlayer
Формат XML и M3U страниц ForkPlayer
Свой DNS сервер Bind9 для запуска ForkPlayer
Владелец существующего вебсайта?
Увеличьте аудиторию и ее лояльность сделав версию под ForkPlayer!
Сигнатуры ForkPlayer - по наличию GET параметра box_mac определяем что ваш сайт был открыт в ForkPlayer
Пример готовых решений и структуры порталов под ForkPlayer
Модуль, если у вас сайт на DLE DLE FXML
Готовый простенький портал на PHP с авторизацией, поиском и выводом страниц в JSON https://github.com/alexkdpu/kino.pub_forkplayerPHP/blob/master/index.php
От авторов / издателей
Dstore - магазин приложений, кинозалы, новостные и развлекательные порталы
Тем кто хочет помочь
Создать свою страницу на ForkPlayer Wiki - дополнить проект недостающей информацией!
Пример прасера на Nodejs с обходом РКН через Tor на телефоне андроид c termux
termux
https://f-droid.org/ru/packages/com.termux/
install ubuntu
https://wiki.termux.com/wiki/PRoot
https://asciinema.org/a/355177
terminal ubuntu
sudo apt update sudo apt install nodejs sudo apt install npm sudo apt install tor
sudo apt install obfs4proxy
-- if Failed to fetch http://ports.ubuntu.com/ubuntu-ports/pool/universe/o/obfs4proxy/obfs4proxy_0.0.8-1build2_arm64.deb
-- install optional from http://ports.ubuntu.com/ubuntu-ports/pool/universe/o/obfs4proxy/
TEMP_DEB="$(mktemp)" && wget -O "$TEMP_DEB" 'http://ports.ubuntu.com/ubuntu-ports/pool/universe/o/obfs4proxy/obfs4proxy_0.0.8-1_arm64.deb' && sudo dpkg -i "$TEMP_DEB" rm -f "$TEMP_DEB"
Get tor bridges https://bridges.torproject.org/options/
Add works bridges in '/etc/tor/torrc' Ubuntu
https://www.npmjs.com/package/tor-request
npm install tor-request
npm install pm2 -g
Autostart tor + kinobase.js
add in 'etc/profile' Ubuntu
pm2 start 'DIRname'/kinobase.js;
tor --ExitNodes {ua},{by},{am},{kz};
Autostart vnc server with port:5901 and ip:192.168.1.xx
add in 'etc/profile' Ubuntu
rm -rf /tmp/.X1-lock;
rm -rf /tmp/.X11-unix/X1;
vncserver -localhost no;
kinobase.js
const http = require('http'); const https = require('https'); const tr = require('tor-request'); const hostname = '192.168.1.66'; // ip телефона в wifi const port = 8001; // port const server = http.createServer((req, res) => { const headers = { 'Access-Control-Allow-Origin': '*', 'Access-Control-Allow-Methods': 'OPTIONS, POST, GET', 'Access-Control-Allow-Headers': 'Accept, Content-Type', 'Content-Type': 'text/html; charset=utf-8', }; res.writeHead(200, headers); function returnXml(a, b, c) { if (!c)c = ""; return "<channel>\n\ <title>" + a + " </title>\n\ <stream_url>" + b + "</stream_url>\n\ <description><div style='font-size:24px'>" + c + "</div>\n\ </channel>\n"; } if (e = req.url.match(/getm3u8\/(.*?.)((\?|&)box_|$)/)) { var urlreq = decodeURI(e[1].toString()); var com = "<items> \n" tr.request(urlreq, function(err, resTOR, body) { if (!err && resTOR.statusCode == 200) { const regexT = /<div class="data clearfix">(.*?)<div id="watch_block">/gms; while ((m = regexT.exec(body)) !== null) { com += returnXml('info', 0, ((m[1].replace(/(<([^>]+)>)|\n/gi, "")).trim()).replace(/\s\s\s+/gi, "<br>")) } const regf = /MOVIE_ID = (\d+).*?PLAYER_CUID = "(.*?)".*?IDENTIFIER = "(.*?)"/gms; while ((m = regf.exec(body)) !== null) { var idm = m[1]; var CUID = m[2]; var IDENTIFIER = m[3]; var time = Math.floor(Date.now() / 100) + '00'; var requrl = 'https://kinobase.org/user_data?page=movie&movie_id=' + idm + '&cuid=' + CUID + '&device=DESKTOP&_=' + time; tr.request(requrl, function(err, resTOR, body) { if (!err && resTOR.statusCode == 200) { const regs = /"vod_time":(\d+),"vod_hash":"(.*?)"/gms; while ((m = regs.exec(body)) !== null) { var vod_time = m[1]; var vod_hash = m[2]; time = Date.now(); var vodurl = 'https://kinobase.org/vod/' + idm + '?identifier=' + IDENTIFIER + '&player_type=new&file_type=hls&st=' + vod_hash + '&e=' + vod_time + '&_=' + time; tr.request(vodurl, function(err, resTOR, body) { if (!err && resTOR.statusCode == 200) { let f; var regex = /pl\|(.*?)\|/gm; while ((m = regex.exec(body)) !== null) {f = JSON.parse(m[1])} //Playlist if (Array.isArray(f)) { for (let x of f) { if (x.playlist) { // have seasons for (let y of x.playlist) { var regex = /\[(.*?)\](.*?)(,|$)/gm; // have quality while ((m = regex.exec(y.file)) !== null) { if (/\{/.test(m[2])) { var regex2 = /\{(.*?)\}(http.*?.mp4|http.*?.m3u8)\s+/gi; // have translate while ((v = regex2.exec(m[2])) !== null) {com += returnXml(x.comment + " " + y.comment + " " + m[1] + " " + v[1], v[2], 0);} } else { var regex2 = /(http.*?.mp4|http.*?.m3u8)\s+/gi; // without translate while ((v = regex2.exec(m[2])) !== null) {com += returnXml(x.comment + " " + y.comment + " " + m[1], v[1], 0);} } } } } else { // no have seasons var regex = /\[(.*?)\](.*?)(,|$)/gm; while ((m = regex.exec(x.file)) !== null) { var t = m[2]; //file if (/\{/.test(t)) { var regex2 = /\{(.*?)\}(http.*?.mp4|http.*?.m3u8)\s+/gi; // have translate while ((v = regex2.exec(t)) !== null) {com += returnXml(x.comment + " " + m[1] + " " + v[1], v[2], 0);} } else { var regex2 = /(http.*?.mp4|http.*?.m3u8)\s+/gi; // without translate while ((v = regex2.exec(t)) !== null) {com += returnXml(x.comment + " " + m[1], v[1], 0);} } } } } } // not playlist else { var regex = /\[(.*?)\](.*?)(,|$)/gm; while ((m = regex.exec(body)) !== null) { var t = m[2]; //file if (/\{/.test(t)) { var regex2 = /\{(.*?)\}(http.*?.mp4|http.*?.m3u8)\s+/gi; // have translate while ((v = regex2.exec(t)) !== null) {com += returnXml(m[1] + " " + v[1], v[2], 0);} } else { var regex2 = /(http.*?.mp4|http.*?.m3u8)\s+/gi; // without translate while ((v = regex2.exec(t)) !== null) {com += returnXml(m[1], v[1], 0);} } } } com += "</items> \n" res.write(com); res.end(); } }) } } }) } } }) } }); server.listen(port, hostname, () => {console.log("Server running at "+hostname+":"+port);});
адрес для телевизора http://192.168.1.66:8001/getm3u8/https://kinobase.org/film/4756-radiovolna