This commit is contained in:
SeungJu Lim 2024-02-12 16:47:20 +09:00
commit 507a19b7cb
6 changed files with 4440 additions and 0 deletions

6
.gitignore vendored Normal file
View File

@ -0,0 +1,6 @@
.env
node_modules
Old_
research
result
*.png

525
cgvAjax.js Normal file
View File

@ -0,0 +1,525 @@
import * as fs from 'fs';
import axios from 'axios';
import dotenv from 'dotenv';
import moment from 'moment';
import puppeteer from 'puppeteer';
import TelegramBot from 'node-telegram-bot-api';
import _ from 'lodash';
import * as util from './commonUtil.js';
import * as crypto from './cryptoUtil.js';
dotenv.config();
const COOKIE = process.env.COOKIE;
const BOT_TOKEN = process.env.BOT_TOKEN;
const BOT_CHATID = process.env.BOT_CHATID;
const bot = new TelegramBot(BOT_TOKEN, { polling: true });
const __INFO__ = async (str) => {
const prefix = '[INFO]'
await bot.sendMessage(BOT_CHATID, `${prefix} ${str}`);
console.log(`${prefix} ${str}`);
}
const __ERROR__ = async (str) => {
const prefix = '[ERROR]'
await bot.sendMessage(BOT_CHATID, `${prefix} ${str}`);
console.error(`${prefix} ${str}`);
}
class cgvGetter {
constructor(cookie) {
this.seatAspx = 'https://m.cgv.co.kr/WebApp/Reservation/seat.aspx?';
this.loginAspx = 'https://m.cgv.co.kr/Webapp/Member/Login.aspx?';
}
// 에러 발생 시 null 반환
async _post(postUrl, payload, headers) {
return axios.post(postUrl, payload, { headers, timeout: 3000 })
.then(response => {
return { cookies: response.headers['set-cookie'], data: response.data };
})
.catch(error => {
__ERROR__('\t> Post request failed:', error.message);
return null;
});
}
async _get(getUrl, headers) {
return axios.get(getUrl, { headers, timeout: 3000 })
.then(response => {
return { cookies: response.headers['set-cookie'], data: response.data };
})
.catch(error => {
__ERROR__('\t> Get request failed:', error.message);
return null;
});
}
// 가중치 계산식
calculateSeatWeight(seatPosition, centerPosition) {
// 중앙점 설정
const centerX = centerPosition.x;
const centerY = centerPosition.y.charCodeAt(0) - 'A'.charCodeAt(0) + 1;
// 좌석 위치 설정
const seatX = seatPosition.x;
const seatY = seatPosition.y.charCodeAt(0) - 'A'.charCodeAt(0) + 1;
// 중앙으로부터의 수평/수직 거리 계산
const horizontalDistance = Math.abs(seatX - centerX);
const verticalDistance = Math.abs(seatY - centerY);
// 비선형 거리 가중치
const distanceWeight = Math.pow(horizontalDistance + verticalDistance, 2);
// 앞쪽 행 선호도를 직접 반영
const rowPreference = (centerY > seatY) ? (centerY - seatY) * 0.00001 : 0;
// 총 가중치 계산 - 거리 가중치와 앞쪽 행 선호도를 합산
const totalWeight = distanceWeight - rowPreference;
return totalWeight;
}
// calculateSeatWeight(seat, centerSeat) {
// const horizontalDistance = Math.abs(seat.x - centerSeat.x);
// const verticalDistance = Math.abs(seat.y.charCodeAt(0) - centerSeat.y.charCodeAt(0));
// let positionWeight;
// if (horizontalDistance === 0 && verticalDistance === 0) {
// positionWeight = 1.0;
// } else if (horizontalDistance === 1 && verticalDistance === 0) {
// positionWeight = 0.9;
// } else if (horizontalDistance === 0 && verticalDistance === 1) {
// positionWeight = seat.y === centerSeat.y ? 0.85 : 0.8;
// } else {
// positionWeight = 1.0 - (horizontalDistance + verticalDistance) / 10.0;
// }
// const seatPreference = horizontalDistance + verticalDistance - positionWeight;
// return seatPreference + 0.1;
// }
// 인접한 예매 가능 좌석 확인 - 넘겨받은 좌석 수 포함한 값 반환
countAdjacentSeats(seats, seatname) {
const seatInfo = seats.find(seat => seat.seatname === seatname);
if (!seatInfo) return -1;
const startX = Math.min(...seats.map(seat => seat.locxnm));
const endX = Math.max(...seats.map(seat => seat.locxnm));
const row = seatInfo.locynm;
const col = seatInfo.locxnm;
let leftAdjacentCount = 0;
let rightAdjacentCount = 0;
// 왼쪽
for (let i = col - 1; i >= startX; i--) {
const adjacentSeat = seats.find(seat => seat.locynm === row && seat.locxnm === i);
if (adjacentSeat) leftAdjacentCount++;
else break; // 연속된 좌석이 아닌 경우 반복문을 종료합니다.
}
// 오른쪽
for (let i = col + 1; i <= endX; i++) {
const adjacentSeat = seats.find(seat => seat.locynm === row && seat.locxnm === i);
if (adjacentSeat) rightAdjacentCount++;
else break; // 연속된 좌석이 아닌 경우 반복문을 종료합니다.
}
return leftAdjacentCount + rightAdjacentCount + 1;
}
async getResultData(mgCD = null, ymd = null, td = null, fmac = null, fsrc = null) {
const postUrl = "https://m.cgv.co.kr/WebAPP/Reservation/Common/ajaxTheaterScheduleList.aspx/GetTheaterScheduleList";
const param = {
strRequestType: "COMPARE", // 영화별예매 : MOVIE, 극장별예매/상영시간표 : THEATER, 비교예매 : COMPARE
strUserID: '',
strPlayYMD: ymd ? ymd : '', // 비어있을 시 가장 빠른 날짜
strMovieGroupCd: mgCD ? mgCD : '20035290', // 영화 그룹코드
strTheaterCd: td ? td : '0013',
strMovieTypeCd: fmac ? fmac : '04', // 영화 속성 IMAX
strScreenTypeCd: '',
strRankType: 'MOVIE'
};
const headers = {
'User-Agent': 'Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Mobile Safari/537.36',
'Accept-Language': 'ko,en-US;q=0.9,en;q=0.8,ko-KR;q=0.7,de;q=0.6',
'Content-Type': 'application/json',
Origin: 'https://m.cgv.co.kr',
Cookie: COOKIE
};
const result = await this._post(postUrl, param, headers);
return result
}
async main_dev(mgCD = '20035290', ymd = null, dateAft = moment('2024-03-09', 'YYYY-MM-DD')) {
const generalSeatCnt = 2;
// const startTime = moment('2024-02-05');
// const endTIme = moment('2024-02-12');
const tmRange = { start: '1100', end: '1800' };
let targetYMD = null;
let targetSchedule = null;
let attemptCnt = 0;
const init = moment();
//==================================================
// requset - 날짜 탐색
//==================================================
while (1) {
// 스케줄 데이터 요청
const { cookies, data: result } = await this.getResultData(mgCD, targetYMD); // 20035290 듄, 20035256 웡카
// 예외 - 요청 성공 확인
if (result === null) throw new Error(`\t> Request failed`);
// 예외(실험적) - 응답 데이터 구조 확인
if (!result.hasOwnProperty('d')) throw new Error(`\t> Response lacks the expected "d" property`);
const gScheduleList = JSON.parse(result.d);
// 예외 - CGV 응답 코드 확인
if (gScheduleList.ResultCode !== "00000" || gScheduleList.ResultMessage !== "성공") throw new Error(`\t> Invalid schedule response`);
// 예외 - 영화 스케줄 존재 여부 확인
if (gScheduleList.ResultSchedule.ListPlayYmd.length === 0 && gScheduleList.ResultSchedule.ScheduleList.length === 0) {
throw new Error(`\t> Schedule not found`);
}
// 날짜 추출 - YMD 문자열 moment로 변환
const listPlayYmd = gScheduleList.ResultSchedule.ListPlayYmd.split('|').map(YMD => util.YMDConvert(YMD));
// 예매 가능한 일정만 필터
const filteredDates = util.dayFilter(listPlayYmd);
const targetDates = filteredDates.filter(date => date.isSameOrAfter(dateAft));
//==================================================
// requset - 날짜 찾았을 때
//==================================================
if (targetYMD === null && targetDates.length !== 0) {
targetYMD = util.YMDConvert(targetDates[0]);
await __INFO__(`조건 부합 날짜 식별됨 - 시도: ${++attemptCnt} / 시간: ${moment().format('MM-DD HH:mm:ss')} / 경과: ${init.fromNow()}`);
continue
//==================================================
// requset - 시간표 찾았을 때
//==================================================
} else if (targetYMD !== null && targetSchedule === null) {
/* ++++++++++++ 남은 좌석 수 필터링 추가 'SeatRemainCnt' */
// 상영시간표 추출
const scheduleList = gScheduleList.ResultSchedule.ScheduleList;
const seatFilteredSchedule = scheduleList.filter(schedule => schedule.SeatRemainCnt >= 550);
const tmFilteredSchedule = seatFilteredSchedule.filter(schedule => {
const scheduleTm = moment(schedule.PlayStartTm, 'HHmm');
const t_start = moment(tmRange.start, 'HHmm');
const t_end = moment(tmRange.end, 'HHmm');
return scheduleTm.isBetween(t_start, t_end, null, '[]');
})
if (tmFilteredSchedule.length !== 0) {
targetSchedule = tmFilteredSchedule[0];
await __INFO__(`조건 부합 스케줄 식별됨 - 시도: ${++attemptCnt} / 시간: ${moment().format('MM-DD HH:mm:ss')} / 경과: ${init.fromNow()}`);
break
}
}
console.log(`탐색 중 - 시도: ${++attemptCnt} / 시간: ${moment().format('MM-DD HH:mm:ss')} / 경과: ${init.fromNow()}`);
console.log(`\t 조건 - 날짜: ${targetYMD ? targetYMD : '빠른순'}`);
await new Promise((resolve) => setTimeout(resolve, 60000));
}
await __INFO__(`예매 시작 / 시간: ${moment().format('MM.DD HH:mm:ss')}`);
await __INFO__(`\t 조건 - 날짜: ${targetSchedule.PlayYmd} / 시간: ${targetSchedule.PlayStartTm}`);
//==================================================
// requset - 예매 시작
//==================================================
// 추출한 영화 정보 -> 페이로드로 재구성
const tcktDate = targetSchedule;
const TicketTypeOrderInfo = [
{ "TicketTypeName": "일반", "TicketTypeCode": "01", "TicketTypeCss": "General", "TicketTypeCount": generalSeatCnt },
{ "TicketTypeName": "청소년", "TicketTypeCode": "02", "TicketTypeCss": "Student", "TicketTypeCount": 0 }
]
const payload = {
"hfTheaterCd": tcktDate.TheaterCd,
"hfTheaterNm": tcktDate.TheaterNm,
"hfCGVCode": tcktDate.MovieCd,
"hfPlayYMD": tcktDate.PlayYmd,
"hfPlayNum": tcktDate.PlayNum,
"hfScreenCd": tcktDate.ScreenCd,
"hfScreenNM": tcktDate.ScreenNm,
"hfPlayTimeCd": tcktDate.PlayTimeCd,
"hfScreenRatingCd": tcktDate.ScreenRatingCd,
"hfRating": tcktDate.MovieRatingCd,
"hfScreenRatingNm": tcktDate.ScreenRatingNm,
"hfStartHHMM": tcktDate.PlayStartTm,
"hfEndHHMM": tcktDate.PlayEndTm,
"hfKidsScreenType": tcktDate.KidsScreenType,
"hfmovieIdx": tcktDate.MovieIdx,
"hfmovieName": tcktDate.MovieNmKor,
"hfPlayEndTM": tcktDate.PlayEndTm,
"hfPlatformNM": tcktDate.PlatformNm,
"hfMovieRatingNM": tcktDate.MovieRatingNm,
"hfPlayTimeNM": tcktDate.PlayTimeNm,
"hfMoviePkgYn": tcktDate.MoviePkgYn,
"hfMovieNoShowYn": tcktDate.MovieNoshowYn,
"hfPlatformCd": tcktDate.PlatformCd,
"hfGeneral_count": generalSeatCnt,
"hfSpecialSeat_count": 0,
"hfStudent_count": 0,
"hfKid_count": 0,
"hfGiveSpecial_count": 0,
"hfSenior_count": 0,
"hfArmy_count": 0,
"hfMovieGroupCd": tcktDate.MovieGroupCd,
"hfMovieAttrCd": tcktDate.MovieAttrCd,
"hfMovieAttrNm": tcktDate.MovieAttrNm,
"hfSeatRemainRate": tcktDate.SeatRate,
"hfTicketTypeOrderInfo": JSON.stringify(TicketTypeOrderInfo)
}
// 페이로드 -> 요청용 쿼리 (암호화)
const encryptedPayload = crypto.encrypt_hfObject(payload);
const seatQuery = new URLSearchParams(encryptedPayload).toString();
const url_seat = this.seatAspx + seatQuery;
//==================================================
// requset - 로그인 쿠키
//==================================================
// 로그인 세션 쿠키 추출
const loginParam = {
"hfUserId": crypto.cgv_encrypt_hf(process.env.CGV_ID),
"hfPasswordInter": encodeURIComponent(crypto.cgv_sha256_encrypt(process.env.CGV_PW)),
"hfPasswordLocal": encodeURIComponent(crypto.cgv_sha256_encrypt(crypto.cgv_md5_encrypt(process.env.CGV_PW))),
"hfReUrl": crypto.cgv_encrypt_hf('https%3a%2f%2fm.cgv.co.kr%2f'),
"hfAgree": crypto.cgv_encrypt_hf('0'),
"nonmemberStateCd": crypto.cgv_encrypt_hf('0')
};
const loginHeaders = {
'User-Agent': 'Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Mobile Safari/537.36',
'Accept-Language': 'ko,en-US;q=0.9,en;q=0.8,ko-KR;q=0.7,de;q=0.6',
'Content-Type': 'application/x-www-form-urlencoded',
Origin: 'https://m.cgv.co.kr',
Cookie: COOKIE
}
const { cookies: loginCookies } = await this._post(this.loginAspx, loginParam, loginHeaders);
/* ++++++++++++ 예외 추가 or 예외 자체를 요청과정에 합병 */
// cookie list -> cookie string
const joinedLoginCookie = util.joinCookie(loginCookies);
const parsedLoginCookie = util.parseCookie(joinedLoginCookie)
const PuppeteerCookie = Object.entries(parsedLoginCookie).map(([key, val]) => ({ 'name': key, 'value': val }));
//==================================================
// puppeteer - 기본 설정
//==================================================
// const browser = await puppeteer.launch({ headless: false });
const browser = await puppeteer.launch({ headless: 'new' });
const page = await browser.newPage();
await page.setViewport({ width: 1920, height: 1080, });
await page.setRequestInterception(true);
page.on('request', (req) => {
if (['font'].includes(req.resourceType())) req.abort();
else req.continue();
});
page.on('popup', async popup => {
await __INFO__(await popup.title());
// await page.screenshot({path: 'popup.png'});
await popup.close();
});
page.on('dialog', async dialog => {
await __INFO__(dialog.message());
// await page.screenshot({path: 'dialog.png'});
await dialog.dismiss(); // 혹은 accept()로 확인 가능
});
//==================================================
// puppeteer - seatAspx
//==================================================
await page.goto('https://m.cgv.co.kr/');
await page.setCookie(...PuppeteerCookie);
let failCnt = 0;
// 반복 시작: 좌석 추출 -> 좌석 선택 -> 결제 시도
while (1) {
try {
await page.goto(url_seat);
await __INFO__(` ===== 좌석창 들어옴`);
// 연령 등급 팝업 제거 - jQeury함수
await page.evaluate(() => { jQuery.fn.closePopup('popAge') });
//==================================================
// puppeteer - 좌석 추출
//==================================================
// 예매 가능한 일반석 시트 정보 추출
const availableSeats = await page.$$eval('#seat_table > tbody > tr > td[reservation="Yes"].pointer[rating_nm="일반석"]', elements => {
return elements.map(element => {
const seatname = element.getAttribute('seatname').trim();
const locynm = element.getAttribute('locynm').trim();
const locxnm = Number(element.getAttribute('locxnm').trim());
return { seatname, locynm, locxnm };
});
});
//==================================================
// puppeteer - 최적 좌석 탐색 알고리즘
//==================================================
// 중심 및 범위 설정
const setCenterSeat = { x: 23, y: 'H' };
const setSeatRange = { startX: 16, endX: 29, startY: 'H', endY: 'K' }
// 탐색 범위 제한
const filteredX = availableSeats.filter(seat => seat.locxnm >= setSeatRange.startX && seat.locxnm <= setSeatRange.endX);
const filteredY = filteredX.filter(seat => {
const seatChar = seat.seatname.charAt(0);
return seatChar >= setSeatRange.startY && seatChar <= setSeatRange.endY;
});
// 가중치 적용
const seatWeights = filteredY.map(seat => ({ ...seat, weight: this.calculateSeatWeight({ x: seat.locxnm, y: seat.locynm }, setCenterSeat) }));
// 설정한 인원수 불충족 좌석 필터링 (= 좌석 연속성 확인)
const continuousSeats = seatWeights.filter(seat => this.countAdjacentSeats(seatWeights, seat.seatname) >= generalSeatCnt);
// 낮은 가중치순 정렬
const sortedSeats = continuousSeats.sort((a, b) => a.weight - b.weight);
if (sortedSeats.length === 0) {
await __INFO__(`자리가 없음!!!!`)
return -1
}
// 좌석 선택
const finalSeat = sortedSeats[0].seatname
await page.evaluate((seatname) => { jQuery(`#seat_table > tbody > tr > td[seatname='${seatname}']`).click(); }, finalSeat);
// 스마트결제로 이동
// await new Promise((resolve) => setTimeout(resolve, 1000000));
await page.evaluate(() => { submitSeat('O') });
//==================================================
// puppeteer - 결제 카드 선택
//==================================================
await page.waitForSelector('#ContainerView > article > div > ul > li:nth-child(1)', { visible: true });
await __INFO__(` ===== 카드창 들어옴`);
// 첫번째 결제 카드 클릭
await page.evaluate(() => { jQuery('#ContainerView > article > div > ul > li:nth-child(1)').click(); });
// 결제 버튼 클릭
await page.evaluate(() => { jQuery('#next').click(); });
//==================================================
// puppeteer - 결제 비번 입력
//==================================================
await page.waitForSelector('#ownKeypad', { visible: true });
await page.waitForFunction(() => window.nshc.finishedCallback !== null);
await __INFO__(` ===== 비번창 들어옴`);
// 간편결제 비밀번호 입력
const nums = process.env.EZPAY_PW.split('');
for (let num of nums) {
await page.evaluate((num) => {
jQuery(`#ownKeypad > div.kpdWrap.kpd-img.kpdNum.typeB > div.nfilter_keypad_div.kpdGrp.number > [aria-label="${num}"]`).click();
}, num);
}
await page.evaluate(() => { jQuery(`#ownKeypad > div.kpdWrap.kpd-img.kpdNum.typeB > div.nfilter_keypad_div.kpdGrp.number > #nfilter_enter`).click(); });
break
} catch (e) {
await __ERROR__(e);
failCnt++;
if (failCnt >= 10) return -1
else continue
}
}
await new Promise(resolve => setTimeout(resolve, 5000));
await page.screenshot({ path: 'result.png' });
return 1
}
}
await (async () => {
console.time('timer');
const CGV = new cgvGetter();
let mode = null;
if (process.argv[2] && process.argv[2] === '0') mode = 0;
else if (process.argv[2] && process.argv[2] === '1') mode = 1;
else if (process.argv[2] && process.argv[2] === '2') mode = 2;
switch (mode) {
/*====== infinite loop ======*/
case 0:
let cnt = 0;
const startTime = moment();
let movieData;
let prevMovieData = null; // 이전 movieData를 저장할 변수 추가
while (1) {
try {
cnt++;
const { cookies, data: result } = await CGV.getResultData("20035479", '20240218'); // 20035290 듄, 20035256 웡카
// 예외 - 요청 성공 확인
if (result === null) throw new Error(`\t> Request failed`);
// 예외(실험적) - 응답 데이터 구조 확인
if (!result.hasOwnProperty('d') || result.d.length === 0) throw new Error(`\t> Response lacks the expected "d" property`);
const gScheduleList = JSON.parse(result.d);
// 예외 - CGV 응답 코드 확인
if (gScheduleList.ResultCode !== "00000" || gScheduleList.ResultMessage !== "성공") throw new Error(`\t> Invalid schedule response`);
// 예외 - 영화 스케줄 존재 여부 확인
if (gScheduleList.ResultSchedule.ListPlayYmd.length === 0 && gScheduleList.ResultSchedule.ScheduleList.length === 0) {
throw new Error(`\t> Schedule not found`)
}
// 데이터 가공
const listPlayYmd = gScheduleList.ResultSchedule.ListPlayYmd;
movieData = listPlayYmd; // 현재 movieData 업데이트
// 이전 movieData와 현재 movieData 비교
if (!_.isEqual(movieData, prevMovieData)) {
await __INFO__(`스케줄 변경됨\n\t> 시도: ${cnt} / 현재시간: ${moment().format('MM-DD HH:mm:ss')} / 경과: ${startTime.fromNow()}\n\t${movieData.replace(/\|/g, "\n\t")}`);
prevMovieData = movieData; // 현재 movieData를 이전 movieData로 업데이트
} else {
await __INFO__(`스케줄 변경 없음\n\t> 시도: ${cnt} / 현재시간: ${moment().format('YYYY-MM-DD HH:mm:ss')} / 경과: ${startTime.fromNow()}`);
}
} catch (err) {
await __ERROR__(`사유: \n${err.message}\n\t> 시도: ${cnt} / 현재시간: ${moment().format('YYYY-MM-DD HH:mm:ss')} / 경과: ${startTime.fromNow()}`);
}
await new Promise(resolve => setTimeout(resolve, 60000));
}
break;
/*====== Dev Mode ======*/
case 1:
const result = await CGV.main_dev('20035290', null, moment('2024-03-09', 'YYYY-MM-DD')); // '20035290', null, moment('2024-03-09', 'YYYY-MM-DD' //// '20035479', null, moment('2024-02-16', 'YYYY-MM-DD'
if (result === -1) await __ERROR__('예매에 실패했습니다!');
else if (result === 1) await __INFO__(`예매에 성공했습니다!`);
break
/*====== Test Mode ======*/
case 2:
// let url = 'https://m.cgv.co.kr/Webapp/Member/Login.aspx';
// let param = {
// "hfUserId": "yymyeAbXPYSBovwH6LsywQ%3D%3D",
// "hfPasswordInter": "a273558ce0cebd7bb31d76055eb9cf4886d72ccb997fadd908f1ff0e4e4dd089",
// "hfPasswordLocal": "cfe9a2ee0e2989179a480c1dcbe8f07a556cc831f869cb069e623aa3f9d9e231",
// "hfReUrl": "Nkzul22VO0IswIN0zK0vHDanF3%2By9OXs4cqrIhv8vP0%3D",
// "hfAgree": "9LcHk229qY1EvJegzd%2FTQg%3D%3D",
// "nonmemberStateCd": "QB1eobbBB2OMOrekypNmww%3D%3D"
// };
// let headers = {
// 'Accept-Language': 'ko,en-US;q=0.9,en;q=0.8,ko-KR;q=0.7,de;q=0.6',
// 'Content-Type': 'application/x-www-form-urlencoded',
// Origin: 'https://m.cgv.co.kr',
// Cookie: COOKIE
// }
// let { cookies, data } = await CGV._post(url, param, headers);
// const joinedCookie = util.joinCookie(cookies);
// await __INFO__(util.parseCookie(joinedCookie))
// headers.Cookie = joinedCookie;
// ({ cookies, data } = await CGV._get('https://m.cgv.co.kr/WebApp/MyCgvV5/myMain.aspx', headers));
// await __INFO__(util.parseCookie(util.joinCookie(cookies)))
// const foo = {
// "hfUserId": "yymyeAbXPYSBovwH6LsywQ%3D%3D",
// "hfPasswordInter": "자체검열",
// "hfPasswordLocal": "자체검열",
// "hfReUrl": "Nkzul22VO0IswIN0zK0vHDanF3%2By9OXs4cqrIhv8vP0%3D",
// "hfAgree": "9LcHk229qY1EvJegzd%2FTQg%3D%3D",
// "nonmemberStateCd": "QB1eobbBB2OMOrekypNmww%3D%3D"
// }
// await __INFO__(crypto.decrypt_hfObject(foo))
const { cookies, data } = await CGV.getResultData('20035290', '20240228');
await __INFO__(data)
}
console.timeEnd('timer');
process.exit()
})();

2996
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

22
package.json Normal file
View File

@ -0,0 +1,22 @@
{
"name": "cgv_auto_reserve",
"version": "1.0.0",
"description": "",
"main": "app.js",
"scripts": {
"loop": "node ./cgvAjax.js 0",
"dev": "node ./cgvAjax.js 1",
"test": "node ./cgvAjax.js 2"
},
"author": "SeungJu Lim",
"license": "ISC",
"type": "module",
"dependencies": {
"axios": "^1.6.6",
"crypto-js": "^4.2.0",
"dotenv": "^16.4.1",
"moment": "^2.30.1",
"node-telegram-bot-api": "^0.64.0",
"puppeteer": "^22.0.0"
}
}

49
src/utils/commonUtil.js Normal file
View File

@ -0,0 +1,49 @@
import * as fs from 'fs';
import moment from 'moment';
import _ from 'lodash';
// moment obj <--> data string
const YMDConvert = (YYMMDD, format = 'YYYYMMDD') => {
let result;
if (typeof (YYMMDD) === 'string') {
result = moment(YYMMDD, format);
} else if (moment.isMoment(YYMMDD)) {
result = YYMMDD.format(format);
}
return result;
}
// filter specific day of week
const dayFilter = (dates) => {
return dates.filter(date => {
const day = date.day();
return day === 0 // || day === 1;
// sunday: 0 ~ saturday: 6
});
}
// cookie list from axios response -> joined cookie string
const joinCookie = (cookies) => {
return cookies.map(cookie => cookie.split(';')[0]).join('; ');
}
// joined cookie string -> cookie object
const parseCookie = (cookies) => {
return Object.fromEntries(cookies.split(';').map(cookie => cookie.split('=').map(c => c.trim())));
}
const mkHtml = (fileName, str) => {
fs.writeFileSync(path ? path : `./result/${fileName ? fileName : 'temp'}.html`, str);
}
const mkJson = (fileName, str) => {
fs.writeFileSync(`./result/${fileName ? fileName : 'temp'}.json`, str);
}
export {
YMDConvert,
dayFilter,
joinCookie,
parseCookie,
mkHtml,
mkJson
};

842
src/utils/cryptoUtil.js Normal file
View File

@ -0,0 +1,842 @@
import CryptoJS from 'crypto-js';
// var CryptoJS =
// CryptoJS ||
// (function (u, p) {
// var d = {},
// l = (d.lib = {}),
// s = function () { },
// t = (l.Base = {
// extend: function (a) {
// s.prototype = this;
// var c = new s();
// a && c.mixIn(a);
// c.hasOwnProperty('init') ||
// (c.init = function () {
// c.$super.init.apply(this, arguments);
// });
// c.init.prototype = c;
// c.$super = this;
// return c;
// },
// create: function () {
// var a = this.extend();
// a.init.apply(a, arguments);
// return a;
// },
// init: function () { },
// mixIn: function (a) {
// for (var c in a) a.hasOwnProperty(c) && (this[c] = a[c]);
// a.hasOwnProperty('toString') && (this.toString = a.toString);
// },
// clone: function () {
// return this.init.prototype.extend(this);
// },
// }),
// r = (l.WordArray = t.extend({
// init: function (a, c) {
// a = this.words = a || [];
// this.sigBytes = c != p ? c : 4 * a.length;
// },
// toString: function (a) {
// return (a || v).stringify(this);
// },
// concat: function (a) {
// var c = this.words,
// e = a.words,
// j = this.sigBytes;
// a = a.sigBytes;
// this.clamp();
// if (j % 4)
// for (var k = 0; k < a; k++)
// c[(j + k) >>> 2] |=
// ((e[k >>> 2] >>> (24 - 8 * (k % 4))) & 255) << (24 - 8 * ((j + k) % 4));
// else if (65535 < e.length) for (k = 0; k < a; k += 4) c[(j + k) >>> 2] = e[k >>> 2];
// else c.push.apply(c, e);
// this.sigBytes += a;
// return this;
// },
// clamp: function () {
// var a = this.words,
// c = this.sigBytes;
// a[c >>> 2] &= 4294967295 << (32 - 8 * (c % 4));
// a.length = u.ceil(c / 4);
// },
// clone: function () {
// var a = t.clone.call(this);
// a.words = this.words.slice(0);
// return a;
// },
// random: function (a) {
// for (var c = [], e = 0; e < a; e += 4) c.push((4294967296 * u.random()) | 0);
// return new r.init(c, a);
// },
// })),
// w = (d.enc = {}),
// v = (w.Hex = {
// stringify: function (a) {
// var c = a.words;
// a = a.sigBytes;
// for (var e = [], j = 0; j < a; j++) {
// var k = (c[j >>> 2] >>> (24 - 8 * (j % 4))) & 255;
// e.push((k >>> 4).toString(16));
// e.push((k & 15).toString(16));
// }
// return e.join('');
// },
// parse: function (a) {
// for (var c = a.length, e = [], j = 0; j < c; j += 2)
// e[j >>> 3] |= parseInt(a.substr(j, 2), 16) << (24 - 4 * (j % 8));
// return new r.init(e, c / 2);
// },
// }),
// b = (w.Latin1 = {
// stringify: function (a) {
// var c = a.words;
// a = a.sigBytes;
// for (var e = [], j = 0; j < a; j++)
// e.push(String.fromCharCode((c[j >>> 2] >>> (24 - 8 * (j % 4))) & 255));
// return e.join('');
// },
// parse: function (a) {
// for (var c = a.length, e = [], j = 0; j < c; j++)
// e[j >>> 2] |= (a.charCodeAt(j) & 255) << (24 - 8 * (j % 4));
// return new r.init(e, c);
// },
// }),
// x = (w.Utf8 = {
// stringify: function (a) {
// try {
// return decodeURIComponent(escape(b.stringify(a)));
// } catch (c) {
// throw Error('Malformed UTF-8 data');
// }
// },
// parse: function (a) {
// return b.parse(unescape(encodeURIComponent(a)));
// },
// }),
// q = (l.BufferedBlockAlgorithm = t.extend({
// reset: function () {
// this._data = new r.init();
// this._nDataBytes = 0;
// },
// _append: function (a) {
// 'string' == typeof a && (a = x.parse(a));
// this._data.concat(a);
// this._nDataBytes += a.sigBytes;
// },
// _process: function (a) {
// var c = this._data,
// e = c.words,
// j = c.sigBytes,
// k = this.blockSize,
// b = j / (4 * k),
// b = a ? u.ceil(b) : u.max((b | 0) - this._minBufferSize, 0);
// a = b * k;
// j = u.min(4 * a, j);
// if (a) {
// for (var q = 0; q < a; q += k) this._doProcessBlock(e, q);
// q = e.splice(0, a);
// c.sigBytes -= j;
// }
// return new r.init(q, j);
// },
// clone: function () {
// var a = t.clone.call(this);
// a._data = this._data.clone();
// return a;
// },
// _minBufferSize: 0,
// }));
// l.Hasher = q.extend({
// cfg: t.extend(),
// init: function (a) {
// this.cfg = this.cfg.extend(a);
// this.reset();
// },
// reset: function () {
// q.reset.call(this);
// this._doReset();
// },
// update: function (a) {
// this._append(a);
// this._process();
// return this;
// },
// finalize: function (a) {
// a && this._append(a);
// return this._doFinalize();
// },
// blockSize: 16,
// _createHelper: function (a) {
// return function (b, e) {
// return new a.init(e).finalize(b);
// };
// },
// _createHmacHelper: function (a) {
// return function (b, e) {
// return new n.HMAC.init(a, e).finalize(b);
// };
// },
// });
// var n = (d.algo = {});
// return d;
// })(Math);
// (function () {
// var u = CryptoJS,
// p = u.lib.WordArray;
// u.enc.Base64 = {
// stringify: function (d) {
// var l = d.words,
// p = d.sigBytes,
// t = this._map;
// d.clamp();
// d = [];
// for (var r = 0; r < p; r += 3)
// for (
// var w =
// (((l[r >>> 2] >>> (24 - 8 * (r % 4))) & 255) << 16) |
// (((l[(r + 1) >>> 2] >>> (24 - 8 * ((r + 1) % 4))) & 255) << 8) |
// ((l[(r + 2) >>> 2] >>> (24 - 8 * ((r + 2) % 4))) & 255),
// v = 0;
// 4 > v && r + 0.75 * v < p;
// v++
// )
// d.push(t.charAt((w >>> (6 * (3 - v))) & 63));
// if ((l = t.charAt(64))) for (; d.length % 4;) d.push(l);
// return d.join('');
// },
// parse: function (d) {
// var l = d.length,
// s = this._map,
// t = s.charAt(64);
// t && ((t = d.indexOf(t)), -1 != t && (l = t));
// for (var t = [], r = 0, w = 0; w < l; w++)
// if (w % 4) {
// var v = s.indexOf(d.charAt(w - 1)) << (2 * (w % 4)),
// b = s.indexOf(d.charAt(w)) >>> (6 - 2 * (w % 4));
// t[r >>> 2] |= (v | b) << (24 - 8 * (r % 4));
// r++;
// }
// return p.create(t, r);
// },
// _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=',
// };
// })();
// (function (u) {
// function p(b, n, a, c, e, j, k) {
// b = b + ((n & a) | (~n & c)) + e + k;
// return ((b << j) | (b >>> (32 - j))) + n;
// }
// function d(b, n, a, c, e, j, k) {
// b = b + ((n & c) | (a & ~c)) + e + k;
// return ((b << j) | (b >>> (32 - j))) + n;
// }
// function l(b, n, a, c, e, j, k) {
// b = b + (n ^ a ^ c) + e + k;
// return ((b << j) | (b >>> (32 - j))) + n;
// }
// function s(b, n, a, c, e, j, k) {
// b = b + (a ^ (n | ~c)) + e + k;
// return ((b << j) | (b >>> (32 - j))) + n;
// }
// for (
// var t = CryptoJS, r = t.lib, w = r.WordArray, v = r.Hasher, r = t.algo, b = [], x = 0;
// 64 > x;
// x++
// )
// b[x] = (4294967296 * u.abs(u.sin(x + 1))) | 0;
// r = r.MD5 = v.extend({
// _doReset: function () {
// this._hash = new w.init([1732584193, 4023233417, 2562383102, 271733878]);
// },
// _doProcessBlock: function (q, n) {
// for (var a = 0; 16 > a; a++) {
// var c = n + a,
// e = q[c];
// q[c] = (((e << 8) | (e >>> 24)) & 16711935) | (((e << 24) | (e >>> 8)) & 4278255360);
// }
// var a = this._hash.words,
// c = q[n + 0],
// e = q[n + 1],
// j = q[n + 2],
// k = q[n + 3],
// z = q[n + 4],
// r = q[n + 5],
// t = q[n + 6],
// w = q[n + 7],
// v = q[n + 8],
// A = q[n + 9],
// B = q[n + 10],
// C = q[n + 11],
// u = q[n + 12],
// D = q[n + 13],
// E = q[n + 14],
// x = q[n + 15],
// f = a[0],
// m = a[1],
// g = a[2],
// h = a[3],
// f = p(f, m, g, h, c, 7, b[0]),
// h = p(h, f, m, g, e, 12, b[1]),
// g = p(g, h, f, m, j, 17, b[2]),
// m = p(m, g, h, f, k, 22, b[3]),
// f = p(f, m, g, h, z, 7, b[4]),
// h = p(h, f, m, g, r, 12, b[5]),
// g = p(g, h, f, m, t, 17, b[6]),
// m = p(m, g, h, f, w, 22, b[7]),
// f = p(f, m, g, h, v, 7, b[8]),
// h = p(h, f, m, g, A, 12, b[9]),
// g = p(g, h, f, m, B, 17, b[10]),
// m = p(m, g, h, f, C, 22, b[11]),
// f = p(f, m, g, h, u, 7, b[12]),
// h = p(h, f, m, g, D, 12, b[13]),
// g = p(g, h, f, m, E, 17, b[14]),
// m = p(m, g, h, f, x, 22, b[15]),
// f = d(f, m, g, h, e, 5, b[16]),
// h = d(h, f, m, g, t, 9, b[17]),
// g = d(g, h, f, m, C, 14, b[18]),
// m = d(m, g, h, f, c, 20, b[19]),
// f = d(f, m, g, h, r, 5, b[20]),
// h = d(h, f, m, g, B, 9, b[21]),
// g = d(g, h, f, m, x, 14, b[22]),
// m = d(m, g, h, f, z, 20, b[23]),
// f = d(f, m, g, h, A, 5, b[24]),
// h = d(h, f, m, g, E, 9, b[25]),
// g = d(g, h, f, m, k, 14, b[26]),
// m = d(m, g, h, f, v, 20, b[27]),
// f = d(f, m, g, h, D, 5, b[28]),
// h = d(h, f, m, g, j, 9, b[29]),
// g = d(g, h, f, m, w, 14, b[30]),
// m = d(m, g, h, f, u, 20, b[31]),
// f = l(f, m, g, h, r, 4, b[32]),
// h = l(h, f, m, g, v, 11, b[33]),
// g = l(g, h, f, m, C, 16, b[34]),
// m = l(m, g, h, f, E, 23, b[35]),
// f = l(f, m, g, h, e, 4, b[36]),
// h = l(h, f, m, g, z, 11, b[37]),
// g = l(g, h, f, m, w, 16, b[38]),
// m = l(m, g, h, f, B, 23, b[39]),
// f = l(f, m, g, h, D, 4, b[40]),
// h = l(h, f, m, g, c, 11, b[41]),
// g = l(g, h, f, m, k, 16, b[42]),
// m = l(m, g, h, f, t, 23, b[43]),
// f = l(f, m, g, h, A, 4, b[44]),
// h = l(h, f, m, g, u, 11, b[45]),
// g = l(g, h, f, m, x, 16, b[46]),
// m = l(m, g, h, f, j, 23, b[47]),
// f = s(f, m, g, h, c, 6, b[48]),
// h = s(h, f, m, g, w, 10, b[49]),
// g = s(g, h, f, m, E, 15, b[50]),
// m = s(m, g, h, f, r, 21, b[51]),
// f = s(f, m, g, h, u, 6, b[52]),
// h = s(h, f, m, g, k, 10, b[53]),
// g = s(g, h, f, m, B, 15, b[54]),
// m = s(m, g, h, f, e, 21, b[55]),
// f = s(f, m, g, h, v, 6, b[56]),
// h = s(h, f, m, g, x, 10, b[57]),
// g = s(g, h, f, m, t, 15, b[58]),
// m = s(m, g, h, f, D, 21, b[59]),
// f = s(f, m, g, h, z, 6, b[60]),
// h = s(h, f, m, g, C, 10, b[61]),
// g = s(g, h, f, m, j, 15, b[62]),
// m = s(m, g, h, f, A, 21, b[63]);
// a[0] = (a[0] + f) | 0;
// a[1] = (a[1] + m) | 0;
// a[2] = (a[2] + g) | 0;
// a[3] = (a[3] + h) | 0;
// },
// _doFinalize: function () {
// var b = this._data,
// n = b.words,
// a = 8 * this._nDataBytes,
// c = 8 * b.sigBytes;
// n[c >>> 5] |= 128 << (24 - (c % 32));
// var e = u.floor(a / 4294967296);
// n[(((c + 64) >>> 9) << 4) + 15] =
// (((e << 8) | (e >>> 24)) & 16711935) | (((e << 24) | (e >>> 8)) & 4278255360);
// n[(((c + 64) >>> 9) << 4) + 14] =
// (((a << 8) | (a >>> 24)) & 16711935) | (((a << 24) | (a >>> 8)) & 4278255360);
// b.sigBytes = 4 * (n.length + 1);
// this._process();
// b = this._hash;
// n = b.words;
// for (a = 0; 4 > a; a++)
// (c = n[a]),
// (n[a] = (((c << 8) | (c >>> 24)) & 16711935) | (((c << 24) | (c >>> 8)) & 4278255360));
// return b;
// },
// clone: function () {
// var b = v.clone.call(this);
// b._hash = this._hash.clone();
// return b;
// },
// });
// t.MD5 = v._createHelper(r);
// t.HmacMD5 = v._createHmacHelper(r);
// })(Math);
// (function () {
// var u = CryptoJS,
// p = u.lib,
// d = p.Base,
// l = p.WordArray,
// p = u.algo,
// s = (p.EvpKDF = d.extend({
// cfg: d.extend({ keySize: 4, hasher: p.MD5, iterations: 1 }),
// init: function (d) {
// this.cfg = this.cfg.extend(d);
// },
// compute: function (d, r) {
// for (
// var p = this.cfg,
// s = p.hasher.create(),
// b = l.create(),
// u = b.words,
// q = p.keySize,
// p = p.iterations;
// u.length < q;
// ) {
// n && s.update(n);
// var n = s.update(d).finalize(r);
// s.reset();
// for (var a = 1; a < p; a++) (n = s.finalize(n)), s.reset();
// b.concat(n);
// }
// b.sigBytes = 4 * q;
// return b;
// },
// }));
// u.EvpKDF = function (d, l, p) {
// return s.create(p).compute(d, l);
// };
// })();
// CryptoJS.lib.Cipher ||
// (function (u) {
// var p = CryptoJS,
// d = p.lib,
// l = d.Base,
// s = d.WordArray,
// t = d.BufferedBlockAlgorithm,
// r = p.enc.Base64,
// w = p.algo.EvpKDF,
// v = (d.Cipher = t.extend({
// cfg: l.extend(),
// createEncryptor: function (e, a) {
// return this.create(this._ENC_XFORM_MODE, e, a);
// },
// createDecryptor: function (e, a) {
// return this.create(this._DEC_XFORM_MODE, e, a);
// },
// init: function (e, a, b) {
// this.cfg = this.cfg.extend(b);
// this._xformMode = e;
// this._key = a;
// this.reset();
// },
// reset: function () {
// t.reset.call(this);
// this._doReset();
// },
// process: function (e) {
// this._append(e);
// return this._process();
// },
// finalize: function (e) {
// e && this._append(e);
// return this._doFinalize();
// },
// keySize: 4,
// ivSize: 4,
// _ENC_XFORM_MODE: 1,
// _DEC_XFORM_MODE: 2,
// _createHelper: function (e) {
// return {
// encrypt: function (b, k, d) {
// return ('string' == typeof k ? c : a).encrypt(e, b, k, d);
// },
// decrypt: function (b, k, d) {
// return ('string' == typeof k ? c : a).decrypt(e, b, k, d);
// },
// };
// },
// }));
// d.StreamCipher = v.extend({
// _doFinalize: function () {
// return this._process(!0);
// },
// blockSize: 1,
// });
// var b = (p.mode = {}),
// x = function (e, a, b) {
// var c = this._iv;
// c ? (this._iv = u) : (c = this._prevBlock);
// for (var d = 0; d < b; d++) e[a + d] ^= c[d];
// },
// q = (d.BlockCipherMode = l.extend({
// createEncryptor: function (e, a) {
// return this.Encryptor.create(e, a);
// },
// createDecryptor: function (e, a) {
// return this.Decryptor.create(e, a);
// },
// init: function (e, a) {
// this._cipher = e;
// this._iv = a;
// },
// })).extend();
// q.Encryptor = q.extend({
// processBlock: function (e, a) {
// var b = this._cipher,
// c = b.blockSize;
// x.call(this, e, a, c);
// b.encryptBlock(e, a);
// this._prevBlock = e.slice(a, a + c);
// },
// });
// q.Decryptor = q.extend({
// processBlock: function (e, a) {
// var b = this._cipher,
// c = b.blockSize,
// d = e.slice(a, a + c);
// b.decryptBlock(e, a);
// x.call(this, e, a, c);
// this._prevBlock = d;
// },
// });
// b = b.CBC = q;
// q = (p.pad = {}).Pkcs7 = {
// pad: function (a, b) {
// for (
// var c = 4 * b,
// c = c - (a.sigBytes % c),
// d = (c << 24) | (c << 16) | (c << 8) | c,
// l = [],
// n = 0;
// n < c;
// n += 4
// )
// l.push(d);
// c = s.create(l, c);
// a.concat(c);
// },
// unpad: function (a) {
// a.sigBytes -= a.words[(a.sigBytes - 1) >>> 2] & 255;
// },
// };
// d.BlockCipher = v.extend({
// cfg: v.cfg.extend({ mode: b, padding: q }),
// reset: function () {
// v.reset.call(this);
// var a = this.cfg,
// b = a.iv,
// a = a.mode;
// if (this._xformMode == this._ENC_XFORM_MODE) var c = a.createEncryptor;
// else (c = a.createDecryptor), (this._minBufferSize = 1);
// this._mode = c.call(a, this, b && b.words);
// },
// _doProcessBlock: function (a, b) {
// this._mode.processBlock(a, b);
// },
// _doFinalize: function () {
// var a = this.cfg.padding;
// if (this._xformMode == this._ENC_XFORM_MODE) {
// a.pad(this._data, this.blockSize);
// var b = this._process(!0);
// } else (b = this._process(!0)), a.unpad(b);
// return b;
// },
// blockSize: 4,
// });
// var n = (d.CipherParams = l.extend({
// init: function (a) {
// this.mixIn(a);
// },
// toString: function (a) {
// return (a || this.formatter).stringify(this);
// },
// })),
// b = ((p.format = {}).OpenSSL = {
// stringify: function (a) {
// var b = a.ciphertext;
// a = a.salt;
// return (a ? s.create([1398893684, 1701076831]).concat(a).concat(b) : b).toString(r);
// },
// parse: function (a) {
// a = r.parse(a);
// var b = a.words;
// if (1398893684 == b[0] && 1701076831 == b[1]) {
// var c = s.create(b.slice(2, 4));
// b.splice(0, 4);
// a.sigBytes -= 16;
// }
// return n.create({ ciphertext: a, salt: c });
// },
// }),
// a = (d.SerializableCipher = l.extend({
// cfg: l.extend({ format: b }),
// encrypt: function (a, b, c, d) {
// d = this.cfg.extend(d);
// var l = a.createEncryptor(c, d);
// b = l.finalize(b);
// l = l.cfg;
// return n.create({
// ciphertext: b,
// key: c,
// iv: l.iv,
// algorithm: a,
// mode: l.mode,
// padding: l.padding,
// blockSize: a.blockSize,
// formatter: d.format,
// });
// },
// decrypt: function (a, b, c, d) {
// d = this.cfg.extend(d);
// b = this._parse(b, d.format);
// return a.createDecryptor(c, d).finalize(b.ciphertext);
// },
// _parse: function (a, b) {
// return 'string' == typeof a ? b.parse(a, this) : a;
// },
// })),
// p = ((p.kdf = {}).OpenSSL = {
// execute: function (a, b, c, d) {
// d || (d = s.random(8));
// a = w.create({ keySize: b + c }).compute(a, d);
// c = s.create(a.words.slice(b), 4 * c);
// a.sigBytes = 4 * b;
// return n.create({ key: a, iv: c, salt: d });
// },
// }),
// c = (d.PasswordBasedCipher = a.extend({
// cfg: a.cfg.extend({ kdf: p }),
// encrypt: function (b, c, d, l) {
// l = this.cfg.extend(l);
// d = l.kdf.execute(d, b.keySize, b.ivSize);
// l.iv = d.iv;
// b = a.encrypt.call(this, b, c, d.key, l);
// b.mixIn(d);
// return b;
// },
// decrypt: function (b, c, d, l) {
// l = this.cfg.extend(l);
// c = this._parse(c, l.format);
// d = l.kdf.execute(d, b.keySize, b.ivSize, c.salt);
// l.iv = d.iv;
// return a.decrypt.call(this, b, c, d.key, l);
// },
// }));
// })();
// (function () {
// for (
// var u = CryptoJS,
// p = u.lib.BlockCipher,
// d = u.algo,
// l = [],
// s = [],
// t = [],
// r = [],
// w = [],
// v = [],
// b = [],
// x = [],
// q = [],
// n = [],
// a = [],
// c = 0;
// 256 > c;
// c++
// )
// a[c] = 128 > c ? c << 1 : (c << 1) ^ 283;
// for (var e = 0, j = 0, c = 0; 256 > c; c++) {
// var k = j ^ (j << 1) ^ (j << 2) ^ (j << 3) ^ (j << 4),
// k = (k >>> 8) ^ (k & 255) ^ 99;
// l[e] = k;
// s[k] = e;
// var z = a[e],
// F = a[z],
// G = a[F],
// y = (257 * a[k]) ^ (16843008 * k);
// t[e] = (y << 24) | (y >>> 8);
// r[e] = (y << 16) | (y >>> 16);
// w[e] = (y << 8) | (y >>> 24);
// v[e] = y;
// y = (16843009 * G) ^ (65537 * F) ^ (257 * z) ^ (16843008 * e);
// b[k] = (y << 24) | (y >>> 8);
// x[k] = (y << 16) | (y >>> 16);
// q[k] = (y << 8) | (y >>> 24);
// n[k] = y;
// e ? ((e = z ^ a[a[a[G ^ z]]]), (j ^= a[a[j]])) : (e = j = 1);
// }
// var H = [0, 1, 2, 4, 8, 16, 32, 64, 128, 27, 54],
// d = (d.AES = p.extend({
// _doReset: function () {
// for (
// var a = this._key,
// c = a.words,
// d = a.sigBytes / 4,
// a = 4 * ((this._nRounds = d + 6) + 1),
// e = (this._keySchedule = []),
// j = 0;
// j < a;
// j++
// )
// if (j < d) e[j] = c[j];
// else {
// var k = e[j - 1];
// j % d
// ? 6 < d &&
// 4 == j % d &&
// (k =
// (l[k >>> 24] << 24) |
// (l[(k >>> 16) & 255] << 16) |
// (l[(k >>> 8) & 255] << 8) |
// l[k & 255])
// : ((k = (k << 8) | (k >>> 24)),
// (k =
// (l[k >>> 24] << 24) |
// (l[(k >>> 16) & 255] << 16) |
// (l[(k >>> 8) & 255] << 8) |
// l[k & 255]),
// (k ^= H[(j / d) | 0] << 24));
// e[j] = e[j - d] ^ k;
// }
// c = this._invKeySchedule = [];
// for (d = 0; d < a; d++)
// (j = a - d),
// (k = d % 4 ? e[j] : e[j - 4]),
// (c[d] =
// 4 > d || 4 >= j
// ? k
// : b[l[k >>> 24]] ^ x[l[(k >>> 16) & 255]] ^ q[l[(k >>> 8) & 255]] ^ n[l[k & 255]]);
// },
// encryptBlock: function (a, b) {
// this._doCryptBlock(a, b, this._keySchedule, t, r, w, v, l);
// },
// decryptBlock: function (a, c) {
// var d = a[c + 1];
// a[c + 1] = a[c + 3];
// a[c + 3] = d;
// this._doCryptBlock(a, c, this._invKeySchedule, b, x, q, n, s);
// d = a[c + 1];
// a[c + 1] = a[c + 3];
// a[c + 3] = d;
// },
// _doCryptBlock: function (a, b, c, d, e, j, l, f) {
// for (
// var m = this._nRounds,
// g = a[b] ^ c[0],
// h = a[b + 1] ^ c[1],
// k = a[b + 2] ^ c[2],
// n = a[b + 3] ^ c[3],
// p = 4,
// r = 1;
// r < m;
// r++
// )
// var q = d[g >>> 24] ^ e[(h >>> 16) & 255] ^ j[(k >>> 8) & 255] ^ l[n & 255] ^ c[p++],
// s = d[h >>> 24] ^ e[(k >>> 16) & 255] ^ j[(n >>> 8) & 255] ^ l[g & 255] ^ c[p++],
// t = d[k >>> 24] ^ e[(n >>> 16) & 255] ^ j[(g >>> 8) & 255] ^ l[h & 255] ^ c[p++],
// n = d[n >>> 24] ^ e[(g >>> 16) & 255] ^ j[(h >>> 8) & 255] ^ l[k & 255] ^ c[p++],
// g = q,
// h = s,
// k = t;
// q =
// ((f[g >>> 24] << 24) |
// (f[(h >>> 16) & 255] << 16) |
// (f[(k >>> 8) & 255] << 8) |
// f[n & 255]) ^
// c[p++];
// s =
// ((f[h >>> 24] << 24) |
// (f[(k >>> 16) & 255] << 16) |
// (f[(n >>> 8) & 255] << 8) |
// f[g & 255]) ^
// c[p++];
// t =
// ((f[k >>> 24] << 24) |
// (f[(n >>> 16) & 255] << 16) |
// (f[(g >>> 8) & 255] << 8) |
// f[h & 255]) ^
// c[p++];
// n =
// ((f[n >>> 24] << 24) |
// (f[(g >>> 16) & 255] << 16) |
// (f[(h >>> 8) & 255] << 8) |
// f[k & 255]) ^
// c[p++];
// a[b] = q;
// a[b + 1] = s;
// a[b + 2] = t;
// a[b + 3] = n;
// },
// keySize: 8,
// }));
// u.AES = p._createHelper(d);
// })();
/*
* Reversing Assets
* Extracted Components from CGV used for reverse engineering and reconstructing CGV's cryptography functions.
*/
const decryptCode = (encodedStr, base, count, keyArray, decodeFunc, dictionary) => {
decodeFunc = function (code) {
return (code < base ? "" : decodeFunc(parseInt(code / base))) +
((code = code % base) > 35 ? String.fromCharCode(code + 29) : code.toString(36));
};
if (!''.replace(/^/, String)) {
while (count--) dictionary[decodeFunc(count)] = keyArray[count] || decodeFunc(count);
keyArray = [function (code) { return dictionary[code]; }];
decodeFunc = function () { return '\\w+'; };
count = 1;
};
while (count--)
if (keyArray[count]) encodedStr = encodedStr.replace(new RegExp('\\b' + decodeFunc(count) + '\\b', 'g'), keyArray[count]);
return encodedStr;
}
const decryptedCode = decryptCode(
'q 9=["\\d\\H\\u\\E\\a\\f\\a\\g\\a\\f\\z\\r\\a\\F\\v\\g\\h\\n\\D\\I\\h\\n\\k\\A\\d\\M\\a\\r\\d\\M\\d\\A\\w\\V\\C\\U\\w\\x\\10\\Z\\Y\\x\\C\\o","\\G\\i\\N\\P","\\m\\G\\p\\R\\e","\\u\\i\\Q\\15","\\e\\B\\k","\\d\\H\\u\\E\\a\\f\\a\\g\\a\\f\\z\\r\\a\\F\\v\\g\\h\\n\\D\\I\\h\\y\\o\\o","\\e\\B\\k\\p\\L\\m\\i","\\y\\v\\14","\\13\\e\\k\\p\\L\\m\\i"];q s=c[9[4]][9[3]][9[2]](J[9[1]](9[0]));q t=c[9[4]][9[3]][9[2]](J[9[1]](9[5]));j 12(b){l c[9[7]][9[6]](b,s,{K:t})};j 11(b){l c[9[7]][9[8]](b,s,{K:t}).16(c[9[4]].S)};j O(b){l c.T(b)};j X(b){l c.W(b)};',
62,
69,
'|||||||||_0xd5d5|x4D|_0x9430x4|CryptoJS|x59|x65|x57|x33|x4E|x74|function|x63|return|x70|x44|x3D|x72|var|x35|encrypt_key|encrypt_iv|x55|x45|x5A|x54|x41|x49|x31|x6E|x51|x68|x78|x32|x61|x6A|x6D|window|iv|x79|x7A|x6F|cgv_sha256_encrypt|x62|x66|x73|Utf8|SHA256|x77|x47|MD5|cgv_md5_encrypt|x4F|x6C|x46|cgv_decrypt|cgv_encrypt|x64|x53|x38|toString'.split('|'),
0,
{}
)
/*
* Reconstructed Functions
* Functional encryption and decryption methods derived from the reversing process.
*/
const _0xd5d5 = [
'\x59\x6A\x55\x78\x4D\x57\x4D\x33\x4D\x57\x49\x35\x4D\x32\x45\x33\x4E\x44\x68\x6D\x4E\x44\x63\x31\x59\x7A\x4D\x35\x59\x7A\x59\x31\x5A\x47\x51\x77\x5A\x54\x46\x6C\x4F\x54\x51\x3D',
'\x61\x74\x6F\x62',
'\x70\x61\x72\x73\x65',
'\x55\x74\x66\x38',
'\x65\x6E\x63',
'\x59\x6A\x55\x78\x4D\x57\x4D\x33\x4D\x57\x49\x35\x4D\x32\x45\x33\x4E\x44\x68\x6D\x4E\x41\x3D\x3D',
'\x65\x6E\x63\x72\x79\x70\x74',
'\x41\x45\x53',
'\x64\x65\x63\x72\x79\x70\x74',
];
const encrypt_key = CryptoJS.enc.Utf8.parse(atob("YjUxMWM3MWI5M2E3NDhmNDc1YzM5YzY1ZGQwZTFlOTQ="));
const encrypt_iv = CryptoJS.enc.Utf8.parse(atob("YjUxMWM3MWI5M2E3NDhmNA=="));
const cgv_encrypt = (_0x9430x4) => { return CryptoJS.AES.encrypt(_0x9430x4, encrypt_key, { iv: encrypt_iv }) };
const cgv_decrypt = (_0x9430x4) => { return CryptoJS[_0xd5d5[7]][_0xd5d5[8]](_0x9430x4, encrypt_key, { iv: encrypt_iv }).toString(CryptoJS[_0xd5d5[4]].Utf8) };
const cgv_sha256_encrypt = (_0x9430x4) => { return CryptoJS.SHA256(_0x9430x4) };
const cgv_md5_encrypt = (_0x9430x4) => { return CryptoJS.MD5(_0x9430x4) };
const cgv_encrypt_hf = (_0x9430x4) => { return encodeURIComponent(cgv_encrypt(_0x9430x4)) }; // use with URLSearchParams()
const cgv_decrypt_hf = (_0x9430x4) => { return cgv_decrypt(decodeURIComponent(decodeURIComponent(_0x9430x4))) };
const encrypt_hfObject = (obj) => { return Object.fromEntries(Object.entries(obj).map(([key, value]) => [key, typeof value === 'string' ? cgv_encrypt_hf(value) : value])) };
const decrypt_hfObject = (obj) => { return Object.fromEntries(Object.entries(obj).map(([key, value]) => [key, typeof value === 'string' ? cgv_decrypt_hf(value) : value])) };
export {
cgv_encrypt,
cgv_decrypt,
cgv_sha256_encrypt,
cgv_md5_encrypt,
cgv_encrypt_hf,
cgv_decrypt_hf,
encrypt_hfObject,
decrypt_hfObject
};