├── .gitignore ├── README.md ├── batch ├── app.js ├── instance.js ├── master.js └── worker.js ├── browser ├── cache.js ├── cookie.js ├── index.js ├── test.js └── ua.js ├── comm └── index.js ├── conf.js ├── db ├── index.js └── model.js ├── export ├── fetch.js ├── rename.js └── saved.js ├── import ├── 1_caobi45.js ├── 2_668wy.js ├── 3_bka8.js ├── 4_14xav.js ├── 5_331sss.js └── 6_8x3a.js ├── index.js ├── others ├── move.js ├── reext.js └── reid.js ├── package-lock.json ├── package.json ├── runing.png ├── static ├── ckplayer │ ├── ckplayer.js │ ├── ckplayer.min.js │ ├── ckplayer.swf │ ├── ckplayer.xml │ ├── hls │ │ ├── LICENSE │ │ ├── hls.js │ │ └── hls.min.js │ ├── language.xml │ └── style.xml ├── crossdomain.xml └── favicon.ico └── www └── index.ejs /.gitignore: -------------------------------------------------------------------------------- 1 | .idea 2 | node_modules 3 | *.log 4 | *.iml 5 | *.db 6 | *.db-journal 7 | .cache 8 | fetch.page 9 | fetch.txt 10 | rename.err 11 | rename.txt 12 | threads 13 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # 91video 2 | 3 | > The adult video from 91 team 4 | 5 | ### Usage 6 | 7 | - install NodeJS 8 | - npm install 9 | 10 | ### Import videos 11 | 12 | - cd import 13 | - node 6_8x3a.js 14 | 15 | ### Export videos 16 | 17 | - cd export 18 | - node fetch js 19 | - batch download use Thunder 20 | - node rename.js 21 | 22 | ### Run local web 23 | 24 | - setting table name 25 | - npm start 26 | 27 | *more than 40000+ videos, good climax for you* 28 | 29 | ![](running.png) -------------------------------------------------------------------------------- /batch/app.js: -------------------------------------------------------------------------------- 1 | const cluster = require('cluster'); 2 | // 主程序 3 | const Master = require('./master'); 4 | const Worker = require('./worker'); 5 | // 爬虫实例 6 | const instance = require('./instance'); 7 | 8 | (async () => { 9 | if (cluster.isMaster) { 10 | try { 11 | await Master.start(instance); 12 | } catch (e) { 13 | console.error(e); 14 | Master.stop(); 15 | } 16 | } else { 17 | Worker.start(instance); 18 | } 19 | })().catch(err => console.error(err)); 20 | -------------------------------------------------------------------------------- /batch/instance.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs'); 2 | const path = require('path'); 3 | const fetch = require('node-fetch'); 4 | const cheerio = require('cheerio'); 5 | 6 | const outDir = '/Volumes/Macintosh SD'; 7 | const mmDir = path.join(outDir, 'Meizi'); 8 | const dataDir = path.join(outDir, 'Meizi-Data'); 9 | const cacheDir = path.join(outDir, 'Meizi-Cache'); 10 | 11 | const instance = {}; 12 | 13 | instance.urls = []; 14 | 15 | // 主线程刷列表 16 | instance.main = async () => { 17 | let tags = ['性感', '平面模特', '内地90后', '模特', '日本偶像', '日本模特', '内地平面模特', '中国模特', '比基尼', '正妹', '美腿', 18 | 'showgirl', '巨乳', '台湾正妹', '妹子', '内地模特', '诱惑', 'cosplay', '清纯', '美乳', '酥胸', '台湾模特', '美国模特', '爆乳', 19 | '写真', '美女', '日本演员', 'Coser', '私房', 'Jkf女郎', '翘臀', '车模', '可爱', '童颜巨乳', '女神', '日本歌手', '半裸', '大尺度', 20 | '韩国模特', '全裸']; 21 | for (let i = 0; i < tags.length; i++) { 22 | console.debug('===>', `${i + 1}/${tags.length}`, tags[i]); 23 | let tag = {}; 24 | tag.name = tags[i]; 25 | tag.url = `http://www.177521.com/e/tags/?tagname=${encodeURIComponent(tag.name)}`; 26 | const cache_file = path.join(dataDir, `${tag.name}.json`); 27 | if (fs.existsSync(cache_file)) { 28 | tag = JSON.parse(fs.readFileSync(cache_file)); 29 | } else { 30 | let $ = await getWeb(tag.url); 31 | if ($ == null) { 32 | return; 33 | } 34 | tag.mm = []; 35 | let tags = $('.update_area_lists a'); 36 | console.debug(tag.name, '第1页', tags.length, 'tag'); 37 | tags.each(function () { 38 | tag.mm.push($(this).attr('href')); 39 | }); 40 | const nexts = []; 41 | $('.nav-links a').each(function () { 42 | if ($(this).text().startsWith('第')) { 43 | nexts.push($(this).attr('href').replace(tag.name, encodeURIComponent(tag.name))); 44 | } 45 | }); 46 | for (let i = 0; i < nexts.length; i++) { 47 | $ = await getWeb(nexts[i]); 48 | if ($ == null) { 49 | continue; 50 | } 51 | tags = $('.update_area_lists a'); 52 | console.debug(`${tag.name} 第${parseInt(i) + 2}页`, tags.length, 'tag'); 53 | tags.each(function () { 54 | tag.mm.push($(this).attr('href')); 55 | }); 56 | } 57 | fs.writeFileSync(cache_file, JSON.stringify(tag, null, 2)); 58 | } 59 | for (let i = 0; i < tag.mm.length; i++) { 60 | instance.urls.push(tag.mm[i]); 61 | } 62 | } 63 | } 64 | 65 | // 主线程分配任务 66 | instance.getTask = function () { 67 | const url = instance.urls.shift(); 68 | console.log(`取到了任务 ${url}`); 69 | return url; 70 | } 71 | 72 | // 子线程执行任务 73 | instance.execTask = async function (url) { 74 | const id = url.split('/')[2].split('.')[0]; 75 | if (id === 'www') { 76 | return; 77 | } 78 | let mm = {}; 79 | mm.id = id; 80 | const cache_file = path.join(dataDir, `${id}.json`); 81 | if (fs.existsSync(cache_file)) { 82 | mm = JSON.parse(fs.readFileSync(cache_file)); 83 | console.debug(mm.id, mm.name); 84 | } else { 85 | mm.url = `http://www.177521.com` + url; 86 | let $ = await getWeb(mm.url); 87 | if ($ == null) { 88 | return; 89 | } 90 | mm.name = $('title').text().split('--177521')[0].split('-177521')[0].replaceAll('/', '_'); 91 | mm.img = []; 92 | console.debug(mm.id, mm.name); 93 | let imgs = $('.content_left img'); 94 | if (imgs.length > 0) { 95 | console.debug(`${mm.id} 第1页`, imgs.length, 'img'); 96 | imgs.each(function () { 97 | mm.img.push($(this).attr('src')); 98 | }); 99 | } 100 | imgs = $('.image_div img'); 101 | if (imgs.length > 0) { 102 | console.debug(mm.id, '第1页', imgs.length, 'img'); 103 | imgs.each(function () { 104 | mm.img.push($(this).attr('src')); 105 | }); 106 | let nexts = []; 107 | $('.page_imges a').each(function () { 108 | if ($(this).text().startsWith('第')) { 109 | nexts.push($(this).attr('href')); 110 | } 111 | }); 112 | for (let i = 0; i < nexts.length; i++) { 113 | $ = await getWeb(nexts[i]); 114 | if ($ == null) { 115 | continue; 116 | } 117 | imgs = $('.image_div img'); 118 | console.debug(`${mm.id} 第${parseInt(i) + 2}页`, imgs.length, 'img'); 119 | imgs.each(function () { 120 | mm.img.push($(this).attr('src')); 121 | }); 122 | } 123 | } 124 | fs.writeFileSync(cache_file, JSON.stringify(mm, null, 2)); 125 | } 126 | if (mm.img.length > 0) { 127 | mm.name = mm.name.replaceAll('/', '_'); 128 | for (let i = 0; i < mm.img.length; i++) { 129 | if (typeof mm.img[i] === 'string') { 130 | const name = path.basename(mm.img[i]); 131 | const newId = (Array(5).join('0') + mm.id).slice(-5); 132 | const idx = (Array(3).join('0') + (i + 1)).slice(-3) 133 | const newName = `${newId}-${idx}${path.extname(name)}`; 134 | const dst = path.join(mmDir, newName); 135 | if (!fs.existsSync(dst)) { 136 | console.log('mmDir =>', dst); 137 | await getImg(mm, i, dst); 138 | } 139 | } 140 | } 141 | } 142 | } 143 | 144 | function opts(url) { 145 | return { 146 | headers: { 147 | 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3', 148 | 'Accept-Encoding': 'gzip, deflate, br', 149 | 'Accept-Language': 'zh-CN,zh;q=0.9,en;q=0.8,ja;q=0.7,zh-TW;q=0.6', 150 | 'Cache-Control': 'no-cache', 151 | 'Connection': 'keep-alive', 152 | 'Cookie': 'Hm_lvt_b96fe021352b520f1524a6deb63c9bc8=1617280098,1617295170,1617298156,1617636064', 153 | 'Host': url.split('/')[2], 154 | 'Pragma': 'no-cache', 155 | 'Sec-Fetch-Mode': 'navigate', 156 | 'Sec-Fetch-Site': 'none', 157 | 'Sec-Fetch-User': '?1', 158 | 'Upgrade-Insecure-Requests': '1', 159 | 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36', 160 | } 161 | } 162 | } 163 | 164 | // 下载网页 165 | async function getWeb(url) { 166 | try { 167 | if (url.startsWith('/')) { 168 | url = 'http://www.177521.com' + url; 169 | } 170 | let cache_url = url.split('://')[1].replaceAll('/', '_'); 171 | let cache_file = path.join(cacheDir, cache_url); 172 | let body; 173 | if (fs.existsSync(cache_file)) { 174 | body = fs.readFileSync(cache_file); 175 | } else { 176 | console.debug('--->', url); 177 | body = await fetch(url, opts(url)).then(res => res.text()); 178 | } 179 | if (body && body.length > 100) { 180 | fs.writeFileSync(cache_file, body); 181 | return cheerio.load(body); 182 | } 183 | } catch (e) { 184 | console.error(e.message); 185 | } 186 | return null; 187 | } 188 | 189 | // 下载图片 190 | async function getImg(mm, i, dst) { 191 | try { 192 | let url = mm.img[i]; 193 | if (url.startsWith('/')) { 194 | url = 'http://www.177521.com' + url; 195 | } 196 | await fetch(url, opts(url)).then(res => { 197 | if (res.headers.get('content-type') === 'text/html') { 198 | console.error('下载失败,返回的是text/html', url); 199 | } else { 200 | res.body.pipe(fs.createWriteStream(dst)); 201 | console.log(mm.id, 'save', url, 'success'); 202 | } 203 | }); 204 | } catch (e) { 205 | console.error(e.message); 206 | } 207 | } 208 | 209 | module.exports = instance; -------------------------------------------------------------------------------- /batch/master.js: -------------------------------------------------------------------------------- 1 | const os = require('os'); 2 | const cluster = require('cluster'); 3 | const cupNums = os.cpus().length; 4 | 5 | module.exports.start = async function(instance) { 6 | console.log(`主进程${process.pid}开始`); 7 | for (let i = 0; i < cupNums; i++) { 8 | let worker = cluster.fork(); 9 | worker.send({ 10 | do: 'start', 11 | tip: `工人${worker.id}号,开始工作吧`, 12 | data: worker.id, 13 | }); 14 | } 15 | cluster.on('message', function(worker, message) { 16 | console.log(`[0-${process.pid}]: 收到[${worker.id}-${worker.process.pid}]的消息:`, message); 17 | if (message === '我要下班') { 18 | console.log(`[0-${process.pid}]: 让[${worker.id}-${worker.process.pid}]下班`); 19 | worker.send({ 20 | do: 'stop', 21 | tip: `你下班吧,给你1秒钟消失`, 22 | }); 23 | setTimeout(function() { 24 | console.log(`[0-${process.pid}]: 断开与[${worker.id}-${worker.process.pid}]的IPC管道`); 25 | worker.disconnect(); // 主动断开IPC管道 26 | }, 1000); 27 | } else { 28 | console.log(`[0-${process.pid}]: 给[${worker.id}-${worker.process.pid}]派发一个任务`); 29 | worker.send({ 30 | do: 'task', 31 | tip: '起来干活', 32 | data: instance.getTask(), 33 | }); 34 | } 35 | }); 36 | cluster.on('online', function(worker) { 37 | console.log(`[0-${process.pid}]: 启动[${worker.id}-${worker.process.pid}]`); 38 | }); 39 | cluster.on('listening', function(worker, address) { 40 | console.log(`[0-${process.pid}]: [${worker.id}-${worker.process.pid}]正在监听 ${address.address || '*'}:${address.port}`); 41 | }); 42 | cluster.on('disconnect', (worker) => { 43 | console.log(`[0-${process.pid}]: [${worker.id}-${worker.process.pid}]IPC管道已断开`); 44 | }); 45 | cluster.on('exit', function(worker, code, signal) { 46 | console.log(`[0-${process.pid}]: [${worker.id}-${worker.process.pid}]已停止,退出码=${code} 信号=${signal}`); 47 | if (worker.exitedAfterDisconnect === true) { 48 | console.log(`[0-${process.pid}]: 这是主线程主动断开的,无需重启。`); 49 | } else { 50 | console.log(`[0-${process.pid}]: 启动一个新的工作进程`); 51 | let worker = cluster.fork(); 52 | worker.send({ 53 | do: 'start', 54 | tip: `工人${worker.id}号,接替${worker.id}-${worker.process.pid}的工作吧`, 55 | data: worker.id, 56 | }); 57 | } 58 | }); 59 | await instance.main(); 60 | } 61 | 62 | // 主线程异常,全部退出 63 | module.exports.stop = function () { 64 | for (let id in cluster.workers) { 65 | let worker = cluster.workers[id]; 66 | worker.send({ 67 | do: 'stop', 68 | tip: `工厂倒闭了,工人${id}号,立刻下班,给你1秒钟消失` 69 | }); 70 | setTimeout(function() { 71 | console.log(`[0-${process.pid}]: 断开与[${worker.id}-${worker.process.pid}]的IPC管道`); 72 | worker.disconnect(); // 主动断开IPC管道 73 | }, 1000); 74 | } 75 | } -------------------------------------------------------------------------------- /batch/worker.js: -------------------------------------------------------------------------------- 1 | const http = require('http'); 2 | 3 | module.exports.start = async function (instance) { 4 | console.log(`工作进程${process.pid}开始`); 5 | process.on('message', async function (msg) { 6 | console.log(`[${process.id || '?'}-${process.pid}]: 收到消息:${msg.tip}`); 7 | switch (msg.do) { 8 | case 'start': 9 | process.id = msg.data; 10 | process.sleepCount = 0; 11 | process.send(`好的`); 12 | const server = http.createServer(function (req, res) { 13 | res.writeHead(200); 14 | res.end(`你好,我是 ${process.id}-${process.pid} !\n`); 15 | }); 16 | server.on('close', function () { 17 | console.log(`[${process.id}-${process.pid}]: 监听已停止`); 18 | }); 19 | server.listen(); 20 | process.server = server; 21 | break; 22 | case 'stop': 23 | console.log(`[${process.id}-${process.pid}]: 我停止监听,准备下班`); 24 | process.server.close(); 25 | break; 26 | case 'task': 27 | if (msg.data) { 28 | console.log(`[${process.id}-${process.pid}]: 开始任务 ${msg.data}`); 29 | await instance.execTask(msg.data); 30 | process.send(`任务完成了`); 31 | } else { 32 | process.sleepCount++; 33 | if (process.sleepCount > 3) { 34 | process.send('我要下班') 35 | } else { 36 | console.log(`[${process.id}-${process.pid}]: 收到空任务,休息一会儿,第${process.sleepCount}次休息`); 37 | setTimeout(function () { 38 | process.send(`我休息好了`); 39 | }, 1000 * 2); 40 | } 41 | } 42 | break; 43 | default: 44 | process.send(`你说啥?`); 45 | } 46 | }); 47 | } -------------------------------------------------------------------------------- /browser/cache.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const fs = require('fs'); 3 | 4 | const Conf = require('../conf'); 5 | const Comm = require('../comm'); 6 | 7 | const cacheDir = Conf.imp.cacheDir; 8 | 9 | Comm.mkDirs(cacheDir); 10 | 11 | module.exports.write = function (options, data) { 12 | const file = path.join(cacheDir, options.host, Comm.winName(options.path)); 13 | Comm.mkDirs(path.dirname(file)); 14 | fs.writeFile(file, data, 'utf8', function (err) { 15 | if (err) { 16 | console.error('write cache err', err); 17 | } 18 | }); 19 | }; 20 | 21 | module.exports.read = function (options) { 22 | const file = path.join(cacheDir, options.host, Comm.winName(options.path)); 23 | if (fs.existsSync(file)) { 24 | return fs.readFileSync(file, 'utf8'); 25 | } 26 | return ''; 27 | }; -------------------------------------------------------------------------------- /browser/cookie.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const fs = require('fs'); 3 | const Cookie = require('cookie'); 4 | 5 | const Conf = require('../conf'); 6 | const Comm = require('../comm'); 7 | 8 | const file = path.join(Conf.imp.cacheDir, '.cookie.json'); 9 | let obj = undefined; 10 | 11 | function loadCache() { 12 | if (fs.existsSync(file)) { 13 | try { 14 | const data = fs.readFileSync(file, 'utf8'); 15 | obj = JSON.parse(data) || {} 16 | } catch (err) { 17 | console.log(err) 18 | } 19 | } 20 | } 21 | 22 | module.exports.append = function(headers) { 23 | if (obj === undefined) { 24 | loadCache() 25 | } 26 | const cookies = []; 27 | if (headers['Cookie']) { 28 | cookies.push(headers['Cookie']); 29 | } 30 | if (headers['cookie']) { 31 | cookies.push(headers['cookie']); 32 | delete headers['cookie']; 33 | } 34 | for (let k in obj) { 35 | cookies.push(`${k}=${obj[k][k]}`); 36 | } 37 | headers['Cookie'] = cookies.join('; '); 38 | }; 39 | 40 | module.exports.remember = function(options, headers) { 41 | if (obj === undefined) { 42 | loadCache() 43 | } 44 | const cookies = headers['set-cookie'] || []; 45 | if (cookies.length > 0) { 46 | for (let i = 0; i < cookies.length; i++) { 47 | const cookie = Cookie.parse(cookies[i].trim()); 48 | obj[Object.keys(cookie)[i]] = cookie; 49 | } 50 | fs.writeFile(file, JSON.stringify(obj, null, 2), 'utf8', function(err) { 51 | if (err) { 52 | console.error('write cookie err', err) 53 | } 54 | }); 55 | } 56 | }; 57 | -------------------------------------------------------------------------------- /browser/index.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const URL = require('url'); 3 | const zlib = require('zlib'); 4 | const http = require('http'); 5 | const https = require('https'); 6 | const iconv = require('iconv-lite'); 7 | const querystring = require('querystring'); 8 | 9 | const ua = require('./ua'); 10 | const cache = require('./cache'); 11 | const Cookie = require('./cookie'); 12 | 13 | const hostname_charset = {}; 14 | 15 | module.exports.setCharset = function (hostname, charset) { 16 | hostname_charset[hostname] = charset; 17 | }; 18 | 19 | module.exports.GET = function (url, headers) { 20 | return request('GET', url, headers); 21 | }; 22 | 23 | module.exports.POST = function (url, headers, form) { 24 | return request('POST', url, headers, form); 25 | }; 26 | 27 | function request(method, url, headers, form) { 28 | console.log(`Browser ${method} ${url}`); 29 | return function (cb) { 30 | headers = headers || {}; 31 | form = form || {}; 32 | const postData = querystring.stringify(form); 33 | const options = URL.parse(url); 34 | options.method = method; 35 | options.headers = { 36 | 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8', 37 | 'Accept-Encoding': 'gzip, deflate', 38 | 'Accept-Language': 'zh-CN,zh;q=0.9,en;q=0.8,ja;q=0.7,zh-TW;q=0.6', 39 | 'Cache-Control': 'max-age=0', 40 | 'Connection': 'keep-alive', 41 | 'Upgrade-Insecure-Requests': '1', 42 | 'User-Agent': ua(), 43 | }; 44 | if (method === 'POST') { 45 | options.headers['Content-Type'] = 'application/x-www-form-urlencoded'; 46 | options.headers['Content-Length'] = postData.length; 47 | } 48 | for (let k in headers) { 49 | options.headers[k] = headers[k]; 50 | } 51 | const html = cache.read(options); 52 | if (html) { 53 | const ret = {code: 304, headers: options.headers, body: html}; 54 | cb(null, ret); 55 | return; 56 | } 57 | Cookie.append(options.headers); 58 | const agent = options.protocol === 'https:' ? https : http; 59 | const req = agent.request(options, (res) => { 60 | res.setTimeout(3000); 61 | const data = []; 62 | res.on('data', (chunk) => { 63 | data.push(chunk); 64 | }); 65 | res.on('end', () => { 66 | co(function* () { 67 | let buff; 68 | if (res.headers['content-encoding'] === 'gzip') { // Gzip supper 69 | buff = yield gzip(Buffer.concat(data)); 70 | } else { 71 | buff = Buffer.concat(data); 72 | } 73 | const charset = hostname_charset[options.hostname]; 74 | if (charset) { 75 | buff = iconv.decode(buff, charset); 76 | } 77 | const body = buff.toString(); 78 | if (body.length > 500) { 79 | cache.write(options, body); 80 | } 81 | Cookie.remember(options, res.headers); 82 | const ret = {code: res.statusCode, headers: res.headers, body: body}; 83 | cb(null, ret); 84 | }).catch((err) => { 85 | console.log('Browser err', err); 86 | const ret = {code: 500, headers: {}, body: ''}; 87 | cb(null, ret); 88 | }); 89 | }) 90 | }); 91 | req.setTimeout(3000); 92 | req.on('error', (err) => { 93 | console.log('Browser err', err); 94 | const ret = {code: 500, headers: {}, body: ''}; 95 | cb(null, ret); 96 | }); 97 | if (method !== 'GET') { 98 | req.write(postData) 99 | } 100 | req.end(); 101 | } 102 | } 103 | 104 | function gzip(data) { 105 | return function (cb) { 106 | zlib.gunzip(data, function (err, decoded) { 107 | cb(err, decoded); 108 | }); 109 | } 110 | } 111 | -------------------------------------------------------------------------------- /browser/test.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | 3 | const Browser = require('./index'); 4 | 5 | co(function* () { 6 | Browser.setCharset('www.caobi45.com', 'gb2312'); 7 | const ret = yield Browser.GET('http://www.caobi45.com/index.html'); 8 | console.log(ret); 9 | }).catch((err) => { 10 | console.error(err); 11 | }); -------------------------------------------------------------------------------- /browser/ua.js: -------------------------------------------------------------------------------- 1 | const UA = [ 2 | // Chrome 3 | "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.1 (KHTML, like Gecko) Chrome/22.0.1207.1 Safari/537.1", 4 | "Mozilla/5.0 (X11; CrOS i686 2268.111.0) AppleWebKit/536.11 (KHTML, like Gecko) Chrome/20.0.1132.57 Safari/536.11", 5 | "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.6 (KHTML, like Gecko) Chrome/20.0.1092.0 Safari/536.6", 6 | "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.6 (KHTML, like Gecko) Chrome/20.0.1090.0 Safari/536.6", 7 | "Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.1 (KHTML, like Gecko) Chrome/19.77.34.5 Safari/537.1", 8 | "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/536.5 (KHTML, like Gecko) Chrome/19.0.1084.9 Safari/536.5", 9 | "Mozilla/5.0 (Windows NT 6.0) AppleWebKit/536.5 (KHTML, like Gecko) Chrome/19.0.1084.36 Safari/536.5", 10 | "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1063.0 Safari/536.3", 11 | "Mozilla/5.0 (Windows NT 5.1) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1063.0 Safari/536.3", 12 | "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_8_0) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1063.0 Safari/536.3", 13 | "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1062.0 Safari/536.3", 14 | "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1062.0 Safari/536.3", 15 | "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.1 Safari/536.3", 16 | "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.1 Safari/536.3", 17 | "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.1 Safari/536.3", 18 | "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.0 Safari/536.3", 19 | "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/535.24 (KHTML, like Gecko) Chrome/19.0.1055.1 Safari/535.24", 20 | "Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/535.24 (KHTML, like Gecko) Chrome/19.0.1055.1 Safari/535.24", 21 | // Firefox 22 | "Mozilla/5.0 (Macintosh; U; Mac OS X Mach-O; en-US; rv:2.0a) Gecko/20040614 Firefox/3.0.0 ", 23 | "Mozilla/5.0 (Macintosh; U; PPC Mac OS X 10.5; en-US; rv:1.9.0.3) Gecko/2008092414 Firefox/3.0.3", 24 | "Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10.5; en-US; rv:1.9.1) Gecko/20090624 Firefox/3.5", 25 | "Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10.6; en-US; rv:1.9.2.14) Gecko/20110218 AlexaToolbar/alxf-2.0 Firefox/3.6.14", 26 | "Mozilla/5.0 (Macintosh; U; PPC Mac OS X 10.5; en-US; rv:1.9.2.15) Gecko/20110303 Firefox/3.6.15", 27 | "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.6; rv:2.0.1) Gecko/20100101 Firefox/4.0.1", 28 | // Opera 29 | "Opera/9.80 (Windows NT 6.1; U; en) Presto/2.8.131 Version/11.11", 30 | "Opera/9.80 (Android 2.3.4; Linux; Opera mobi/adr-1107051709; U; zh-cn) Presto/2.8.149 Version/11.10", 31 | // Safari 32 | "Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US) AppleWebKit/531.21.8 (KHTML, like Gecko) Version/4.0.4 Safari/531.21.10", 33 | "Mozilla/5.0 (Windows; U; Windows NT 5.2; en-US) AppleWebKit/533.17.8 (KHTML, like Gecko) Version/5.0.1 Safari/533.17.8", 34 | "Mozilla/5.0 (Windows; U; Windows NT 6.1; en-US) AppleWebKit/533.19.4 (KHTML, like Gecko) Version/5.0.2 Safari/533.18.5", 35 | // IE 36 | "Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.1; Trident/5.0", 37 | "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.0; Trident/4.0)", 38 | "Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 6.0)", 39 | "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1)", 40 | ] 41 | 42 | module.exports = function() { 43 | return UA[Math.floor(Math.random() * UA.length)]; 44 | } -------------------------------------------------------------------------------- /comm/index.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs'); 2 | const path = require('path'); 3 | const exec = require('child_process').exec; 4 | 5 | module.exports.mkDirs = function (dir) { 6 | if (!fs.existsSync(dir)) { 7 | fs.mkdirSync(dir, {recursive: true}); 8 | } 9 | }; 10 | 11 | module.exports.mp4TempFiles = function (dir) { 12 | const files = []; 13 | const temps = fs.existsSync(dir) ? fs.readdirSync(dir) : []; 14 | for (let i = 0; i < temps.length; i++) { 15 | const idx = temps[i].indexOf('.mp4'); 16 | if (idx !== -1) { 17 | files.push(temps[i].substring(0, idx) + '.mp4'); 18 | } 19 | } 20 | return files; 21 | }; 22 | 23 | module.exports.mp4Files = function (dir) { 24 | const files = []; 25 | const temps = fs.existsSync(dir) ? fs.readdirSync(dir) : []; 26 | for (let i = 0; i < temps.length; i++) { 27 | if (temps[i].endsWith('.mp4')) { 28 | files.push(temps[i]); 29 | } 30 | } 31 | return files; 32 | }; 33 | 34 | module.exports.readFileInt = function (file, def) { 35 | if (fs.existsSync(file)) { 36 | const data = fs.readFileSync(file, 'utf8') || 0; 37 | return parseInt(data.trim() || def); 38 | } 39 | return def; 40 | }; 41 | 42 | module.exports.writeFileVal = function (file, val) { 43 | if (fs.existsSync(file)) { 44 | fs.unlinkSync(file); 45 | } 46 | fs.writeFileSync(file, val); 47 | }; 48 | 49 | module.exports.winName = function (name) { 50 | return name.trim().replace(/[\/\\:*?"|]+?/gim, '_').trim(); 51 | }; 52 | 53 | module.exports.newName = function (id, name) { 54 | return `${('00000' + id).slice(-5)}_${module.exports.winName(name)}`.trim() + '.mp4'; 55 | }; 56 | 57 | module.exports.openVideo = function (conf, data) { 58 | return function (cb) { 59 | const src = path.join(conf.dlDir, path.basename(data.mp4)); 60 | const dst = fs.existsSync(src) ? src : path.join(conf.reDir, module.exports.newName(data.id, data.title)); 61 | exec('start "' + dst + '"', function (err, stdout, stderr) { 62 | cb(null); 63 | }); 64 | } 65 | }; 66 | 67 | module.exports.exec = function (command) { 68 | console.log(command); 69 | exec(`start cmd /c ${command}`, function(err, stdout, stderr) {}); 70 | }; 71 | -------------------------------------------------------------------------------- /conf.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | db: { 3 | dialect: 'sqlite', 4 | showSql: false, 5 | database: "91video", 6 | sqlite: { 7 | storage: '91video.db', 8 | }, 9 | mysql: { 10 | host: "127.0.0.1", 11 | port: 3306, 12 | user: "root", 13 | password: "123456", 14 | } 15 | }, 16 | imp: { 17 | cacheDir: 'F:/91Cache', 18 | }, 19 | exp: { 20 | dlDir: 'F:/91Download', // Thunder download dir 21 | reDir: 'F:/91Video', // rename output dir 22 | }, 23 | www: { 24 | port: 8080, 25 | table: 'caobi45', // the database table name 26 | mode: 'online', // load internet url 27 | // mode: 'offline', // open local disk file 28 | } 29 | }; -------------------------------------------------------------------------------- /db/index.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const fs = require('fs'); 3 | const path = require('path'); 4 | const Sequelize = require('sequelize'); 5 | 6 | const conf = require('../conf').db; 7 | const model = require('./model'); 8 | 9 | const DB = {}; 10 | 11 | DB.useMySQL = conf.dialect === 'mysql'; 12 | 13 | DB.sequelize = undefined; 14 | 15 | function init() { 16 | DB.sequelize = DB.useMySQL ? new Sequelize(conf.database, conf.mysql.user, conf.mysql.password, { 17 | dialect: 'mysql', 18 | timezone: '+08:00', 19 | dialectOptions: {charset: "utf8", }, 20 | host: conf.mysql.host, 21 | port: conf.mysql.port, 22 | pool: {min: 0, max: 5, acquire: 30000, idle: 10000, }, 23 | logging: conf.showSql === false ? false : console.log, 24 | }) : new Sequelize(conf.database, null, null, { 25 | dialect: 'sqlite', 26 | storage: path.join(__dirname, '../' + conf.sqlite.storage), 27 | logging: conf.showSql === false ? false : console.log, 28 | }); 29 | } 30 | 31 | DB.query = function (sql, pms) { 32 | return DB.sequelize.query(sql, { 33 | raw: true, 34 | replacements: pms, 35 | type: DB.sequelize.QueryTypes.SELECT, 36 | }); 37 | }; 38 | 39 | DB.update = function (sql, pms) { 40 | return DB.sequelize.query(sql, { 41 | replacements: pms, 42 | }); 43 | }; 44 | 45 | DB.use = function (table) { 46 | if (DB.sequelize === undefined) { 47 | init(); 48 | } 49 | return function (cb) { 50 | co(function* () { 51 | DB.table = table; 52 | DB.Model = DB.sequelize.define(table, model, { 53 | tableName: table, 54 | timestamps: false, 55 | charset: 'utf8', 56 | }); 57 | DB.Model.replace = function (obj) { 58 | return DB.update("REPLACE INTO `" + table + "`(id,url,title,mp4,jpg) VALUES(?,?,?,?,?)", 59 | [obj.id, obj.url, obj.title, obj.mp4, obj.jpg]); 60 | }; 61 | yield DB.Model.sync({force: false, alter: false}); 62 | cb(null, DB.Model); 63 | }).catch((err) => { 64 | cb(err); 65 | }); 66 | } 67 | }; 68 | 69 | module.exports = DB; 70 | -------------------------------------------------------------------------------- /db/model.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize'); 2 | 3 | module.exports = { 4 | id: { 5 | type: Sequelize.INTEGER, 6 | allowNull: false, 7 | primaryKey: true, 8 | }, 9 | url: { 10 | type: Sequelize.STRING(128), 11 | allowNull: true, 12 | comment: '地址', 13 | }, 14 | title: { 15 | type: Sequelize.STRING(128), 16 | allowNull: true, 17 | comment: '标题', 18 | }, 19 | mp4: { 20 | type: Sequelize.STRING(128), 21 | allowNull: true, 22 | comment: '视频', 23 | }, 24 | jpg: { 25 | type: Sequelize.STRING(128), 26 | allowNull: true, 27 | comment: '图片', 28 | }, 29 | saved: { 30 | type: Sequelize.INTEGER(1), 31 | allowNull: true, 32 | defaultValue: 0, 33 | comment: '是否已硬存', 34 | }, 35 | }; -------------------------------------------------------------------------------- /export/fetch.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const fs = require('fs'); 3 | 4 | const DB = require('../db'); 5 | const Conf = require('../conf'); 6 | const Comm = require('../comm'); 7 | 8 | const dlDir = Conf.exp.dlDir; 9 | const skip = 0; 10 | const rows = 1000; 11 | const page = 1; 12 | const pageFile = 'fetch.page'; 13 | const outFile = 'fetch.txt'; 14 | 15 | co(function*() { 16 | const model = yield DB.use('caobi45'); 17 | // const page = Comm.readFileInt(pageFile, 1); 18 | const exFiles = Comm.mp4TempFiles(dlDir); 19 | let sql = `SELECT id, mp4, SUBSTR(mp4,LENGTH(mp4)-38) AS fname FROM ${model.tableName} WHERE saved=0 and id>${skip}`; 20 | if (exFiles.length > 0) { 21 | sql += ` AND fname NOT IN ('${exFiles.join(`','`)}')`; 22 | } 23 | sql += ` GROUP BY fname`; 24 | sql += ` ORDER BY id LIMIT ${(page - 1) * rows}, ${rows}`; 25 | let pms = []; 26 | const data = yield DB.query(sql, pms); 27 | console.log(`Fetch page ${page} count ${data.length}`); 28 | if (data.length > 0) { 29 | // Comm.writeFileVal(pageFile, page + 1); 30 | if (fs.existsSync(outFile)) { 31 | fs.unlinkSync(outFile); 32 | } 33 | for (let i = 0; i < data.length; i++) { 34 | fs.appendFileSync(outFile, `${data[i].mp4}#id=${('00000' + data[i].id).slice(-5)}\r\n`); 35 | } 36 | } 37 | console.log(`Complete.`) 38 | }).catch((err) => { 39 | console.error(err); 40 | }); 41 | -------------------------------------------------------------------------------- /export/rename.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const fs = require('fs'); 3 | const path = require('path'); 4 | const Sequelize = require('sequelize'); 5 | const Op = Sequelize.Op; 6 | 7 | const DB = require('../db'); 8 | const Conf = require('../conf'); 9 | const Comm = require('../comm'); 10 | 11 | const dlDir = Conf.exp.dlDir; 12 | const reDir = Conf.exp.reDir; 13 | const logFile = 'rename.log'; 14 | const errFile = 'rename.err'; 15 | 16 | // execute, where Thunder downloaded a batch. 17 | 18 | co(function* () { 19 | yield DB.use('caobi45'); 20 | Comm.mkDirs(reDir); 21 | const videos = Comm.mp4Files(dlDir); 22 | console.log(`Rename ${videos.length} files...`); 23 | for (let i = 0; i < videos.length; i++) { 24 | const file = videos[i]; 25 | const src = path.join(dlDir, file); 26 | if (fs.statSync(src).size < 20480) { 27 | errLog(`Warning invalid file! name=${file}`); 28 | fs.unlinkSync(src); 29 | continue; 30 | } 31 | const data = yield DB.Model.findAll({ 32 | where: { 33 | mp4: {[Op.endsWith]: file}, 34 | }, 35 | raw: true, 36 | }); 37 | if (data.length === 0) { 38 | errLog(`Warning record not found! name=${file}`); 39 | } else { 40 | if (data.length > 1) { 41 | errLog(`Warning ${data.length} record! name=${file}`); 42 | data.map(function (x) { 43 | errLog(`\t${x.id} ${x.title}`); 44 | }); 45 | } 46 | let name = data[0].title.trim(); 47 | if (name.indexOf('http:/') !== -1) { 48 | name = name.split('http:/')[0] + Date.now().toString(); 49 | } 50 | const newName = Comm.newName(data[0].id, name); 51 | const dst = path.join(reDir, newName); 52 | if (fs.existsSync(dst)) { 53 | errLog(`Warning dst exists! name=${file}`); 54 | errLog(`\tdst=${dst}`); 55 | fs.renameSync(src, path.join(dlDir, newName)); 56 | continue; 57 | } 58 | try { 59 | fs.renameSync(src, dst); 60 | // fs.appendFileSync(logFile, `${file} => ${newName}\r\n`, 'utf8'); 61 | const count = yield DB.Model.update({saved: 1}, {where: {mp4: {[Op.endsWith]: file}}}); 62 | console.log(`Rename ${count} record by ${file}`); 63 | } catch (e) { 64 | errLog(`Warning rename failed! ${e.message}`); 65 | errLog(`\tsrc=${src}`); 66 | errLog(`\tdst=${dst}`); 67 | } 68 | } 69 | } 70 | console.log(`Complete.`) 71 | }).catch((err) => { 72 | console.error(err); 73 | }); 74 | 75 | function errLog(msg) { 76 | console.log(msg); 77 | // fs.appendFileSync(errFile, msg + '\r\n', 'utf8'); 78 | } 79 | -------------------------------------------------------------------------------- /export/saved.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const fs = require('fs'); 3 | 4 | const DB = require('../db'); 5 | const Conf = require('../conf'); 6 | 7 | // before use fetch.js, update saved. 8 | 9 | co(function*() { 10 | const tb = 'caobi45'; 11 | yield DB.use(tb); 12 | const ids = []; 13 | const files = fs.readdirSync(Conf.exp.reDir); 14 | for (i in files) { 15 | ids.push(files[i].split('_')[0]); 16 | } 17 | // yield DB.update(`update ${tb} set saved=0`, []); 18 | // yield DB.update(`update ${tb} set saved=1 where id in (${ids.join(',')})`, []); 19 | // yield DB.update(`update ${tb} set saved=1 where SUBSTR(mp4, LENGTH(mp4)-38) in (select SUBSTR(mp4, LENGTH(mp4)-38) tmp from ${tb} where saved=1)`, []); 20 | const data1 = yield DB.query(`select count(*) num from ${tb} where saved=1`, []) 21 | const data2 = yield DB.query(`select count(*) num from ${tb} where saved=0`, []) 22 | console.log(`${files.length} files downloaded, ${data1[0].num} rows saved, and ${data2[0].num} not.`); 23 | }).catch(function(err) { 24 | console.log(err); 25 | }) -------------------------------------------------------------------------------- /import/1_caobi45.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const cheerio = require('cheerio'); 3 | 4 | const DB = require('../db'); 5 | const Browser = require('../browser'); 6 | 7 | const mode = 0; 8 | 9 | co(function* () { 10 | yield DB.use('caobi45'); 11 | Browser.setCharset('www.caobi45.com', 'gb2312'); 12 | if (mode >= 0) { 13 | /* climb play page, get videos */ 14 | for (let id = 1688; id < 50000; id++) { 15 | const url = `http://www.caobi45.com/player/index${id}.html`; 16 | const res = yield Browser.GET(url); 17 | const $ = cheerio.load(res.body); 18 | const result = /f:'(http[^\']+.mp4)',/.exec(res.body) || ['', '']; 19 | if (result[1]) { 20 | const title = $('title').text().trim(); 21 | const video = {id: id, url: url, title: title.substring(5), mp4: result[1]}; 22 | console.debug('Video', JSON.stringify(video)); 23 | yield DB.Model.replace(video); 24 | } 25 | } 26 | } 27 | if (mode <= 0) { 28 | /* climb list page, update preview jpg */ 29 | const list = []; 30 | add(list, 'http://www.caobi45.com/index.html'); 31 | add(list, 'http://www.caobi45.com/list/index27.html', 117); 32 | add(list, 'http://www.caobi45.com/list/index28.html', 279); 33 | add(list, 'http://www.caobi45.com/list/index29.html', 27); 34 | add(list, 'http://www.caobi45.com/list/index34.html', 111); 35 | add(list, 'http://www.caobi45.com/list/index54.html', 147); 36 | add(list, 'http://www.caobi45.com/list/index55.html', 444); 37 | add(list, 'http://www.caobi45.com/list/index56.html', 38); 38 | add(list, 'http://www.caobi45.com/list/index57.html', 1354); 39 | for (let i = 0; i < list.length; i++) { 40 | const res = yield Browser.GET(list[i]); 41 | const $ = cheerio.load(res.body); 42 | const array = $('a.pic').map(function () { 43 | const href = $(this).attr('href'); 44 | const id = href.substring(11, href.length - 5); 45 | const jpg = $(this).find('img').attr('src'); 46 | return {id: id, jpg: jpg}; 47 | }); 48 | for (let j = 0; j < array.length; j++) { 49 | const video = array[j]; 50 | console.debug('Image', JSON.stringify(video)); 51 | if (video.id && video.jpg) { 52 | yield DB.Model.update({jpg: video.jpg}, {where: {id: video.id}}); 53 | } 54 | } 55 | } 56 | } 57 | }).catch((err) => { 58 | console.error(err); 59 | }); 60 | 61 | function add(list, url, count) { 62 | for (let i = 0; i < count + 10; i++) { 63 | list.push(i === 0 ? url : url.replace('.html', '') + '_' + i + '.html'); 64 | } 65 | } -------------------------------------------------------------------------------- /import/2_668wy.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const cheerio = require('cheerio'); 3 | 4 | const DB = require('../db'); 5 | const Browser = require('../browser'); 6 | 7 | co(function* () { 8 | // 668wy.com or 0154s.com 9 | yield DB.use('668wy'); 10 | for (let id = 1; id < 5530; id++) { 11 | const url = `http://668wy.com/?m=vod-play-id-${id}-src-1-num-1.html`; 12 | const res = yield Browser.GET(url, {Cookie: 'PHPSESSID=uign95nav471a0c3ujcn97qi30;'}); 13 | const $ = cheerio.load(res.body); 14 | const result = /unescape\('([\S\s]+)'\);/.exec(res.body) || ['', '']; 15 | if (result[1]) { 16 | const title = $('.title_all').text().trim(); 17 | const video = {id: id, url: url, title: title, mp4: unescape(result[1])}; 18 | console.debug('Video', JSON.stringify(video)); 19 | yield DB.Model.replace(video); 20 | } 21 | } 22 | }).catch((err) => { 23 | console.error(err); 24 | }); 25 | -------------------------------------------------------------------------------- /import/3_bka8.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const cheerio = require('cheerio'); 3 | 4 | const DB = require('../db'); 5 | const Browser = require('../browser'); 6 | 7 | co(function* () { 8 | // www.bka8.com or www.ud35.com 9 | yield DB.use('bka8'); 10 | for (let id = 2273; id < 8298; id++) { 11 | const url = `http://www.bka8.com/?m=vod-play-id-${id}-src-1-num-1.html`; 12 | const res = yield Browser.GET(url, {Cookie: 'PHPSESSID=1gendmsq3r3p0ohi8osg1mefi5;'}); 13 | const $ = cheerio.load(res.body); 14 | const result = /unescape\('([\S\s]+)'\);/.exec(res.body) || ['', '']; 15 | if (result[1]) { 16 | const title = $('.position').find('a').last().text().trim(); 17 | const video = {id: id, url: url, title: title, mp4: unescape(result[1])}; 18 | console.debug('Video', JSON.stringify(video)); 19 | yield DB.Model.replace(video); 20 | } 21 | } 22 | }).catch((err) => { 23 | console.error(err); 24 | }); 25 | -------------------------------------------------------------------------------- /import/4_14xav.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const cheerio = require('cheerio'); 3 | 4 | const DB = require('../db'); 5 | const Browser = require('../browser'); 6 | 7 | co(function* () { 8 | // 14xav.com or 10xav.com 9 | yield DB.use('14xav'); 10 | Browser.setCharset('10xav.com', 'gb2312'); 11 | for (let id = 3393; id < 50000; id++) { 12 | const url = `http://10xav.com/player/index${id}.html`; 13 | const res = yield Browser.GET(url); 14 | const $ = cheerio.load(res.body); 15 | const result = /f:'(http[^\']+.mp4)',/.exec(res.body) || ['', '']; 16 | if (result[1]) { 17 | const title = $('title').text().trim(); 18 | const video = {id: id, url: url, title: title.substring(5), mp4: result[1]}; 19 | console.debug('Video', JSON.stringify(video)); 20 | yield DB.Model.replace(video); 21 | } 22 | } 23 | }).catch((err) => { 24 | console.error(err); 25 | }); 26 | -------------------------------------------------------------------------------- /import/5_331sss.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const cheerio = require('cheerio'); 3 | 4 | const DB = require('../db'); 5 | const Browser = require('../browser'); 6 | 7 | co(function* () { 8 | // 331sss.com or 225sss.com or 661sss.com 9 | yield DB.use('331sss'); 10 | Browser.setCharset('331sss.com', 'gb2312'); 11 | for (let id = 1644; id < 50000; id++) { 12 | /*const url = `http://14xav.com/player/index${id}.html`; 13 | const res = yield Browser.GET(url); 14 | const $ = cheerio.load(res.body); 15 | const result = /f:'(http[^\']+.mp4)',/.exec(res.body) || ['', '']; 16 | if (result[1]) { 17 | const title = $('title').text().trim(); 18 | const video = {id: id, url: url, title: title.substring(5), mp4: result[1]}; 19 | console.debug('Video', JSON.stringify(video)); 20 | yield DB.Model.replace(video); 21 | }*/ 22 | } 23 | }).catch((err) => { 24 | console.error(err); 25 | }); 26 | -------------------------------------------------------------------------------- /import/6_8x3a.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const cheerio = require('cheerio'); 3 | 4 | const DB = require('../db'); 5 | const Comm = require('../comm'); 6 | const Browser = require('../browser'); 7 | 8 | const start = 1; // 开始数 9 | const end = 1244; // 结束数 10 | const threads = 16; // 开多少个窗口 11 | let N = process.argv[2]; // 当前窗口序号 12 | if (N === undefined) { 13 | for (let n = 0; n < threads; n++) { 14 | Comm.exec(`node ${process.argv[1]} ${n}`); 15 | } 16 | setTimeout(function () { 17 | process.exit(); 18 | }, 500); 19 | return; 20 | } else { 21 | console.log(`#${process.pid} ${JSON.stringify(process.argv)}`); 22 | N = parseInt(N) 23 | } 24 | 25 | co(function*() { 26 | yield DB.use('8x3a'); 27 | for (let page = start; page <= end; page += threads) { 28 | const res = yield Browser.GET(`https://8dni.com/html/category/video/page_${page + N}.html`); 29 | const $ = cheerio.load(res.body); 30 | const arr = []; 31 | $('.l_b li').each(function() { 32 | const path = $(this).find('.t_p a').attr('href'); 33 | if (path && path !== '/') { 34 | const url = 'https://8dni.com' + path; 35 | const id = path.substring(6, path.length - 1); 36 | const title = $(this).find('.w_z h3').text(); 37 | const jpg = $(this).find('.t_p a img').attr('data-original'); 38 | const video = { id: id, url: url, title: title, jpg: jpg }; 39 | console.log(video); 40 | arr.push(video) 41 | } 42 | }); 43 | for (let i in arr) { 44 | const video = arr[i]; 45 | const data = yield DB.Model.findOne({where: {id: video.id}}); 46 | if (!data || !data.mp4) { 47 | const res = yield Browser.GET(video.url); 48 | const $ = cheerio.load(res.body); 49 | const mp4 = $('.sp_kj .x_z a').first().attr('href'); 50 | if (mp4 && mp4.endsWith('.mp4')) { 51 | video.mp4 = mp4; 52 | console.log(video); 53 | yield DB.Model.replace(video) 54 | } 55 | } 56 | } 57 | } 58 | }).catch((err) => { 59 | console.error(err) 60 | }); -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const http = require('http'); 3 | const path = require('path'); 4 | const express = require('express'); 5 | 6 | const DB = require('./db'); 7 | const Conf = require('./conf'); 8 | const Comm = require('./comm'); 9 | 10 | const app = express(); 11 | app.set('views', path.join(__dirname, 'www')); 12 | app.set('view engine', 'ejs'); 13 | app.use(express.json()); 14 | app.use(express.urlencoded({extended: false})); 15 | app.use(express.static(path.join(__dirname, 'static'))); 16 | 17 | app.get('/', (req, res, next) => { 18 | co(function* () { 19 | const id = req.query.id; 20 | if (id > 0) { 21 | const data = yield DB.Model.findOne({where: {id: id}}); 22 | if (Conf.www.mode === 'offline') { 23 | const flag = yield Comm.openVideo(Conf.exp, data); 24 | res.end(flag ? 'Play failure.' : '') 25 | } else { 26 | res.render('index', {data: data}); 27 | } 28 | } else { 29 | const page = parseInt(req.query.page) || 1; 30 | const rows = parseInt(req.query.rows) || 48; 31 | const data = yield DB.Model.findAndCountAll({ 32 | where: Conf.www.mode === 'offline' ? {saved: 1} : {}, 33 | order: ['id'], 34 | limit: rows, 35 | offset: (page - 1) * rows, 36 | }); 37 | res.render('index', {page: page, rows: rows, data: data, len: data.rows.length}); 38 | } 39 | }).catch((err) => { 40 | next(err); 41 | }); 42 | }); 43 | 44 | app.use((req, res, next) => { 45 | res.status(404).send('404 is Fuck') 46 | }); 47 | app.use((err, req, res, next) => { 48 | console.log(err); 49 | res.status(500).send('500 Server Error') 50 | }); 51 | const server = http.createServer(app); 52 | server.on('error', (err) => { 53 | console.log(err); 54 | }); 55 | server.on('listening', () => { 56 | co(function* () { 57 | yield DB.use(Conf.www.table); 58 | console.log(`Web start http://localhost:${server.address().port}/`); 59 | }).catch((err) => { 60 | console.log(err); 61 | }); 62 | }); 63 | server.listen(Conf.www.port); -------------------------------------------------------------------------------- /others/move.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs'); 2 | const path = require('path'); 3 | 4 | (async () => { 5 | const inDir = '/Users/liuzy/MeiZi'; 6 | const outDir = '/Volumes/Macintosh SD/MeiZi'; 7 | const dirs = fs.readdirSync(inDir); 8 | for (let i = 0; i < dirs.length; i++) { 9 | const dir = path.join(inDir, dirs[i]); 10 | if (dirs[i].indexOf('-') === -1) { 11 | continue; 12 | } 13 | const id = dirs[i].split('-')[0]; 14 | const newId = (Array(5).join('0') + id).slice(-5); 15 | const files = fs.readdirSync(dir); 16 | for (let j = 0; j < files.length; j++) { 17 | const file = path.join(dir, files[j]); 18 | const idx = (Array(3).join('0') + (j+1)).slice(-3) 19 | const newName = `${newId}-${idx}${path.extname(file)}`; 20 | const newFile = path.join(outDir, newName); 21 | console.log(file, '=>', newFile); 22 | fs.copyFileSync(file, newFile); 23 | } 24 | } 25 | })().catch(err => console.log(err)); -------------------------------------------------------------------------------- /others/reext.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs'); 2 | const path = require('path'); 3 | 4 | const Conf = require('../conf'); 5 | 6 | const what = '.tmp'; 7 | 8 | const files = fs.readdirSync(Conf.exp.dlDir); 9 | for (let i = 0; i < files.length; i++) { 10 | const file = files[i]; 11 | if (file.endsWith(what)) { 12 | const src = path.join(Conf.exp.dlDir, file); 13 | const dst = path.join(Conf.exp.dlDir, file.replace(what, '.mp4')); 14 | fs.renameSync(src, dst); 15 | } 16 | } -------------------------------------------------------------------------------- /others/reid.js: -------------------------------------------------------------------------------- 1 | const co = require('co'); 2 | const Sequelize = require('sequelize'); 3 | 4 | co(function* () { 5 | const sequelize = new Sequelize('test', 'root', '111111', { 6 | dialect: 'mysql', 7 | logging: false, 8 | timezone: '+08:00', 9 | dialectOptions: {charset: "utf8",}, 10 | pool: {min: 0, max: 5, acquire: 30000, idle: 10000}, 11 | }); 12 | const video = sequelize.define('video', { 13 | id: {type: Sequelize.INTEGER(11), autoIncrement: true, primaryKey: true}, 14 | title: Sequelize.STRING(128), 15 | mp4: Sequelize.STRING(1024), 16 | saved: Sequelize.INTEGER(1), 17 | }, { 18 | tableName: 'video', 19 | timestamps: false, 20 | charset: 'utf8', 21 | }); 22 | yield video.sync({force: true}); 23 | for (let i = 0; i < 17999; i++) { 24 | const sql = `SELECT * FROM videos order by SUBSTR(mp4, 1, LENGTH(mp4) - LOCATE('/', REVERSE(mp4))+1), \`name\` LIMIT ${i}, 1`; 25 | const data = yield sequelize.query(sql, { 26 | type: sequelize.QueryTypes.SELECT, 27 | }); 28 | if (data.length === 0) { 29 | break; 30 | } 31 | yield sequelize.query(`insert into video(title,mp4,saved) value(?,?,?)`, { 32 | replacements: [data[0].name, data[0].mp4, data[0].saved], 33 | }); 34 | } 35 | }).catch((err) => { 36 | console.log(err); 37 | }); -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "91video", 3 | "version": "1.0.0", 4 | "main": "index.js", 5 | "scripts": { 6 | "start": "node index.js" 7 | }, 8 | "keywords": [ 9 | "spider", 10 | "video", 11 | "climax" 12 | ], 13 | "author": "liuzy", 14 | "license": "ISC", 15 | "dependencies": { 16 | "cheerio": "^1.0.0-rc.3", 17 | "co": "^4.6.0", 18 | "ejs": "^2.7.1", 19 | "express": "^4.17.1", 20 | "iconv-lite": "^0.5.0", 21 | "mysql2": "^1.7.0", 22 | "node-fetch": "^2.6.1", 23 | "sequelize": "^5.18.4", 24 | "sqlite3": "^4.1.0" 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /runing.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/liuzy88/91video/4f2ab77b641fffbf104807832b1340cf57788a5f/runing.png -------------------------------------------------------------------------------- /static/ckplayer/ckplayer.swf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/liuzy88/91video/4f2ab77b641fffbf104807832b1340cf57788a5f/static/ckplayer/ckplayer.swf -------------------------------------------------------------------------------- /static/ckplayer/ckplayer.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | true 6 | 30 7 | 100 8 | true 9 | 0 10 | true 11 | false 12 | true 13 | true 14 | 200 15 | 0 16 | true 17 | true 18 | 200 19 | 20 | true 21 | true 22 | true 23 | true 24 | true 25 | 26 | 10 27 | 0.1 28 | 1 29 | true 30 | false 31 | 32 | false 33 | false 34 | true 35 | true 36 | 37 | 38 | false 39 | 2 40 | start 41 | 42 | false 43 | 1 44 | false 45 | true 46 | 47 | 53 | 54 | 30 55 | , 56 | 57 | adPlay,adPause,playOrPause,videoPlay,videoPause,videoMute,videoEscMute,videoClear,changeVolume,fastBack,fastNext,videoSeek,newVideo,getMetaDate,videoRotation,videoBrightness,videoContrast,videoSaturation,videoHue,videoZoom,videoProportion,videoError,addListener,removeListener,addElement,getElement,deleteElement,elementShow,animate,animateResume,animatePause,deleteAnimate,changeConfig,getConfig,openUrl,fullScreen,quitFullScreen,switchFull,screenshot,custom,changeControlBarShow,getCurrentSrc,changeDefinition,changeSubtitles 58 | 59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 108 | 109 | 110 | -------------------------------------------------------------------------------- /static/ckplayer/hls/LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2017 Dailymotion (http://www.dailymotion.com) 2 | 3 | Licensed under the Apache License, Version 2.0 (the "License"); 4 | you may not use this file except in compliance with the License. 5 | You may obtain a copy of the License at 6 | 7 | http://www.apache.org/licenses/LICENSE-2.0 8 | 9 | Unless required by applicable law or agreed to in writing, software 10 | distributed under the License is distributed on an "AS IS" BASIS, 11 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 | See the License for the specific language governing permissions and 13 | limitations under the License. 14 | 15 | src/remux/mp4-generator.js and src/demux/exp-golomb.js implementation in this project 16 | are derived from the HLS library for video.js (https://github.com/videojs/videojs-contrib-hls) 17 | 18 | That work is also covered by the Apache 2 License, following copyright: 19 | Copyright (c) 2013-2015 Brightcove 20 | 21 | 22 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 23 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 24 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 25 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 26 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 27 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 28 | THE SOFTWARE. 29 | -------------------------------------------------------------------------------- /static/ckplayer/language.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | [$second] 4 | [$second] 5 | 6 | 点击播放 7 | 暂停播放 8 | 静音 9 | 恢复音量 10 | 全屏 11 | 退出全屏 12 | 上一集 13 | 下一集 14 | 点击选择清晰度 15 | 选择字幕 16 | 17 | 18 | 音量:[$volume]% 19 | 20 | [$percentage]% 21 | 22 | [$timeh]:[$timei]:[$times] 23 | 24 | 25 | [$timeh]:[$timei]:[$times] 26 | 27 | 28 | 直播中 [$liveTimeY]-[$liveTimem]-[$liveTimed] [$liveTimeh]:[$liveTimei]:[$liveTimes] 29 | 30 | 31 | 流畅 32 | 低清 33 | 标清 34 | 高清 35 | 超清 36 | 蓝光 37 | 未知 38 | 39 | 40 | 视频地址不存在 41 | 加载失败 42 | 视频格式错误 43 | 44 | 自动 45 | 默认 46 | -------------------------------------------------------------------------------- /static/ckplayer/style.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 0x000000 6 | 7 | 0.5 8 | 3 9 | center 10 | middle 11 | 0 12 | 0 13 | 0 14 | 0 15 | 16 | 17 | 18 | left 19 | bottom 20 | 100% 21 | 38 22 | 0 23 | -38 24 | 25 | 26 | all 27 | 1000 28 | alpha 29 | 0.8 30 | all 31 | 32 |  33 |  34 |  35 | 100% 36 | left 37 | bottom 38 | 0 39 | -2 40 | 41 | 42 | 43 | 44 | 0x000000 45 | 46 | 0.6 47 | 48 | 49 | 121 | 122 | 123 | 124 | 0x333333 125 | 0x0787FF 126 | 0.8 127 | 0 128 | 0x333333 129 | 6 130 | 70 131 | 27 132 | center 133 | 5 134 | 14 135 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 136 | 0xFFFFFF 137 | 0xFFFFFF 138 | false 139 | 1 140 | 141 | 142 | 0x333333 143 | 0.8 144 | 0 145 | 0x333333 146 | 10 147 | 15 148 | 10 149 | 15 150 | 10 151 | 15 152 | 10 153 | 0x333333 154 | 0 155 | 0xFFFFFF 156 | 0.8 157 | 0 158 | 0 159 | 160 | 178 | 179 | 0x005CB2 180 | 0x0787FF 181 | 0.8 182 | 0 183 | 0x333333 184 | 6 185 | 70 186 | 27 187 | center 188 | 5 189 | 14 190 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 191 | 0xFFFFFF 192 | 0xFFFFFF 193 | false 194 | 1 195 | 196 | 197 | 5 198 | 5 199 | 5 200 | 5 201 | 0xFFFFFF 202 | 0 203 | 1 204 | 205 | 206 | right 207 | top 208 | -119 209 | 6 210 | 211 | 212 | right 213 | bottom 214 | -130 215 | -35 216 | true 217 | 218 | click 219 | 220 | 221 | 222 | 223 | 0x333333 224 | 0x0787FF 225 | 0.8 226 | 0 227 | 0x333333 228 | 6 229 | 70 230 | 27 231 | center 232 | 5 233 | 14 234 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 235 | 0xFFFFFF 236 | 0xFFFFFF 237 | false 238 | 1 239 | 240 | 241 | 0x333333 242 | 0.8 243 | 0 244 | 0x333333 245 | 10 246 | 15 247 | 10 248 | 15 249 | 10 250 | 15 251 | 10 252 | 0x333333 253 | 0 254 | 0xFFFFFF 255 | 0.8 256 | 0 257 | 0 258 | 259 | 277 | 278 | 0x005CB2 279 | 0x0787FF 280 | 0.8 281 | 0 282 | 0x333333 283 | 6 284 | 70 285 | 27 286 | center 287 | 5 288 | 14 289 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 290 | 0xFFFFFF 291 | 0xFFFFFF 292 | false 293 | 1 294 | 295 | 296 | 5 297 | 5 298 | 5 299 | 5 300 | 0xFFFFFF 301 | 0 302 | 1 303 | 304 | 305 | right 306 | top 307 | -119 308 | 6 309 | 310 | 311 | right 312 | bottom 313 | -130 314 | -35 315 | true 316 | 317 | click 318 | 319 | 320 | 321 |  322 |  323 | right 324 | top 325 | -200 326 | 14 327 | 331 | 332 | 333 | 334 | 335 | 336 |  337 |  338 |  339 | 100% 340 | left 341 | top 342 | 0 343 | -9 344 | 345 | 346 | 347 |  348 |  349 |  350 | 100% 351 | left 352 | top 353 | 0 354 | -1 355 | 356 | 357 | 361 | 0.3 362 | 0.3 363 | 0.3 364 | 0.3 365 | 366 | 367 | 389 | 390 | 391 | [$timeh]:[$timei]:[$times] / [$durationh]:[$durationi]:[$durations] 392 | left 393 | top 394 | 45 395 | 7 396 | 14 397 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 398 | 0xFFFFFF 399 | 0.5 400 | false 401 | 402 | 403 | [$liveLanguage] 404 | left 405 | top 406 | 45 407 | 7 408 | 14 409 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 410 | 0xFFFFFF 411 | 1 412 | false 413 | 414 | 415 | 416 | 417 | 439 | 440 | 441 |  442 | right 443 | top 444 | -130 445 | 0 446 | 447 | 448 | 449 | 450 | 458 | 459 | 460 | 461 | 462 | 463 |  464 |  465 | center 466 | middle 467 | -40 468 | -40 469 | actionScript->videoPlay 470 | 471 | 472 | 473 |  474 | center 475 | middle 476 | -30 477 | -30 478 | 479 | center 480 | middle 481 | -30 482 | -12 483 | 14 484 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 485 | 0xFFFFFF 486 | 1 487 | false 488 | 60 489 | center 490 | 491 | 492 | 493 | 494 | 496 | 497 | 498 | 499 | 500 | 501 | 502 | 503 | 14 504 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 505 | 0xFFFFFF 506 | 1 507 | false 508 | 120 509 | center 510 | center 511 | middle 512 | -60 513 | 30 514 | 515 | 516 | 517 | 518 | 519 | 0x000000 520 | 1 521 | 522 | 523 | 524 | 30 525 | 0 526 | 30 527 | 0 528 | center 529 | middle 530 | 531 | 532 | 533 | 0xFF0000 534 |  535 | 1 536 | 0 537 | 35 538 | 90 539 | right 540 | top 541 | -100 542 | 5 543 | 544 | 545 | 546 | 16 547 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 548 | 0xFF0000 549 | 1 550 | false 551 | 25 552 | center 553 | right 554 | top 555 | -59 556 | 10 557 | 558 | 559 | 560 |  561 |  562 | right 563 | top 564 | -138 565 | 5 566 | actionScript->adMute 567 | 568 | 569 | 570 |  571 |  572 | right 573 | top 574 | -138 575 | 5 576 | actionScript->escAdMute 577 | 578 | 579 | 580 |  581 |  582 | right 583 | top 584 | -230 585 | 5 586 | javaScript->adjump 587 | 588 | 589 | 590 | 0xFF0000 591 |  592 | 1 593 | 0 594 | 35 595 | 150 596 | right 597 | top 598 | -290 599 | 5 600 | 601 | 602 | 603 | 14 604 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 605 | 0xFFFFFF 606 | 1 607 | false 608 | 25 609 | right 610 | right 611 | top 612 | -278 613 | 10 614 | 615 | 616 | 617 |  618 |  619 | right 620 | bottom 621 | -100 622 | -40 623 | actionScript->openAdLink 624 | 625 | 626 | 627 |  628 |  629 | right 630 | top 631 | -10 632 | -10 633 | actionScript->closePauseAd 634 | 635 | 636 | 637 |  638 |  639 | right 640 | top 641 | -32 642 | 0 643 | 644 | 645 | 646 | 668 | 669 | 670 | 14 671 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 672 | 0xFFFFFF 673 | false 674 | 0.8 675 | 0x333333 676 | 0.8 677 | 0 678 | 0x333333 679 | 3 680 | 15 681 | 2 682 | 15 683 | 4 684 | 5 685 | 15 686 | 10 687 | 0x333333 688 | 0 689 | 0xFFFFFF 690 | 0.8 691 | 0 692 | 0 693 | 694 | 695 | 696 | 0xFFFFFF 697 | 1 698 | 5 699 | 5 700 | 5 701 | 702 | 703 | 704 | 14 705 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 706 | 0xFFFFFF 707 | false 708 | 1 709 | 0x000000 710 | 0.8 711 | 0 712 | 0x333333 713 | 0 714 | 15 715 | 2 716 | 15 717 | 4 718 | 0 719 | 150 720 | 0 721 | 722 | 723 | 724 | 6 725 | 0x004eff 726 | 1 727 | 39 728 | 729 | 730 | 731 | 16 732 | Microsoft YaHei,\5FAE\8F6F\96C5\9ED1,微软雅黑 733 | 0xFFFFFF 734 | true 735 | 1 736 | 30 737 | 50 738 | 739 | -------------------------------------------------------------------------------- /static/crossdomain.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /static/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/liuzy88/91video/4f2ab77b641fffbf104807832b1340cf57788a5f/static/favicon.ico -------------------------------------------------------------------------------- /www/index.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 8 | 9 | 91Video 10 | 26 | 27 | 28 | <% if (data.mp4) { %> 29 |
30 |
31 |

上一集<%=data.title%>下一集

32 |
33 | 34 | 44 | <% } else if (data.rows) { 45 | let width = 11, pad = Math.floor(width / 2), start = 1, end = width, max = Math.ceil(data.count / rows); 46 | if (max <= width) { 47 | end = max; 48 | } else if (rows > width) { 49 | if (page > pad) { 50 | start += page - pad; 51 | end += page - pad; 52 | if (end > max) { 53 | end = max; 54 | start -= end - max; 55 | } 56 | } 57 | } else {end = rows;} 58 | %> 59 |
60 | 69 |
70 |
71 | 76 |
77 |
78 | 83 |
84 |
85 | 94 | <% } %> 95 | 96 | --------------------------------------------------------------------------------