1
0
mirror of https://git.sr.ht/~cadence/cloudtube synced 2024-11-14 12:27:28 +00:00

Refactor video access route

- Move errors to separate files instead of embedding
- Consistent interface for error generation
- Move renderVideo function into route
- Fix unhandled FetchError regression from dbbe950
This commit is contained in:
Cadence Ember 2021-04-28 00:56:00 +12:00
parent 94e12a2ee8
commit 81a4d10474
No known key found for this signature in database
GPG Key ID: BC1C2C61CF521B17
7 changed files with 110 additions and 94 deletions

View File

@ -14,6 +14,9 @@ class InstanceError extends Error {
} }
} }
class MessageError extends Error {
}
function formatOrder(format) { function formatOrder(format) {
// most significant to least significant // most significant to least significant
// key, max, order, transform // key, max, order, transform
@ -90,9 +93,46 @@ function sortFormats(video, preference) {
return formats return formats
} }
async function renderVideo(video, {user, settings, id, instanceOrigin}, locals = {}) { module.exports = [
{
route: "/watch", methods: ["GET", "POST"], upload: true, code: async ({req, url, body}) => {
// Prepare data needed to render video page
const user = getUser(req)
const settings = user.getSettingsOrDefaults()
const id = url.searchParams.get("v")
// Media fragment
const t = url.searchParams.get("t")
let mediaFragment = converters.tToMediaFragment(t)
// Continuous mode
const continuous = url.searchParams.get("continuous") === "1"
const autoplay = url.searchParams.get("autoplay") === "1"
const swp = url.searchParams.get("session-watched")
const sessionWatched = swp ? swp.split(" ") : []
const sessionWatchedNext = sessionWatched.concat([id]).join("+")
if (continuous) settings.quality = 0 // autoplay with synced streams does not work
// Work out how to fetch the video
if (req.method === "GET") {
if (settings.local) { // skip to the local fetching page, which will then POST video data in a moment
return render(200, "pug/local-video.pug", {id})
}
var instanceOrigin = settings.instance
var outURL = `${instanceOrigin}/api/v1/videos/${id}`
var videoFuture = request(outURL).then(res => res.json())
} else { // req.method === "POST"
var instanceOrigin = "http://localhost:3000"
var videoFuture = JSON.parse(new URLSearchParams(body.toString()).get("video"))
}
try { try {
if (!video) throw new Error("The instance returned null.") // Fetch the video
const video = await videoFuture
// Error handling
if (!video) throw new MessageError("The instance returned null.")
if (video.error) throw new InstanceError(video.error, video.identifier) if (video.error) throw new InstanceError(video.error, video.identifier)
// process stream list ordering // process stream list ordering
@ -123,96 +163,35 @@ async function renderVideo(video, {user, settings, id, instanceOrigin}, locals =
// rewrite description // rewrite description
video.descriptionHtml = converters.rewriteVideoDescription(video.descriptionHtml, id) video.descriptionHtml = converters.rewriteVideoDescription(video.descriptionHtml, id)
return render(200, "pug/video.pug", Object.assign(locals, {video, formats, subscribed, instanceOrigin})) return render(200, "pug/video.pug", {
} catch (e) { video, formats, subscribed, instanceOrigin, mediaFragment, autoplay, continuous,
// show an appropriate error message sessionWatched, sessionWatchedNext
// these should probably be split out to their own files })
let message = pug.render("pre= error", {error: e.stack || e.toString()})
if (e instanceof fetch.FetchError) { } catch (error) {
const template = ` // Something went wrong, somewhere! Find out where.
p The selected instance, #[code= instanceOrigin], did not respond correctly.
` let errorType = "unrecognised-error"
message = pug.render(template, {instanceOrigin}) const locals = {instanceOrigin, error}
} else if (e instanceof InstanceError) {
if (e.identifier === "RATE_LIMITED_BY_YOUTUBE" || e.message === "Could not extract video info. Instance is likely blocked.") { // Sort error category
const template = ` if (error instanceof fetch.FetchError) {
.blocked-explanation errorType = "fetch-error"
img(src="/static/images/instance-blocked.svg" width=552 height=96) } else if (error instanceof MessageError) {
.rows errorType = "message-error"
.row } else if (error instanceof InstanceError) {
h3.actor You if (error.identifier === "RATE_LIMITED_BY_YOUTUBE" || error.message === "Could not extract video info. Instance is likely blocked.") {
| Working errorType = "rate-limited"
.row
h3.actor CloudTube
| Working
.row
h3.actor Instance
| Blocked by YouTube
.row
h3.actor YouTube
| Working
p.
CloudTube needs a working NewLeaf/Invidious instance in order to get data about videos.
However, the selected instance, #[code= instanceOrigin], has been temporarily blocked by YouTube.
p.
You will be able to watch this video if you select a working instance in settings.
#[br]#[a(href="/settings") Go to settings ]
p.
(Tip: Try #[code https://invidious.snopyta.org] or #[code https://invidious.site].)
p.
This situation #[em will] be improved in the future!
`
message = pug.render(template, {instanceOrigin})
} else { } else {
const template = ` errorType = "instance-error"
p #[strong= error.message]
if error.identifier
p #[code= error.identifier]
p That error was generated by #[code= instanceOrigin].
`
message = pug.render(template, {instanceOrigin, error: e})
} }
} }
// Create appropriate formatted message
const message = render(0, `pug/errors/${errorType}.pug`, locals).content
return render(500, "pug/video.pug", {video: {videoId: id}, error: true, message}) return render(500, "pug/video.pug", {video: {videoId: id}, error: true, message})
} }
}
module.exports = [
{
route: "/watch", methods: ["GET", "POST"], upload: true, code: async ({req, url, body}) => {
const user = getUser(req)
const settings = user.getSettingsOrDefaults()
const id = url.searchParams.get("v")
// Media fragment
const t = url.searchParams.get("t")
let mediaFragment = converters.tToMediaFragment(t)
// Continuous mode
const continuous = url.searchParams.get("continuous") === "1"
const autoplay = url.searchParams.get("autoplay") === "1"
const swp = url.searchParams.get("session-watched")
const sessionWatched = swp ? swp.split(" ") : []
const sessionWatchedNext = sessionWatched.concat([id]).join("+")
if (continuous) settings.quality = 0 // autoplay with synced streams does not work
if (req.method === "GET") {
if (settings.local) { // skip to the local fetching page, which will then POST video data in a moment
return render(200, "pug/local-video.pug", {id})
}
var instanceOrigin = settings.instance
var outURL = `${instanceOrigin}/api/v1/videos/${id}`
var video = await request(outURL).then(res => res.json())
} else { // req.method === "POST"
var instanceOrigin = "http://localhost:3000"
var video = JSON.parse(new URLSearchParams(body.toString()).get("video"))
}
return renderVideo(video, {
user, settings, id, instanceOrigin
}, {
mediaFragment, autoplay, continuous, sessionWatched, sessionWatchedNext
})
} }
} }
] ]

View File

@ -0,0 +1,5 @@
p The selected instance, #[code= instanceOrigin], was unreachable.
details
summary If the instance is on a private network
p The instance URL was resolved by the server. If the instance is on a private network, CloudTube will not be able to connect back to it.
p To get around this error, you may be able to use local mode, or you can run your own CloudTube within the network.

View File

@ -0,0 +1,4 @@
p #[strong= error.message]
if error.identifier
p #[code= error.identifier]
p That error was generated by #[code= instanceOrigin].

View File

@ -0,0 +1 @@
pre= error.toString()

View File

@ -0,0 +1,25 @@
.blocked-explanation
img(src="/static/images/instance-blocked.svg" width=552 height=96)
.rows
.row
h3.actor You
| Working
.row
h3.actor CloudTube
| Working
.row
h3.actor Instance
| Blocked by YouTube
.row
h3.actor YouTube
| Working
p.
CloudTube needs a working NewLeaf/Invidious instance in order to get data about videos.
However, the selected instance, #[code= instanceOrigin], has been temporarily blocked by YouTube.
p.
You will be able to watch this video if you select a working instance in settings.
#[br]#[a(href="/settings") Go to settings →]
p.
(Tip: Try #[code https://invidious.snopyta.org] or #[code https://invidious.site].)
p.
This situation #[em will] be improved in the future!

View File

@ -0,0 +1 @@
pre= error.stack || error.toString()

View File

@ -16,6 +16,7 @@ const {setInstance} = require("pinski/plugins")
server.addRoute("/static/css/main.css", "sass/main.sass", "sass") server.addRoute("/static/css/main.css", "sass/main.sass", "sass")
server.addPugDir("pug", ["pug/includes"]) server.addPugDir("pug", ["pug/includes"])
server.addPugDir("pug/errors")
server.addRoute("/cant-think", "pug/cant-think.pug", "pug") server.addRoute("/cant-think", "pug/cant-think.pug", "pug")
server.addRoute("/privacy", "pug/privacy.pug", "pug") server.addRoute("/privacy", "pug/privacy.pug", "pug")
server.addRoute("/js-licenses", "pug/js-licenses.pug", "pug") server.addRoute("/js-licenses", "pug/js-licenses.pug", "pug")