cgv_auto_reservation/cgvAjax.js

557 lines
24 KiB
JavaScript

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 './src/utils/commonUtil.js';
import * as crypto from './src/utils/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]'
try{
await bot.sendMessage(BOT_CHATID, `${prefix} ${str}`);
}catch(err){
}
console.log(`${prefix} ${str}`);
}
const __ERROR__ = async (str) => {
const prefix = '[ERROR]'
try{
await bot.sendMessage(BOT_CHATID, `${prefix} ${str}`);
}catch(err){
}
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: '1200', end: '1800' };
let targetYMD = null;
let targetSchedule = null;
let attemptCnt = 0;
const init = moment();
//==================================================
// requset - 날짜 탐색
//==================================================
while (1) {
// 스케줄 데이터 요청
let cookies, result;
let attempts = 0;
while (attempts < 10) {
const response = await this.getResultData(mgCD, targetYMD);
if (response) {
({ cookies, data: result } = response);
break;
}
attempts++;
await new Promise(resolve => setTimeout(resolve, 1000));
}
if (!cookies && !result) {
await __ERROR__(`사유: 스케줄 추출 작업 에러 / 현재시간: ${moment().format('YYYY-MM-DD HH:mm:ss')}`);
return -1
}
// 예외(실험적) - 응답 데이터 구조 확인
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
}
let loginCookies;
let attempts = 0;
while (attempts < 10) {
const response = await this._post(this.loginAspx, loginParam, loginHeaders);
if (response) {
({ cookies: loginCookies } = response);
break;
}
attempts++;
await new Promise(resolve => setTimeout(resolve, 1000));
}
if (!loginCookies) {
await __ERROR__(`사유: 쿠키 추출 작업 에러 / 현재시간: ${moment().format('YYYY-MM-DD HH:mm:ss')}`);
return -1
}
/* ++++++++++++ 예외 추가 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-16', '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()
})();