Showing
12 changed files
with
220 additions
and
12 deletions
api/card/crawlerMulligan.js
0 → 100644
| 1 | +const puppeteer=require('puppeteer'); | ||
| 2 | +const cheerio=require('cheerio'); | ||
| 3 | + | ||
| 4 | +exports.getDecks=(cardIds)=>{ | ||
| 5 | + | ||
| 6 | + let idInQuery=cardIds[0] | ||
| 7 | + for(let i=1;i<cardIds.length;i++){ | ||
| 8 | + idInQuery+= '%2C'+cardIds[i] | ||
| 9 | + } | ||
| 10 | + const getContent=()=>{ | ||
| 11 | + return new Promise((resolve,reject)=>{ | ||
| 12 | + const asyncFunc=async ()=>{ | ||
| 13 | + const browser=await puppeteer.launch() | ||
| 14 | + try{ | ||
| 15 | + const page=await browser.newPage() | ||
| 16 | + await page.setViewport({width:1366,height:768}) | ||
| 17 | + await page.goto(`https://hsreplay.net/decks/#timeRange=LAST_30_DAYS&includedCards=${idInQuery}`,{waitUntil: 'networkidle2'}) | ||
| 18 | + const content=await page.content() | ||
| 19 | + browser.close() | ||
| 20 | + return content | ||
| 21 | + } | ||
| 22 | + catch(err) | ||
| 23 | + { | ||
| 24 | + console.log(err) | ||
| 25 | + browser.close() | ||
| 26 | + } | ||
| 27 | + } | ||
| 28 | + resolve(asyncFunc()) | ||
| 29 | + }) | ||
| 30 | + } | ||
| 31 | + | ||
| 32 | + const getDeckHref=(content)=>{ | ||
| 33 | + const $=cheerio.load(content) | ||
| 34 | + let deck=$('#decks-container > div > main > div.deck-list > ul > li:nth-child(2)').find('a') | ||
| 35 | + const deckHref=$(deck).attr('href') | ||
| 36 | + return deckHref | ||
| 37 | + } | ||
| 38 | + const getDeckConetent=(href)=>{ | ||
| 39 | + const asyncFunc=async ()=>{ | ||
| 40 | + const browser=await puppeteer.launch() | ||
| 41 | + try{ | ||
| 42 | + const page=await browser.newPage() | ||
| 43 | + await page.setViewport({width:1366,height:768}) | ||
| 44 | + await page.goto(`https://hsreplay.net${href}?hl=ko`,{waitUntil: 'networkidle2'}) | ||
| 45 | + const content=await page.content() | ||
| 46 | + browser.close() | ||
| 47 | + return content | ||
| 48 | + } | ||
| 49 | + catch(err) | ||
| 50 | + { | ||
| 51 | + console.log(err) | ||
| 52 | + browser.close() | ||
| 53 | + } | ||
| 54 | + } | ||
| 55 | + return asyncFunc() | ||
| 56 | + } | ||
| 57 | + const getMulligan=(content)=>{ | ||
| 58 | + const $=cheerio.load(content) | ||
| 59 | + let cardNames=$('.card-name') | ||
| 60 | + let cardWinRates=$('.table-cell') | ||
| 61 | + let cards=[] | ||
| 62 | + for(let i=0;i<cardNames.length;i++){ | ||
| 63 | + let cardName=$(cardNames[i]).text() | ||
| 64 | + let cardWinRate=$(cardWinRates[6*i]).text() | ||
| 65 | + cardWinRate=cardWinRate.replace('▼','') | ||
| 66 | + cardWinRate=cardWinRate.replace('▲','') | ||
| 67 | + cardWinRate=cardWinRate.replace('%','') | ||
| 68 | + cards.push({cardName:cardName,cardWinRate:cardWinRate}) | ||
| 69 | + } | ||
| 70 | + cards.sort((a,b)=>{ | ||
| 71 | + return a.cardWinRate<b.cardWinRate ? 1:-1 | ||
| 72 | + }) | ||
| 73 | + console.log(cards) | ||
| 74 | + | ||
| 75 | + } | ||
| 76 | + getContent() | ||
| 77 | + .then(getDeckHref) | ||
| 78 | + .then(getDeckConetent) | ||
| 79 | + .then(getMulligan) | ||
| 80 | +} |
api/card/crawlerOpponent.js
0 → 100644
| 1 | +const puppeteer=require('puppeteer'); | ||
| 2 | +const cheerio=require('cheerio'); | ||
| 3 | + | ||
| 4 | +exports.getDecks=(opponentClass)=>{ | ||
| 5 | + | ||
| 6 | + const getContent=()=>{ | ||
| 7 | + return new Promise((resolve,reject)=>{ | ||
| 8 | + const asyncFunc=async ()=>{ | ||
| 9 | + const browser=await puppeteer.launch() | ||
| 10 | + try{ | ||
| 11 | + const page=await browser.newPage() | ||
| 12 | + await page.setViewport({width:1366,height:768}) | ||
| 13 | + await page.goto(`https://hsreplay.net/decks/#timeRange=LAST_30_DAYS&playerClasses=${opponentClass}?hl=ko`,{waitUntil: 'networkidle2'}) | ||
| 14 | + const content=await page.content() | ||
| 15 | + browser.close() | ||
| 16 | + return content | ||
| 17 | + } | ||
| 18 | + catch(err) | ||
| 19 | + { | ||
| 20 | + console.log(err) | ||
| 21 | + browser.close() | ||
| 22 | + } | ||
| 23 | + } | ||
| 24 | + resolve(asyncFunc()) | ||
| 25 | + }) | ||
| 26 | + } | ||
| 27 | + | ||
| 28 | + const getDeckInfo=(content)=>{ | ||
| 29 | + const $=cheerio.load(content) | ||
| 30 | + let deckNames=$('.deck-name') | ||
| 31 | + let deckGames=$('.game-count') | ||
| 32 | + let decks=[] | ||
| 33 | + for(let i=0;i<3;i++){ | ||
| 34 | + let deckName=$(deckNames[i]).text() | ||
| 35 | + let deckGame=$(deckGames[i]).text() | ||
| 36 | + decks.push({deckTitle:deckName,deckGame:deckGame}) | ||
| 37 | + } | ||
| 38 | + console.log(decks) | ||
| 39 | + return decks | ||
| 40 | + } | ||
| 41 | + getContent() | ||
| 42 | + .then(getDeckInfo) | ||
| 43 | +} |
api/card/getCardId.js
0 → 100644
| 1 | +const mysql=require('../../database/mysql') | ||
| 2 | + | ||
| 3 | +exports.GetCardId=(deckId)=>{ | ||
| 4 | + return new Promise((resolve,reject)=>{ | ||
| 5 | + mysql.getConnection((err,connection)=>{ | ||
| 6 | + if (err) throw err | ||
| 7 | + connection.query(`select cardId from card where deckId=\'${deckId}\'`,(err,results,fields)=>{ | ||
| 8 | + if (err) throw err | ||
| 9 | + resolve(results) | ||
| 10 | + }) | ||
| 11 | + connection.release() | ||
| 12 | + }) | ||
| 13 | + }) | ||
| 14 | +} | ||
| ... | \ No newline at end of file | ... | \ No newline at end of file |
api/card/getMulligan.js
0 → 100644
| 1 | +const crawler=require('./crawlerMulligan') | ||
| 2 | +const getCardId=require('./getCardId') | ||
| 3 | + | ||
| 4 | +exports.GetMulligan=(req,res)=>{ | ||
| 5 | + const deckId=req.session.deckId || 113 | ||
| 6 | + const opponentClass=req.body.class || 'PALADIN' | ||
| 7 | + const DataCheck=()=>{ | ||
| 8 | + return new Promise((resolve,reject)=>{ | ||
| 9 | + if (!deckId || !opponentClass){ | ||
| 10 | + return reject({ | ||
| 11 | + code:'query_error', | ||
| 12 | + message:'query error', | ||
| 13 | + }) | ||
| 14 | + } | ||
| 15 | + resolve() | ||
| 16 | + }) | ||
| 17 | + } | ||
| 18 | + const GetCardId=()=>{ | ||
| 19 | + return getCardId.GetCardId(deckId) | ||
| 20 | + } | ||
| 21 | + const CrawlerMulligan=(cardIds)=>{ | ||
| 22 | + return crawler.getDecks(cardIds) | ||
| 23 | + } | ||
| 24 | + DataCheck() | ||
| 25 | + .then(GetCardId) | ||
| 26 | + .then(CrawlerMulligan) | ||
| 27 | + .then((cards)=>{ | ||
| 28 | + res.status(200).json(cards) | ||
| 29 | + }) | ||
| 30 | + .catch((err)=>{ | ||
| 31 | + res.status(500).json(err) | ||
| 32 | + }) | ||
| 33 | +} | ||
| ... | \ No newline at end of file | ... | \ No newline at end of file |
api/card/getOpponent.js
0 → 100644
| 1 | +const request=require('request') | ||
| 2 | +const iconv=require('iconv-lite') | ||
| 3 | +const charset=require('charset') | ||
| 4 | +const crawler=require('./crawlerOpponent') | ||
| 5 | + | ||
| 6 | +exports.GetOpponent=(req,res)=>{ | ||
| 7 | + const opponentClass=req.body.class | ||
| 8 | + const DataCheck=()=>{ | ||
| 9 | + return new Promise((resolve,reject)=>{ | ||
| 10 | + if(!opponentClass){ | ||
| 11 | + return reject({ | ||
| 12 | + code:'query_error', | ||
| 13 | + message:'query_error' | ||
| 14 | + }) | ||
| 15 | + } | ||
| 16 | + resolve() | ||
| 17 | + }) | ||
| 18 | + } | ||
| 19 | + const CralwerOpponent=()=>{ | ||
| 20 | + return crawler.getDecks(opponentClass) | ||
| 21 | + } | ||
| 22 | + DataCheck | ||
| 23 | + .then(CralwerOpponent) | ||
| 24 | + .then((decks)=>{ | ||
| 25 | + res.status(200).json(decks) | ||
| 26 | + }) | ||
| 27 | + .catch((err)=>{ | ||
| 28 | + res.status(500).json(err) | ||
| 29 | + }) | ||
| 30 | +} | ||
| ... | \ No newline at end of file | ... | \ No newline at end of file |
| 1 | const express=require('express') | 1 | const express=require('express') |
| 2 | const router=express.Router() | 2 | const router=express.Router() |
| 3 | 3 | ||
| 4 | +const getMulligan=require('./getMulligan') | ||
| 5 | + | ||
| 6 | +router.post('/getmulligan',getMulligan.GetMulligan) | ||
| 7 | + | ||
| 4 | module.exports=router | 8 | module.exports=router |
| ... | \ No newline at end of file | ... | \ No newline at end of file | ... | ... |
| 1 | -const crawler=require('./crawler') | 1 | +const crawler=require('./crawlerDeckCodes') |
| 2 | const cheerio=require('cheerio') | 2 | const cheerio=require('cheerio') |
| 3 | const addCards=require('../card/addCards') | 3 | const addCards=require('../card/addCards') |
| 4 | const addDeck=require('../../database/deck/addDeck') | 4 | const addDeck=require('../../database/deck/addDeck') | ... | ... |
| ... | @@ -6,7 +6,7 @@ exports.GetDeckId=(deckOwner,deckTitle)=>{ | ... | @@ -6,7 +6,7 @@ exports.GetDeckId=(deckOwner,deckTitle)=>{ |
| 6 | if (err) throw err | 6 | if (err) throw err |
| 7 | connection.query(`select * from deck where deckOwner=\'${deckOwner}\' and deckTitle=\'${deckTitle}\'`,(err,results,field)=>{ | 7 | connection.query(`select * from deck where deckOwner=\'${deckOwner}\' and deckTitle=\'${deckTitle}\'`,(err,results,field)=>{ |
| 8 | if (err) throw err | 8 | if (err) throw err |
| 9 | - console.log('result in getDeckId'+results[0]) | 9 | + //console.log('result in getDeckId'+results[0]) |
| 10 | connection.release() | 10 | connection.release() |
| 11 | resolve(results[0].id) | 11 | resolve(results[0].id) |
| 12 | }) | 12 | }) | ... | ... |
This diff is collapsed. Click to expand it.
| ... | @@ -27,6 +27,7 @@ | ... | @@ -27,6 +27,7 @@ |
| 27 | "mysql": "^2.16.0", | 27 | "mysql": "^2.16.0", |
| 28 | "mysql-apostrophe": "^1.0.8", | 28 | "mysql-apostrophe": "^1.0.8", |
| 29 | "path": "^0.12.7", | 29 | "path": "^0.12.7", |
| 30 | + "puppeteer": "^1.11.0", | ||
| 30 | "request": "^2.88.0", | 31 | "request": "^2.88.0", |
| 31 | "request-promise": "^4.2.2" | 32 | "request-promise": "^4.2.2" |
| 32 | } | 33 | } | ... | ... |
| ... | @@ -32,6 +32,9 @@ | ... | @@ -32,6 +32,9 @@ |
| 32 | } | 32 | } |
| 33 | }) | 33 | }) |
| 34 | }) | 34 | }) |
| 35 | + $('#submitButton').click(function(){ | ||
| 36 | + | ||
| 37 | + }) | ||
| 35 | }) | 38 | }) |
| 36 | </script> | 39 | </script> |
| 37 | </head> | 40 | </head> |
| ... | @@ -66,17 +69,17 @@ | ... | @@ -66,17 +69,17 @@ |
| 66 | <div class="col-md-4"> | 69 | <div class="col-md-4"> |
| 67 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/c/c5/Garrosh_Hellscream%28635%29.png?version=bab934001bb784a94c59a47823d535a7" style="width:150px;height:200px;"/> | 70 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/c/c5/Garrosh_Hellscream%28635%29.png?version=bab934001bb784a94c59a47823d535a7" style="width:150px;height:200px;"/> |
| 68 | <br> | 71 | <br> |
| 69 | - <input type="radio" name="checkOpponent" value="전사"/>전사 | 72 | + <input type="radio" name="checkOpponent" value="WARRIOR"/>전사 |
| 70 | </div> | 73 | </div> |
| 71 | <div class="col-md-4"> | 74 | <div class="col-md-4"> |
| 72 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/4/4b/Thrall%28319%29.png?version=adcee55715548b949a7d973c2fddbd95" style="width:150px;height:200px;"/> | 75 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/4/4b/Thrall%28319%29.png?version=adcee55715548b949a7d973c2fddbd95" style="width:150px;height:200px;"/> |
| 73 | <br> | 76 | <br> |
| 74 | - <input type="radio" name="checkOpponent" value="주술사"/>주술사 | 77 | + <input type="radio" name="checkOpponent" value="SHAMAN"/>주술사 |
| 75 | </div> | 78 | </div> |
| 76 | <div class="col-md-4"> | 79 | <div class="col-md-4"> |
| 77 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/a/a4/Valeera_Sanguinar%282%29.png?version=84a816910b223169eb14cc93c20437b2" style="width:150px;height:200px;"/> | 80 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/a/a4/Valeera_Sanguinar%282%29.png?version=84a816910b223169eb14cc93c20437b2" style="width:150px;height:200px;"/> |
| 78 | <br> | 81 | <br> |
| 79 | - <input type="radio" name="checkOpponent" value="도적"/>도적 | 82 | + <input type="radio" name="checkOpponent" value="ROGUE"/>도적 |
| 80 | </div> | 83 | </div> |
| 81 | </div> | 84 | </div> |
| 82 | <br> | 85 | <br> |
| ... | @@ -86,17 +89,17 @@ | ... | @@ -86,17 +89,17 @@ |
| 86 | <div class="col-md-4"> | 89 | <div class="col-md-4"> |
| 87 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/4/4d/Uther_Lightbringer%28257%29.png?version=b45ade5ac3fdd2579160fe5d7b7c1b20" style="width:150px;height:200px;"/> | 90 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/4/4d/Uther_Lightbringer%28257%29.png?version=b45ade5ac3fdd2579160fe5d7b7c1b20" style="width:150px;height:200px;"/> |
| 88 | <br> | 91 | <br> |
| 89 | - <input type="radio" name="checkOpponent" value="성기사"/>성기사 | 92 | + <input type="radio" name="checkOpponent" value="PALADIN"/>성기사 |
| 90 | </div> | 93 | </div> |
| 91 | <div class="col-md-4"> | 94 | <div class="col-md-4"> |
| 92 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/a/a0/Rexxar%28484%29.png?version=c21b57837db15d20cc814f2bf45682b6" style="width:150px;height:200px;"/> | 95 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/a/a0/Rexxar%28484%29.png?version=c21b57837db15d20cc814f2bf45682b6" style="width:150px;height:200px;"/> |
| 93 | <br> | 96 | <br> |
| 94 | - <input type="radio" name="checkOpponent" value="사냥꾼"/>사냥꾼 | 97 | + <input type="radio" name="checkOpponent" value="HUNTER"/>사냥꾼 |
| 95 | </div> | 98 | </div> |
| 96 | <div class="col-md-4"> | 99 | <div class="col-md-4"> |
| 97 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/f/fa/Malfurion_Stormrage%28621%29.png?version=b3f5a40e33f33d32995f3becbdd7aa94" style="width:150px;height:200px;"/> | 100 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/f/fa/Malfurion_Stormrage%28621%29.png?version=b3f5a40e33f33d32995f3becbdd7aa94" style="width:150px;height:200px;"/> |
| 98 | <br> | 101 | <br> |
| 99 | - <input type="radio" name="checkOpponent" value="드루이드"/>드루이드 | 102 | + <input type="radio" name="checkOpponent" value="DRUID"/>드루이드 |
| 100 | </div> | 103 | </div> |
| 101 | </div> | 104 | </div> |
| 102 | <br> | 105 | <br> |
| ... | @@ -106,17 +109,17 @@ | ... | @@ -106,17 +109,17 @@ |
| 106 | <div class="col-md-4"> | 109 | <div class="col-md-4"> |
| 107 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/0/0a/Gul%27dan%28618%29.png?version=90f421585c6f2d493ba94e259a76190e" style="width:150px;height:200px;"/> | 110 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/0/0a/Gul%27dan%28618%29.png?version=90f421585c6f2d493ba94e259a76190e" style="width:150px;height:200px;"/> |
| 108 | <br> | 111 | <br> |
| 109 | - <input type="radio" name="checkOpponent" value="흑마법사"/>흑마법사 | 112 | + <input type="radio" name="checkOpponent" value="WARLOCK"/>흑마법사 |
| 110 | </div> | 113 | </div> |
| 111 | <div class="col-md-4"> | 114 | <div class="col-md-4"> |
| 112 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/3/3c/Jaina_Proudmoore%28320%29.png?version=75868a59a53f90bce829edeb66126b73" style="width:150px;height:200px;"/> | 115 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/3/3c/Jaina_Proudmoore%28320%29.png?version=75868a59a53f90bce829edeb66126b73" style="width:150px;height:200px;"/> |
| 113 | <br> | 116 | <br> |
| 114 | - <input type="radio" name="checkOpponent" value="마법사"/>마법사 | 117 | + <input type="radio" name="checkOpponent" value="MAGE"/>마법사 |
| 115 | </div> | 118 | </div> |
| 116 | <div class="col-md-4"> | 119 | <div class="col-md-4"> |
| 117 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/8/80/Anduin_Wrynn%28110%29.png?version=ba8ecc39b3fdd4a2ede72e046c434454" style="width:150px;height:200px;"/> | 120 | <img src="https://d1u5p3l4wpay3k.cloudfront.net/hearthstone_gamepedia/8/80/Anduin_Wrynn%28110%29.png?version=ba8ecc39b3fdd4a2ede72e046c434454" style="width:150px;height:200px;"/> |
| 118 | <br> | 121 | <br> |
| 119 | - <input type="radio" name="checkOpponent" value="사제"/>사제 | 122 | + <input type="radio" name="checkOpponent" value="PRIEST"/>사제 |
| 120 | </div> | 123 | </div> |
| 121 | </div> | 124 | </div> |
| 122 | <br> | 125 | <br> |
| ... | @@ -124,7 +127,7 @@ | ... | @@ -124,7 +127,7 @@ |
| 124 | <div class="row"> | 127 | <div class="row"> |
| 125 | <div class="col-md-5"></div> | 128 | <div class="col-md-5"></div> |
| 126 | <div class="col-md-2"> | 129 | <div class="col-md-2"> |
| 127 | - <input class="btn btn-lg btn-primary btn-block" type="button" value="확인"/> | 130 | + <input class="btn btn-lg btn-primary btn-block" type="button" id="submitButton" value="확인"/> |
| 128 | </div> | 131 | </div> |
| 129 | <div class="col-md-5"></div> | 132 | <div class="col-md-5"></div> |
| 130 | </div> | 133 | </div> | ... | ... |
-
Please register or login to post a comment