├── .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 | 
--------------------------------------------------------------------------------
/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 |
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 |
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 |
61 | Page <%=page%> of <%=max%>
62 | - Previous
63 | <% for (let i = start; i <= end; i++) { %>
64 | - <%= i %>
65 | <% } %>
66 | - Next
67 | Count <%=rows%> of <%=data.count%>
68 |
69 |
70 |
71 |
72 | <% for (let i = 0; i < Math.ceil(len / 2); i++) { let d = data.rows[i];%>
73 | - <%=d.title%>
74 | <% } %>
75 |
76 |
77 |
78 |
79 | <% for (let i = Math.ceil(len / 2); i < len; i++) { let d = data.rows[i];%>
80 | - <%=d.title%>
81 | <% } %>
82 |
83 |
84 |
85 |
86 | Page <%=page%> of <%=max%>
87 | - Previous
88 | <% for (let i = start; i <= end; i++) { %>
89 | - <%= i %>
90 | <% } %>
91 | - Next
92 | Count <%=rows%> of <%=data.count%>
93 |
94 | <% } %>
95 |
96 |
--------------------------------------------------------------------------------