Xiaohongshu User Post avatar

Xiaohongshu User Post

Try for free

No credit card required

Go to Store
Xiaohongshu User Post

Xiaohongshu User Post

share_service/xiaohongshu-user-post
Try for free

No credit card required

获取小红书用户最近发布的帖子

.dockerignore

1# configurations
2.idea
3
4# crawlee and apify storage folders
5apify_storage
6crawlee_storage
7storage
8
9# installed files
10node_modules
11
12# git folder
13.git
14
15# dist folder
16dist

.gitignore

1storage
2apify_storage
3crawlee_storage
4node_modules
5dist
6tsconfig.tsbuildinfo
7storage/*
8!storage/key_value_stores
9storage/key_value_stores/*
10!storage/key_value_stores/default
11storage/key_value_stores/default/*
12!storage/key_value_stores/default/INPUT.json
13
14# Added by Apify CLI
15.venv

bun.lockb

Download

package.json

1{
2	"name": "xiaohongshu-user-post",
3	"version": "0.0.1",
4	"type": "module",
5	"description": "This is an example of an Apify actor.",
6	"engines": {
7		"node": ">=18.0.0"
8	},
9	"dependencies": {
10		"apify": "^3.1.10",
11		"axios": "^1.7.7",
12		"cheerio": "^1.0.0-rc.12",
13		"crypto": "^1.0.1"
14	},
15	"devDependencies": {
16		"@apify/tsconfig": "^0.1.0",
17		"tsx": "^4.6.2",
18		"typescript": "^5.3.3"
19	},
20	"scripts": {
21		"start": "npm run start:dev",
22		"start:prod": "node dist/main.js",
23		"start:dev": "tsx src/main.ts",
24		"build": "tsc",
25		"test": "echo \"Error: oops, the actor has no tests yet, sad!\" && exit 1"
26	},
27	"author": "It's not you it's me",
28	"license": "ISC"
29}

tsconfig.json

1{
2    "extends": "@apify/tsconfig",
3    "compilerOptions": {
4        "module": "NodeNext",
5        "moduleResolution": "NodeNext",
6        "target": "ES2022",
7        "outDir": "dist",
8        "noUnusedLocals": false,
9        "skipLibCheck": true,
10        "lib": ["DOM"]
11    },
12    "include": [
13        "./src/**/*"
14    ]
15}

.actor/Dockerfile

1# Specify the base Docker image. You can read more about
2# the available images at https://docs.apify.com/sdk/js/docs/guides/docker-images
3# You can also use any other image from Docker Hub.
4FROM apify/actor-node:20 AS builder
5
6# Check preinstalled packages
7RUN npm ls crawlee apify puppeteer playwright
8
9# Copy just package.json and package-lock.json
10# to speed up the build using Docker layer cache.
11COPY package*.json ./
12
13# Install all dependencies. Don't audit to speed up the installation.
14RUN npm install --include=dev --audit=false
15
16# Next, copy the source files using the user set
17# in the base image.
18COPY . ./
19
20# Install all dependencies and build the project.
21# Don't audit to speed up the installation.
22RUN npm run build
23
24# Create final image
25FROM apify/actor-node:20
26
27# Check preinstalled packages
28RUN npm ls crawlee apify puppeteer playwright
29
30# Copy just package.json and package-lock.json
31# to speed up the build using Docker layer cache.
32COPY package*.json ./
33
34# Install NPM packages, skip optional and development dependencies to
35# keep the image small. Avoid logging too much and print the dependency
36# tree for debugging
37RUN npm --quiet set progress=false \
38    && npm install --omit=dev --omit=optional \
39    && echo "Installed NPM packages:" \
40    && (npm list --omit=dev --all || true) \
41    && echo "Node.js version:" \
42    && node --version \
43    && echo "NPM version:" \
44    && npm --version \
45    && rm -r ~/.npm
46
47# Copy built JS files from builder image
48COPY --from=builder /usr/src/app/dist ./dist
49
50# Next, copy the remaining files and directories with the source code.
51# Since we do this after NPM install, quick build will be really fast
52# for most source file changes.
53COPY . ./
54
55
56# Run the image.
57CMD npm run start:prod --silent

.actor/actor.json

1{
2	"actorSpecification": 1,
3	"name": "xiaohongshu-user-post",
4	"title": "Scrape single page in TypeScript",
5	"description": "Scrape data from single page with provided URL.",
6	"version": "0.0",
7	"meta": {
8		"templateId": "ts-start"
9	},
10	"input": "./input_schema.json",
11	"dockerfile": "./Dockerfile"
12}

.actor/input_schema.json

1{
2    "title": "Scrape data from a web page",
3    "type": "object",
4    "schemaVersion": 1,
5    "properties": {
6        "url": {
7        "title": "小红书用户的个人主页",
8        "type": "string",
9        "description": "小红书用户的个人主页地址",
10        "editor": "textfield",
11        "prefill": "https://www.xiaohongshu.com/user/profile/669bdb9a000000002401ca91"
12        }
13    },
14    "required": ["url"]
15}

src/frontend.ts

1// @ts-nocheck
2import axios, { type AxiosInstance } from "axios";
3import crypto from "crypto";
4import fs from "fs";
5import { get_search_id, xc_sign } from "./sign";
6
7function md5(str: string) {
8  return crypto.createHash("md5").update(str).digest("hex");
9}
10
11export const AES_KEY = "glt6h61ta7kisow7";
12export const AES_IV = "4hrivgw5s342f9b2";
13function desV2Encrypt(str: string, key = AES_KEY, iv = AES_IV) {
14  const data = Buffer.from(str, "utf8");
15  const keyBuffer = Buffer.from(key, "utf8");
16  const ivBuffer = Buffer.from(iv, "utf8");
17
18  const cipher = crypto.createCipheriv("aes-128-cbc", keyBuffer, ivBuffer);
19  let encrypted = cipher.update(data);
20  encrypted = Buffer.concat([encrypted, cipher.final()]);
21
22  return encrypted.toString("hex");
23}
24
25function cookieMap(cookie: string) {
26  return cookie.split(";").reduce((acc, cur) => {
27    const [key, value] = cur.split("=");
28    return { ...acc, [key.trim()]: value.trim() };
29  }, {});
30}
31
32export default class XHU {
33  #axios: AxiosInstance;
34  private readonly x3: string;
35  private readonly cookie: string;
36
37  constructor(cookie: string, needLogin: boolean = true) {
38    const c = cookieMap(cookie);
39    this.x3 = c["a1"] as string;
40    if (!this.x3) {
41      throw new Error("a1 not found");
42    }
43
44    this.#axios = axios.create({
45      baseURL: "https://edith.xiaohongshu.com",
46      headers: {
47        accept: "application/json, text/plain, */*",
48        "accept-language": "en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7",
49        authorization: "edith.xiaohongshu.com",
50        "cache-control": "no-cache",
51        cookie: cookie,
52        pragma: "no-cache",
53        Origin: "https://www.xiaohongshu.com",
54        Referer: "https://www.xiaohongshu.com",
55        "sec-ch-ua": '"Google Chrome";v="119", "Chromium";v="119", "Not?A_Brand";v="24"',
56        "sec-ch-ua-mobile": "?0",
57        "sec-ch-ua-platform": '"macOS"',
58        "sec-fetch-dest": "empty",
59        "sec-fetch-mode": "cors",
60        "sec-fetch-site": "same-origin",
61        "user-agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36",
62      },
63    });
64    this.#axios.interceptors.request.use((config) => {
65      const headers = this._preHeaders(config.url!, config.params, config.data);
66      console.log(headers);
67      config.headers = { ...config.headers, ...headers } as any;
68
69      return config;
70    });
71
72    this.#axios.interceptors.response.use((response) => {
73      return response.data;
74    });
75  }
76
77  _preHeaders(url: string, params: any, data: any) {
78    const queryString = new URLSearchParams(params).toString();
79    const fullPath = queryString ? `${url}?${queryString}` : url;
80    let x1 = md5(`url=${fullPath}`);
81    if (data) {
82      x1 = md5(`url=${fullPath}${JSON.stringify(data)}`);
83    }
84    const x2 = "0|0|0|1|0|0|1|0|0|0|1|0|0|0|0|1|0|0|0";
85    const x3 = this.x3;
86    const x4 = Date.now();
87    const mulStr = `x1=${x1};x2=${x2};x3=${x3};x4=${x4};`;
88
89    const xs_hash = desV2Encrypt(Buffer.from(mulStr).toString("base64"));
90    const payload = { signSvn: "55", signType: "x2", appId: "ugc", signVersion: "1", payload: xs_hash };
91    const xs = `XYW_${Buffer.from(JSON.stringify(payload)).toString("base64")}`;
92    const xt = x4;
93
94    const signs = xc_sign(x3, xs, xt + "");
95    const headers = {
96      "X-S": signs["x-s"],
97      "X-T": signs["x-t"],
98      "X-S-Common": signs["x-s-common"],
99      "X-B3-Traceid": signs["x-b3-traceid"],
100    };
101    return headers;
102  }
103
104  async permit({ file_count = 1, scene = "image" }: { file_count: number; scene: string }) {
105    const response = await this.#axios.get("/api/media/v1/upload/creator/permit", {
106      params: {
107        biz_name: "spectrum",
108        scene,
109        file_count,
110        version: "1",
111        source: "web",
112      },
113    });
114
115    console.log(response.data);
116    const tempPermit = response.data.uploadTempPermits[0];
117    const fileId = tempPermit.fileIds[0];
118    const token = tempPermit.token;
119    return [fileId, token];
120  }
121
122  async createNote(
123    title: string,
124    desc: string,
125    noteType: string,
126    ats: any[] = [],
127    topics: any[] = [],
128    imageInfo: object | null = null,
129    videoInfo: object | null = null,
130    postTime: string | null = null,
131    isPrivate: boolean = true
132  ): Promise<any> {
133    if (postTime) {
134      const postDateTime = new Date(postTime);
135      postTime = postDateTime.getTime().toString();
136    }
137    const uri = "/web_api/sns/v2/note";
138    const businessBinds = {
139      version: 1,
140      noteId: 0,
141      noteOrderBind: {},
142      notePostTiming: { postTime: postTime },
143      noteCollectionBind: { id: "" },
144    };
145
146    if (topics.length > 0) {
147      //  #理想L[话题]#
148      desc = desc + "\n" + topics.map((topic) => `#${topic.name}[话题]#`).join(" ");
149    }
150
151    const data = {
152      common: {
153        type: noteType,
154        title: title,
155        note_id: "",
156        desc: desc,
157        source: '{"type":"web","ids":"","extraInfo":"{\\"subType\\":\\"official\\"}"}',
158        business_binds: JSON.stringify(businessBinds),
159        ats: ats,
160        hash_tag: topics,
161        post_loc: {},
162        privacy_info: { op_type: 1, type: Number(isPrivate) },
163      },
164      image_info: imageInfo,
165      video_info: videoInfo,
166    };
167    const headers = { Referer: "https://creator.xiaohongshu.com/" };
168    return this.#axios.post(uri, data, { headers, baseURL: "https://edith.xiaohongshu.com" });
169  }
170
171  async getFileStream(filePath: string) {
172    if (filePath.startsWith("http")) {
173      const response = await axios.get(filePath, {
174        responseType: "arraybuffer",
175      });
176      return response.data;
177    } else {
178      return fs.createReadStream(filePath);
179    }
180  }
181
182  public async uploadFile(fileId: string, token: string, filePath: string, contentType: string = "image/jpeg"): Promise<any> {
183    const maxFileSize = 5 * 1024 * 1024;
184    const url = `https://ros-upload.xiaohongshu.com/${fileId}`;
185    if (fs.statSync(filePath).size > maxFileSize && contentType === "video/mp4") {
186      throw new Error("video too large, < 5M");
187    } else {
188      const headers = { "X-Cos-Security-Token": token, "Content-Type": contentType };
189
190      const fileStream = fs.createReadStream(filePath);
191      return axios.put(url, fileStream, { headers });
192    }
193  }
194
195  async createImageNote(title: string, desc: string, files: string[], postTime: string | null = null, ats: any[] = [], topics: any[] = [], isPrivate: boolean = true): Promise<any> {
196    const images = [];
197    for (const file of files) {
198      const [imageId, token] = await this.permit({ file_count: 1, scene: "image" });
199      await this.uploadFile(imageId, token, file);
200      images.push({
201        file_id: imageId,
202        metadata: { source: -1 },
203        stickers: { version: 2, floating: [] },
204        extra_info_json: '{"mimeType":"image/jpeg"}',
205      });
206    }
207    return this.createNote(title, desc, "normal", ats, topics, { images: images }, null, postTime, isPrivate);
208  }
209
210  async myInfo() {
211    return this.#axios.get("/api/galaxy/user/my-info");
212  }
213
214  async posted(page = 1) {
215    return this.#axios.get(`/api/galaxy/creator/note/user/posted?tab=0&page=${page}`);
216  }
217
218  async delete(note_id: string) {
219    return this.#axios.post(
220      "/web_api/sns/capa/postgw/note/delete",
221      {
222        note_id: note_id,
223      },
224      {
225        baseURL: "https://edith.xiaohongshu.com",
226      }
227    );
228  }
229
230  public async getNoteByKeyword(keyword: string, page: number = 1, pageSize: number = 20, sort = "general", noteType = 0) {
231    const uri = "/api/sns/web/v1/search/notes";
232    const data = {
233      keyword,
234      image_formats: ["webp", "jpg", "avif"],
235      page,
236      page_size: pageSize,
237      search_id: get_search_id(),
238      sort: sort,
239      note_type: noteType,
240    };
241    return this.#axios.post(uri, data);
242  }
243
244  async homefeed() {
245    const params = {"cursor_score":"","num":35,"refresh_type":1,"note_index":26,"unread_begin_note_id":"","unread_end_note_id":"","unread_note_count":0,"category":"homefeed.movie_and_tv_v3","search_key":"","need_num":10,"image_formats":["jpg","webp","avif"],"need_filter_image":false};
246    return this.#axios.post("/api/sns/web/v1/homefeed", params);
247  }
248  async feed(source_note_id: string, xsec_token: string) {
249    const params = {
250      source_note_id: source_note_id,
251      image_formats: ["jpg", "webp", "avif"],
252      extra: {
253        need_body_topic: "1",
254      },
255      xsec_source: "pc_feed",
256      xsec_token: xsec_token,
257    };
258    console.log(params);
259
260    return this.#axios.post("/api/sns/web/v1/feed", params);
261  }
262}
263
264const cookie = `abRequestId=19f1d061-93f8-5610-9510-e463f266e2da; webBuild=4.33.2; xsecappid=xhs-pc-web; a1=191c80307b89o84mfkaytvynytnwa47jdgqic27pv30000468253; webId=09913c973e88db4de8b93500c05ba961; acw_tc=67d4dbda18f981864438a09b59df99ffbd781765f73179f2492e8eca5202439d; web_session=030037a1e877ec145768d465a3214a636a4e6f; gid=yjySY8q8iqfWyjySY8q8WkJWDYjlY46iT0v9AUhhvIv9I3q8E04WVS8884KYJ2q8ifKdYYiK`;
265const r = new XHU(cookie, false);
266r.homefeed().then(async (data) => {
267  const item = data.data.items[0]
268  
269  await new Promise((resolve) => setTimeout(resolve, 1000));
270  r.feed(item.id, item.xsec_token).then((detail) => {
271    console.log(JSON.stringify(detail.data, null, 2));
272  })
273});

src/main.ts

1// @ts-nocheck
2// Axios - Promise based HTTP client for the browser and node.js (Read more at https://axios-http.com/docs/intro).
3import axios from "axios";
4// Cheerio - The fast, flexible & elegant library for parsing and manipulating HTML and XML (Read more at https://cheerio.js.org/).
5import * as cheerio from "cheerio";
6// Apify SDK - toolkit for building Apify Actors (Read more at https://docs.apify.com/sdk/js/).
7import { Actor } from "apify";
8import XHU from "./frontend.js";
9
10// this is ESM project, and as such, it requires you to specify extensions in your relative imports
11// read more about this here: https://nodejs.org/docs/latest-v18.x/api/esm.html#mandatory-file-extensions
12// note that we need to use `.js` even when inside TS files
13// import { router } from './routes.js';
14
15// The init() call configures the Actor for its environment. It's recommended to start every Actor with an init().
16await Actor.init();
17
18interface Input {
19  url: string;
20}
21// Structure of input is defined in input_schema.json
22const input = await Actor.getInput<Input>();
23if (!input) throw new Error("Input is missing!");
24const { url } = input;
25
26// const cookie = `abRequestId=19f1d061-93f8-5610-9510-e463f266e2da; webBuild=4.33.2; xsecappid=xhs-pc-web; a1=191c80307b89o84mfkaytvynytnwa47jdgqic27pv30000468253; webId=09913c973e88db4de8b93500c05ba961; acw_tc=67d4dbda18f981864438a09b59df99ffbd781765f73179f2492e8eca5202439d; web_session=030037a1e877ec145768d465a3214a636a4e6f; gid=yjySY8q8iqfWyjySY8q8WkJWDYjlY46iT0v9AUhhvIv9I3q8E04WVS8884KYJ2q8ifKdYYiK`;
27// const r = new XHU(cookie, false);
28
29const response = await axios.get(
30url,
31  {
32    headers: {
33      accept: "application/json, text/plain, */*",
34      "accept-language": "en-US,en;q=0.9",
35      "cache-control": "no-cache",
36      "content-type": "application/json;charset=UTF-8",
37      pragma: "no-cache",
38      "sec-ch-ua":
39        '"Google Chrome";v="119", "Chromium";v="119", "Not?A_Brand";v="24"',
40      "sec-ch-ua-mobile": "?0",
41      "sec-ch-ua-platform": '"macOS"',
42      "sec-fetch-dest": "empty",
43      "sec-fetch-mode": "cors",
44      "sec-fetch-site": "same-site",
45      "user-agent":
46        "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36",
47      Cookie:
48        "abRequestId=fb204935-3ff9-5f82-8f18-6fb74c86a42e; acw_tc=594d061c3e470e8551ac518b11738a7b843bcc735d0fdcfc674962de512d65de",
49    },
50  }
51);
52
53console.log(response.data);
54const $ = cheerio.load(response.data);
55const items = $("#userPostedFeeds .note-item").toArray();
56const list = [];
57items.forEach((item) => {
58  const title = $(item).find(".title").text();
59  const image = $(item).find("img").attr("src");
60  const likedCount = $(item).find(".count").text().replace("赞", "0");
61  list.push({ title, image, likedCount });
62});
63
64await Actor.pushData(list);
65
66// Gracefully exit the Actor process. It's recommended to quit all Actors with an exit().
67await Actor.exit();

src/sign.ts

1// @ts-nocheck
2export function xc_sign(a1 = "", x_s = "", x_t = "") {
3  const b1 = "I38rHdgsjopgIvesdVwgIC+oIELmBZ5e3VwXLgFTIxS3bqwErFeexd0ekncAzMFYnqthIhJeSfMDKutRI3KsYorWHPtGrbV0P9WfIi/eWc6eYqtyQApPI37ekmR1QL+5Ii6sdnosjoT5yqtXqqwYrBqoIx++GDi/sVtkIx0sxuwr4qtiIkrwIi/skcc3ICLfI3Oe0utl20DZsL5eDSJejVw0IieexVwL+PtorqthPWKexY8oICR1IErSgVwBGqtRIxE/eDdeVuwjIC0s1qtnIkpKIkRLee3eDoq6cU5sYqtzaDdefzq1zd/eWWF+IxoefutLIEeeDqt7rYrlOoVGIvgeiqtu/YgexjqmIkLwIiDPGamjIvhm+I88IizuBVwlIvGF4eveDS7e1utCIC7sDc==";
4  const common = {
5      s0: 5,
6      s1: "",
7      x0: "1",
8      x1: "3.3.0",
9      x2: "Windows",
10      x3: "xhs-pc-web",
11      x4: "1.4.4",
12      x5: a1,
13      x6: x_t,
14      x7: x_s,
15      x8: b1,
16      x9: mrc(x_t + x_s + b1),
17      x10: 1,
18  };
19  const encodeStr = encodeUtf8(JSON.stringify(common));
20  const x_s_common = b64Encode(encodeStr);
21  const x_b3_traceid = get_b3_trace_id();
22  return {
23      "x-s": x_s,
24      "x-t": x_t,
25      "x-s-common": x_s_common,
26      "x-b3-traceid": x_b3_traceid,
27  };
28}
29
30function get_b3_trace_id() {
31  const chars = "abcdef0123456789";
32  let result = "";
33  for (let i = 0; i < 16; i++) {
34      result += chars.charAt(Math.floor(Math.random() * chars.length));
35  }
36  return result;
37}
38
39function mrc(e) {
40  const ie = [
41      0, 1996959894, 3993919788, 2567524794, 124634137, 1886057615, 3915621685,
42      2657392035, 249268274, 2044508324, 3772115230, 2547177864, 162941995,
43      2125561021, 3887607047, 2428444049, 498536548, 1789927666, 4089016648,
44      2227061214, 450548861, 1843258603, 4107580753, 2211677639, 325883990,
45      1684777152, 4251122042, 2321926636, 335633487, 1661365465, 4195302755,
46      2366115317, 997073096, 1281953886, 3579855332, 2724688242, 1006888145,
47      1258607687, 3524101629, 2768942443, 901097722, 1119000684, 3686517206,
48      2898065728, 853044451, 1172266101, 3705015759, 2882616665, 651767980,
49      1373503546, 3369554304, 3218104598, 565507253, 1454621731, 3485111705,
50      3099436303, 671266974, 1594198024, 3322730930, 2970347812, 795835527,
51      1483230225, 3244367275, 3060149565, 1994146192, 31158534, 2563907772,
52      4023717930, 1907459465, 112637215, 2680153253, 3904427059, 2013776290,
53      251722036, 2517215374, 3775830040, 2137656763, 141376813, 2439277719,
54      3865271297, 1802195444, 476864866, 2238001368, 4066508878, 1812370925,
55      453092731, 2181625025, 4111451223, 1706088902, 314042704, 2344532202,
56      4240017532, 1658658271, 366619977, 2362670323, 4224994405, 1303535960,
57      984961486, 2747007092, 3569037538, 1256170817, 1037604311, 2765210733,
58      3554079995, 1131014506, 879679996, 2909243462, 3663771856, 1141124467,
59      855842277, 2852801631, 3708648649, 1342533948, 654459306, 3188396048,
60      3373015174, 1466479909, 544179635, 3110523913, 3462522015, 1591671054,
61      702138776, 2966460450, 3352799412, 1504918807, 783551873, 3082640443,
62      3233442989, 3988292384, 2596254646, 62317068, 1957810842, 3939845945,
63      2647816111, 81470997, 1943803523, 3814918930, 2489596804, 225274430,
64      2053790376, 3826175755, 2466906013, 167816743, 2097651377, 4027552580,
65      2265490386, 503444072, 1762050814, 4150417245, 2154129355, 426522225,
66      1852507879, 4275313526, 2312317920, 282753626, 1742555852, 4189708143,
67      2394877945, 397917763, 1622183637, 3604390888, 2714866558, 953729732,
68      1340076626, 3518719985, 2797360999, 1068828381, 1219638859, 3624741850,
69      2936675148, 906185462, 1090812512, 3747672003, 2825379669, 829329135,
70      1181335161, 3412177804, 3160834842, 628085408, 1382605366, 3423369109,
71      3138078467, 570562233, 1426400815, 3317316542, 2998733608, 733239954,
72      1555261956, 3268935591, 3050360625, 752459403, 1541320221, 2607071920,
73      3965973030, 1969922972, 40735498, 2617837225, 3943577151, 1913087877,
74      83908371, 2512341634, 3803740692, 2075208622, 213261112, 2463272603,
75      3855990285, 2094854071, 198958881, 2262029012, 4057260610, 1759359992,
76      534414190, 2176718541, 4139329115, 1873836001, 414664567, 2282248934,
77      4279200368, 1711684554, 285281116, 2405801727, 4167216745, 1634467795,
78      376229701, 2685067896, 3608007406, 1308918612, 956543938, 2808555105,
79      3495958263, 1231636301, 1047427035, 2932959818, 3654703836, 1088359270,
80      936918000, 2847714899, 3736837829, 1202900863, 817233897, 3183342108,
81      3401237130, 1404277552, 615818150, 3134207493, 3453421203, 1423857449,
82      601450431, 3009837614, 3294710456, 1567103746, 711928724, 3020668471,
83      3272380065, 1510334235, 755167117,
84  ];
85  let o = -1;
86
87  function rightWithoutSign(num, bit = 0) {
88      return (num >>> bit);
89  }
90
91  for (let n = 0; n < 57; n++) {
92      o = ie[(o & 255) ^ e.charCodeAt(n)] ^ rightWithoutSign(o, 8);
93  }
94  return o ^ -1 ^ 3988292384;
95}
96
97const lookup = [
98  "Z", "m", "s", "e", "r", "b", "B", "o", "H", "Q", "t", "N", "P", "+", "w", "O",
99  "c", "z", "a", "/", "L", "p", "n", "g", "G", "8", "y", "J", "q", "4", "2", "K",
100  "W", "Y", "j", "0", "D", "S", "f", "d", "i", "k", "x", "3", "V", "T", "1", "6",
101  "I", "l", "U", "A", "F", "M", "9", "7", "h", "E", "C", "v", "u", "R", "X", "5"
102];
103
104function tripletToBase64(e) {
105  return lookup[(63 & (e >> 18))] +
106      lookup[(63 & (e >> 12))] +
107      lookup[((e >> 6) & 63)] +
108      lookup[(e & 63)];
109}
110
111function encodeChunk(e, t, r) {
112  let m = [];
113  for (let b = t; b < r; b += 3) {
114      const n = ((16711680 & (e[b] << 16)) +
115          ((e[b + 1] << 8) & 65280) +
116          (e[b + 2] & 255));
117      m.push(tripletToBase64(n));
118  }
119  return m.join('');
120}
121
122function b64Encode(e) {
123  const P = e.length;
124  const W = P % 3;
125  let U = [];
126  const z = 16383;
127  let H = 0;
128  const Z = P - W;
129  while (H < Z) {
130      U.push(encodeChunk(e, H, H + z > Z ? Z : H + z));
131      H += z;
132  }
133  if (W === 1) {
134      const F = e[P - 1];
135      U.push(lookup[F >> 2] + lookup[(F << 4) & 63] + "==");
136  } else if (W === 2) {
137      const F = (e[P - 2] << 8) + e[P - 1];
138      U.push(lookup[F >> 10] + lookup[(F >> 4) & 63] + lookup[(F << 2) & 63] + "=");
139  }
140  return U.join('');
141}
142
143function encodeUtf8(e) {
144  let b = [];
145  const m = encodeURIComponent(e);
146  for (let i = 0; i < m.length; i++) {
147      if (m[i] === "%") {
148          const hex = m.substr(i + 1, 2);
149          b.push(parseInt(hex, 16));
150          i += 2;
151      } else {
152          b.push(m.charCodeAt(i));
153      }
154  }
155  return b;
156}
157
158function base36encode(number, alphabet = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ') {
159  if (typeof number !== 'number' || number % 1 !== 0) {
160      throw new TypeError('number must be an integer');
161  }
162
163  let base36 = '';
164  let sign = '';
165
166  if (number < 0) {
167      sign = '-';
168      number = -number;
169  }
170
171  if (0 <= number && number < alphabet.length) {
172      return sign + alphabet[number];
173  }
174
175  while (number !== 0) {
176      const i = number % alphabet.length;
177      number = Math.floor(number / alphabet.length);
178      base36 = alphabet[i] + base36;
179  }
180
181  return sign + base36;
182}
183
184function base36decode(number) {
185  return parseInt(number, 36);
186}
187
188export function get_search_id() {
189  const e = BigInt(Date.now()) << 64n;
190  const t = BigInt(Math.floor(Math.random() * 2147483646));
191  return base36encode(Number(e + t));
192}
Developer
Maintained by Community

Actor Metrics

  • 11 monthly users

  • 3 stars

  • >99% runs succeeded

  • Created in Sep 2024

  • Modified 4 months ago