mirror of
				https://git.sr.ht/~cadence/bibliogram
				synced 2025-10-31 03:25:36 +00:00 
			
		
		
		
	Basic error checking
This commit is contained in:
		
							parent
							
								
									c2ac1b2259
								
							
						
					
					
						commit
						9f5fa84f9a
					
				| @ -10,13 +10,16 @@ const timelineEntryCache = new TtlCache(constants.resource_cache_time) | |||||||
| 
 | 
 | ||||||
| function fetchUser(username) { | function fetchUser(username) { | ||||||
| 	return requestCache.getOrFetch("user/"+username, () => { | 	return requestCache.getOrFetch("user/"+username, () => { | ||||||
| 		return request(`https://www.instagram.com/${username}/`).then(res => res.text()).then(text => { | 		return request(`https://www.instagram.com/${username}/`).then(res => { | ||||||
| 			// require down here or have to deal with require loop. require cache will take care of it anyway.
 | 			if (res.status === 404) throw constants.symbols.NOT_FOUND | ||||||
| 			// User -> Timeline -> TimelineImage -> collectors -/> User
 | 			else return res.text().then(text => { | ||||||
| 			const User = require("./structures/User") | 				// require down here or have to deal with require loop. require cache will take care of it anyway.
 | ||||||
| 			const sharedData = extractSharedData(text) | 				// User -> Timeline -> TimelineImage -> collectors -/> User
 | ||||||
| 			const user = new User(sharedData.entry_data.ProfilePage[0].graphql.user) | 				const User = require("./structures/User") | ||||||
| 			return user | 				const sharedData = extractSharedData(text) | ||||||
|  | 				const user = new User(sharedData.entry_data.ProfilePage[0].graphql.user) | ||||||
|  | 				return user | ||||||
|  | 			}) | ||||||
| 		}) | 		}) | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| @ -24,7 +27,7 @@ function fetchUser(username) { | |||||||
| /** | /** | ||||||
|  * @param {string} userID |  * @param {string} userID | ||||||
|  * @param {string} after |  * @param {string} after | ||||||
|  * @returns {Promise<import("./types").PagedEdges<import("./types").GraphImage>>} |  * @returns {Promise<import("./types").PagedEdges<import("./types").TimelineEntryN2>>} | ||||||
|  */ |  */ | ||||||
| function fetchTimelinePage(userID, after) { | function fetchTimelinePage(userID, after) { | ||||||
| 	const p = new URLSearchParams() | 	const p = new URLSearchParams() | ||||||
| @ -36,7 +39,7 @@ function fetchTimelinePage(userID, after) { | |||||||
| 	})) | 	})) | ||||||
| 	return requestCache.getOrFetchPromise("page/"+after, () => { | 	return requestCache.getOrFetchPromise("page/"+after, () => { | ||||||
| 		return request(`https://www.instagram.com/graphql/query/?${p.toString()}`).then(res => res.json()).then(root => { | 		return request(`https://www.instagram.com/graphql/query/?${p.toString()}`).then(res => res.json()).then(root => { | ||||||
| 			/** @type {import("./types").PagedEdges<import("./types").GraphImage>} */ | 			/** @type {import("./types").PagedEdges<import("./types").TimelineEntryN2>} */ | ||||||
| 			const timeline = root.data.user.edge_owner_to_timeline_media | 			const timeline = root.data.user.edge_owner_to_timeline_media | ||||||
| 			return timeline | 			return timeline | ||||||
| 		}) | 		}) | ||||||
| @ -86,7 +89,12 @@ function fetchShortcodeData(shortcode) { | |||||||
| 		return request(`https://www.instagram.com/graphql/query/?${p.toString()}`).then(res => res.json()).then(root => { | 		return request(`https://www.instagram.com/graphql/query/?${p.toString()}`).then(res => res.json()).then(root => { | ||||||
| 			/** @type {import("./types").TimelineEntryN3} */ | 			/** @type {import("./types").TimelineEntryN3} */ | ||||||
| 			const data = root.data.shortcode_media | 			const data = root.data.shortcode_media | ||||||
| 			return data | 			if (data == null) { | ||||||
|  | 				// the thing doesn't exist
 | ||||||
|  | 				throw constants.symbols.NOT_FOUND | ||||||
|  | 			} else { | ||||||
|  | 				return data | ||||||
|  | 			} | ||||||
| 		}) | 		}) | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  | |||||||
| @ -16,6 +16,8 @@ module.exports = { | |||||||
| 		TYPE_VIDEO: Symbol("TYPE_VIDEO"), | 		TYPE_VIDEO: Symbol("TYPE_VIDEO"), | ||||||
| 		TYPE_GALLERY: Symbol("TYPE_GALLERY"), | 		TYPE_GALLERY: Symbol("TYPE_GALLERY"), | ||||||
| 		TYPE_GALLERY_IMAGE: Symbol("TYPE_GALLERY_IMAGE"), | 		TYPE_GALLERY_IMAGE: Symbol("TYPE_GALLERY_IMAGE"), | ||||||
| 		TYPE_GALLERY_VIDEO: Symbol("TYPE_GALLERY_VIDEO") | 		TYPE_GALLERY_VIDEO: Symbol("TYPE_GALLERY_VIDEO"), | ||||||
|  | 		NOT_FOUND: Symbol("NOT_FOUND"), | ||||||
|  | 		NO_SHARED_DATA: Symbol("NO_SHARED_DATA") | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -19,8 +19,9 @@ class TimelineEntry extends TimelineBaseMethods { | |||||||
| 		/** @type {import("../types").TimelineEntryAll} some properties may not be available yet! */ | 		/** @type {import("../types").TimelineEntryAll} some properties may not be available yet! */ | ||||||
| 		// @ts-ignore
 | 		// @ts-ignore
 | ||||||
| 		this.data = {} | 		this.data = {} | ||||||
|  | 		const error = new Error("TimelineEntry data was not initalised in same event loop (missing __typename)") // initialise here for a useful stack trace
 | ||||||
| 		setImmediate(() => { // next event loop
 | 		setImmediate(() => { // next event loop
 | ||||||
| 			if (!this.data.__typename) throw new Error("TimelineEntry data was not initalised in same event loop (missing __typename)") | 			if (!this.data.__typename) throw error | ||||||
| 		}) | 		}) | ||||||
| 		/** @type {string} Not available until fetchExtendedOwnerP is called */ | 		/** @type {string} Not available until fetchExtendedOwnerP is called */ | ||||||
| 		this.ownerPfpCacheP = null | 		this.ownerPfpCacheP = null | ||||||
| @ -29,8 +30,12 @@ class TimelineEntry extends TimelineBaseMethods { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	async update() { | 	async update() { | ||||||
| 		const data = await collectors.fetchShortcodeData(this.data.shortcode) | 		return collectors.fetchShortcodeData(this.data.shortcode).then(data => { | ||||||
| 		this.applyN3(data) | 			this.applyN3(data) | ||||||
|  | 		}).catch(error => { | ||||||
|  | 			console.error("TimelineEntry could not self-update; trying to continue anyway...") | ||||||
|  | 			console.error("E:", error) | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/** | 	/** | ||||||
|  | |||||||
| @ -1,3 +1,4 @@ | |||||||
|  | const constants = require("../constants") | ||||||
| const {Parser} = require("./parser/parser") | const {Parser} = require("./parser/parser") | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @ -5,7 +6,8 @@ const {Parser} = require("./parser/parser") | |||||||
|  */ |  */ | ||||||
| function extractSharedData(text) { | function extractSharedData(text) { | ||||||
| 	const parser = new Parser(text) | 	const parser = new Parser(text) | ||||||
| 	parser.seek("window._sharedData = ", {moveToMatch: true, useEnd: true}) | 	const index = parser.seek("window._sharedData = ", {moveToMatch: true, useEnd: true}) | ||||||
|  | 	if (index === -1) throw constants.symbols.NO_SHARED_DATA | ||||||
| 	parser.store() | 	parser.store() | ||||||
| 	const end = parser.seek(";</script>") | 	const end = parser.seek(";</script>") | ||||||
| 	parser.restore() | 	parser.restore() | ||||||
|  | |||||||
| @ -3,14 +3,25 @@ const {fetchUser} = require("../../lib/collectors") | |||||||
| const {render} = require("pinski/plugins") | const {render} = require("pinski/plugins") | ||||||
| 
 | 
 | ||||||
| module.exports = [ | module.exports = [ | ||||||
| 	{route: `/u/(${constants.external.username_regex})/rss.xml`, methods: ["GET"], code: async ({url, fill}) => { | 	{route: `/u/(${constants.external.username_regex})/rss.xml`, methods: ["GET"], code: ({fill}) => { | ||||||
| 		const user = await fetchUser(fill[0]) | 		return fetchUser(fill[0]).then(async user => { | ||||||
| 		const content = await user.timeline.fetchFeed() | 			const content = await user.timeline.fetchFeed() | ||||||
| 		const xml = content.xml() | 			const xml = content.xml() | ||||||
| 		return { | 			return { | ||||||
| 			statusCode: 200, | 				statusCode: 200, | ||||||
| 			contentType: "application/rss+xml", // see https://stackoverflow.com/questions/595616/what-is-the-correct-mime-type-to-use-for-an-rss-feed
 | 				contentType: "application/rss+xml", // see https://stackoverflow.com/questions/595616/what-is-the-correct-mime-type-to-use-for-an-rss-feed
 | ||||||
| 			content: xml | 				content: xml | ||||||
| 		} | 			} | ||||||
|  | 		}).catch(error => { | ||||||
|  | 			if (error === constants.symbols.NOT_FOUND) { | ||||||
|  | 				return render(404, "pug/friendlyerror.pug", { | ||||||
|  | 					statusCode: 404, | ||||||
|  | 					title: "Not found", | ||||||
|  | 					message: "This user doesn't exist." | ||||||
|  | 				}) | ||||||
|  | 			} else { | ||||||
|  | 				throw error | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
| 	}} | 	}} | ||||||
| ] | ] | ||||||
|  | |||||||
| @ -4,39 +4,72 @@ const {render} = require("pinski/plugins") | |||||||
| 
 | 
 | ||||||
| module.exports = [ | module.exports = [ | ||||||
| 	{ | 	{ | ||||||
| 		route: `/u/(${constants.external.username_regex})`, methods: ["GET"], code: async ({url, fill}) => { | 		route: `/u/(${constants.external.username_regex})`, methods: ["GET"], code: ({url, fill}) => { | ||||||
| 			const params = url.searchParams | 			const params = url.searchParams | ||||||
| 			const user = await fetchUser(fill[0]) | 			return fetchUser(fill[0]).then(async user => { | ||||||
| 			const page = +params.get("page") | 				const page = +params.get("page") | ||||||
| 			if (typeof page === "number" && !isNaN(page) && page >= 1) { | 				if (typeof page === "number" && !isNaN(page) && page >= 1) { | ||||||
| 				await user.timeline.fetchUpToPage(page - 1) | 					await user.timeline.fetchUpToPage(page - 1) | ||||||
| 			} | 				} | ||||||
| 			return render(200, "pug/user.pug", {url, user}) | 				return render(200, "pug/user.pug", {url, user}) | ||||||
|  | 			}).catch(error => { | ||||||
|  | 				if (error === constants.symbols.NOT_FOUND) { | ||||||
|  | 					return render(404, "pug/friendlyerror.pug", { | ||||||
|  | 						statusCode: 404, | ||||||
|  | 						title: "Not found", | ||||||
|  | 						message: "This user doesn't exist." | ||||||
|  | 					}) | ||||||
|  | 				} else { | ||||||
|  | 					throw error | ||||||
|  | 				} | ||||||
|  | 			}) | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ | 	{ | ||||||
| 		route: `/fragment/user/(${constants.external.username_regex})/(\\d+)`, methods: ["GET"], code: async ({url, fill}) => { | 		route: `/fragment/user/(${constants.external.username_regex})/(\\d+)`, methods: ["GET"], code: async ({url, fill}) => { | ||||||
| 			const user = await fetchUser(fill[0]) | 			return fetchUser(fill[0]).then(async user => { | ||||||
| 			const pageNumber = +fill[1] | 				const pageNumber = +fill[1] | ||||||
| 			const pageIndex = pageNumber - 1 | 				const pageIndex = pageNumber - 1 | ||||||
| 			await user.timeline.fetchUpToPage(pageIndex) | 				await user.timeline.fetchUpToPage(pageIndex) | ||||||
| 			if (user.timeline.pages[pageIndex]) { | 				if (user.timeline.pages[pageIndex]) { | ||||||
| 				return render(200, "pug/fragments/timeline_page.pug", {page: user.timeline.pages[pageIndex], pageIndex, user, url}) | 					return render(200, "pug/fragments/timeline_page.pug", {page: user.timeline.pages[pageIndex], pageIndex, user, url}) | ||||||
| 			} else { | 				} else { | ||||||
| 				return { | 					return { | ||||||
| 					statusCode: 400, | 						statusCode: 400, | ||||||
| 					contentType: "text/html", | 						contentType: "text/html", | ||||||
| 					content: "That page does not exist" | 						content: "That page does not exist" | ||||||
|  | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			}).catch(error => { | ||||||
|  | 				if (error === constants.symbols.NOT_FOUND) { | ||||||
|  | 					return render(404, "pug/friendlyerror.pug", { | ||||||
|  | 						statusCode: 404, | ||||||
|  | 						title: "Not found", | ||||||
|  | 						message: "This user doesn't exist." | ||||||
|  | 					}) | ||||||
|  | 				} else { | ||||||
|  | 					throw error | ||||||
|  | 				} | ||||||
|  | 			}) | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ | 	{ | ||||||
| 		route: `/p/(${constants.external.shortcode_regex})`, methods: ["GET"], code: async ({fill}) => { | 		route: `/p/(${constants.external.shortcode_regex})`, methods: ["GET"], code: ({fill}) => { | ||||||
| 			const post = await getOrFetchShortcode(fill[0]) | 			return getOrFetchShortcode(fill[0]).then(async post => { | ||||||
| 			await post.fetchChildren() | 				await post.fetchChildren() | ||||||
| 			await post.fetchExtendedOwnerP() // parallel await is okay since intermediate fetch result is cached
 | 				await post.fetchExtendedOwnerP() // parallel await is okay since intermediate fetch result is cached
 | ||||||
| 			return render(200, "pug/post.pug", {post}) | 				return render(200, "pug/post.pug", {post}) | ||||||
|  | 			}).catch(error => { | ||||||
|  | 				if (error === constants.symbols.NOT_FOUND) { | ||||||
|  | 					return render(404, "pug/friendlyerror.pug", { | ||||||
|  | 						statusCode: 404, | ||||||
|  | 						title: "Not found", | ||||||
|  | 						message: "Somehow, you reached a post that doesn't exist." | ||||||
|  | 					}) | ||||||
|  | 				} else { | ||||||
|  | 					throw error | ||||||
|  | 				} | ||||||
|  | 			}) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| ] | ] | ||||||
|  | |||||||
| @ -36,7 +36,6 @@ class NextPage extends FreezeWidth { | |||||||
| 		fetch(`/fragment/user/${this.element.getAttribute("data-username")}/${this.nextPageNumber}`).then(res => res.text()).then(text => { | 		fetch(`/fragment/user/${this.element.getAttribute("data-username")}/${this.nextPageNumber}`).then(res => res.text()).then(text => { | ||||||
| 			q("#next-page-container").remove() | 			q("#next-page-container").remove() | ||||||
| 			this.observer.disconnect() | 			this.observer.disconnect() | ||||||
| 
 |  | ||||||
| 			q("#timeline").insertAdjacentHTML("beforeend", text) | 			q("#timeline").insertAdjacentHTML("beforeend", text) | ||||||
| 			addNextPageControl() | 			addNextPageControl() | ||||||
| 		}) | 		}) | ||||||
|  | |||||||
							
								
								
									
										18
									
								
								src/site/pug/friendlyerror.pug
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/site/pug/friendlyerror.pug
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | //- Needs title, message, statusCode | ||||||
|  | 
 | ||||||
|  | include includes/timeline_page.pug | ||||||
|  | include includes/next_page_button.pug | ||||||
|  | 
 | ||||||
|  | - const numberFormat = new Intl.NumberFormat().format | ||||||
|  | 
 | ||||||
|  | doctype html | ||||||
|  | html | ||||||
|  | 	head | ||||||
|  | 		meta(charset="utf-8") | ||||||
|  | 		meta(name="viewport" content="width=device-width, initial-scale=1") | ||||||
|  | 		title= `${title} | Bibliogram` | ||||||
|  | 		link(rel="stylesheet" type="text/css" href="/static/css/main.css") | ||||||
|  | 	body.error-page | ||||||
|  | 		h1.code= statusCode | ||||||
|  | 		p.message= message | ||||||
|  | 		a(href="javascript:history.back()").back ← Go back? | ||||||
| @ -242,3 +242,32 @@ body | |||||||
| 
 | 
 | ||||||
| 				&:not(:last-child) | 				&:not(:last-child) | ||||||
| 					margin-bottom: 10px | 					margin-bottom: 10px | ||||||
|  | 
 | ||||||
|  | .error-page | ||||||
|  | 	min-height: 100vh | ||||||
|  | 	background: #191919 | ||||||
|  | 	padding: 10px | ||||||
|  | 	text-align: center | ||||||
|  | 	display: flex | ||||||
|  | 	flex-direction: column | ||||||
|  | 	justify-content: center | ||||||
|  | 	align-items: center | ||||||
|  | 
 | ||||||
|  | 	.code, .message, .back-link | ||||||
|  | 		line-height: 1.2 | ||||||
|  | 		margin: 0px | ||||||
|  | 		color: white | ||||||
|  | 
 | ||||||
|  | 	.code | ||||||
|  | 		font-size: 80px | ||||||
|  | 		color: #fff | ||||||
|  | 		margin-bottom: 25px | ||||||
|  | 
 | ||||||
|  | 	.message | ||||||
|  | 		font-size: 35px | ||||||
|  | 		color: #ccc | ||||||
|  | 		margin-bottom: 15vh | ||||||
|  | 
 | ||||||
|  | 	.back | ||||||
|  | 		color: #4a93d2 | ||||||
|  | 		font-size: 25px | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user