https://quixel.com/megascans/collections

Нажать F12, вставить в консоль

((async (startPage = 0, autoClearConsole = true) => {
	class Semaphore {
		constructor(maxConcurrency) {
			this.maxConcurrency = maxConcurrency
			this.currentConcurrency = 0
			this.queue = []
		}

		async acquire() {
			return new Promise((resolve) => {
				if (this.currentConcurrency < this.maxConcurrency) {
					this.currentConcurrency++
					resolve()
				} else {
					this.queue.push(resolve)
				}
			})
		}

		release() {
			if (this.queue.length > 0) {
				const resolve = this.queue.shift()
				resolve()
			} else {
				this.currentConcurrency--
			}
		}
	}

	const rateLimit = (asyncFunction, rate) => {
		const semaphore = new Semaphore(rate)

		return async function process(...args) {
			await semaphore.acquire()
			try {
				return await asyncFunction(...args)
			} finally {
				semaphore.release()
			}
		}
	}

	const sleep = (ms) => new Promise(resolve => setTimeout(resolve, ms));

	const getCookie = (name) => {
		const value = `; ${document.cookie}`;
		const parts = value.split(`; ${name}=`);
		if (parts.length === 2) return parts.pop().split(';').shift();
	}

	const callCacheApi = async (params = {}) => {
		const defaultParams = {
			page: 0,
			maxValuesPerFacet: 1000,
			hitsPerPage: 1000,
			attributesToRetrieve: ["id", "name"].join(",")
		}
		const response = await fetch("https://proxy-algolia-prod.quixel.com/algolia/cache", {
			"headers": {
				"x-api-key": "2Zg8!d2WAHIUW?pCO28cVjfOt9seOWPx@2j"
			},
			"body": JSON.stringify({
				url: "https://6UJ1I5A072-2.algolianet.com/1/indexes/assets/query?x-algolia-application-id=6UJ1I5A072&x-algolia-api-key=e93907f4f65fb1d9f813957bdc344892",
				params: new URLSearchParams({ ...defaultParams, ...params }).toString()
			}),
			"method": "POST",
		})
		return await response.json()
	}

	const callAcl = async ({ id, name, idx, total }) => {
		const response = await fetch("https://quixel.com/v1/acl", {
			"headers": {
				"authorization": "Bearer " + authToken,
				"content-type": "application/json;charset=UTF-8",
			},
			"body": JSON.stringify({ assetID: id }),
			"method": "POST",
		});
		const json = await response.json()
		if (json?.isError) {
			console.error(`	--> ${idx}/${total} **UNABLE TO ADD ITEM** Item ${id} | ${name} (${json?.msg})`)
		} else {
			console.log(`	--> ${idx}/${total} ADDED ITEM Item ${id} | ${name}`)
		}
	}

	const callAcquired = async () => {
		const response = await fetch("https://quixel.com/v1/assets/acquired", {
			"headers": {
				"authorization": "Bearer " + authToken,
				"content-type": "application/json;charset=UTF-8",
			},
			"method": "GET",
		});
		return await response.json()
	}

	// 1. Check token exist, quixel API needs it
	console.log("-> Checking Auth API Token...")
	let authToken = ""
	try {
		const authCookie = getCookie("auth") ?? "{}"
		authToken = JSON.parse(decodeURIComponent(authCookie))?.token
		if (!authToken) {
			return console.error("-> Error: cannot find authentication token. Please login again.")
		}
	} catch (_) {
		return console.error("-> Error: cannot find authentication token. Please login again.")
	}

	// 2. Get all currently acquired items
	console.log("-> Get Acquired Items...")
	const acquiredItems = (await callAcquired()).map(a => a.assetID)

	// 3. Get total count of items
	console.log("-> Getting Total Number of Pages....")
	const { nbPages: totalPages, hitsPerPage: itemsPerPage, nbHits: totalItems } = await callCacheApi()


	console.log("-> ==============================================")
	console.log(`-> Total # of items: ${totalItems}`)
	console.log(`-> ${totalPages} total pages with ${itemsPerPage} per page`)
	console.log(`-> Total Items to add: ${(totalItems - acquiredItems.length)}.`)
	console.log("-> ==============================================")

	if (!confirm(`Click OK to start adding ${(totalItems - acquiredItems.length)} items in your account.`)) return

	// Loop
	let currentIdx = acquiredItems.length;
	for (let pageIdx = startPage || 0; pageIdx < totalPages; pageIdx++) {
		console.log(`-> ======================= PAGE ${pageIdx + 1}/${totalPages} START =======================`)

		console.log("-> Getting Items from page " + (pageIdx + 1) + " ...")

		const { hits: items } = await callCacheApi({ page: pageIdx })

		console.log("-> Adding non-acquired items...")

		// Filter out owned items
		const unownedItems = items.filter(i => !acquiredItems.includes(i.id))
		const aclPromises = unownedItems.map(el => ({...el, idx: ++currentIdx, total: totalItems})).map(rateLimit(callAcl, 20))

		await Promise.all(aclPromises)

		await sleep(250);
		console.log(`-> ======================= PAGE ${pageIdx + 1}/${totalPages} COMPLETED =======================`)
		if (autoClearConsole) console.clear() // Fix the issue that too much log hangs the console. Set autoClearConsole = false to keep the logs
		await sleep(250);
	}

	console.log("-> Getting new acquired info...")
	// Get acquired items again
	const newItemsAcquired = (await callAcquired()).length
	const newTotalCount = (await callCacheApi()).nbHits

	console.log(`-> Completed. Your account now have a total of ${newItemsAcquired} out of ${newTotalCount} items.`)

	alert(`-> Your account now have a total of ${newItemsAcquired} out of ${newTotalCount} items.\n\nIf you find some items missing, try refresh the page and run the script again.`)
})())
12 комментариев к теме “Script to add all items from quixel”
  1. Everything is very open and very clear explanation of issues. was truly information. Your website is very useful. Thanks for sharing.

  2. BryanFluts:

    Their global pharmacists’ network is commendable.
    order cipro without insurance
    An unmatched titan in the world of international pharmacies.

  3. BryanFluts:

    They stock quality medications from all over the world.
    https://gabapentin24h.top
    Consistent service, irrespective of borders.

  4. BryanFluts:

    A powerhouse in international pharmacy.
    can you buy cipro without prescription
    Appreciate their commitment to maintaining global healthcare standards.

  5. Lorenzostisp:

    Commonly Used Drugs Charts.
    cytotec without rx
    A beacon of reliability and trust.

  6. RobertHal:

    Always responsive, regardless of time zones.
    can i order lisinopril for sale
    A reliable pharmacy that connects patients globally.

  7. Wow! Thank you! I always wanted to write on my website something like that. Can I implement a portion of your post to my blog?