UNPKG

53 kBJavaScriptView Raw
1'use strict';
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.renderRoute = undefined;
7
8var _defineProperty2 = require('babel-runtime/helpers/defineProperty');
9
10var _defineProperty3 = _interopRequireDefault(_defineProperty2);
11
12var _keys = require('babel-runtime/core-js/object/keys');
13
14var _keys2 = _interopRequireDefault(_keys);
15
16var _promise = require('babel-runtime/core-js/promise');
17
18var _promise2 = _interopRequireDefault(_promise);
19
20var _extends2 = require('babel-runtime/helpers/extends');
21
22var _extends3 = _interopRequireDefault(_extends2);
23
24var _regenerator = require('babel-runtime/regenerator');
25
26var _regenerator2 = _interopRequireDefault(_regenerator);
27
28var _asyncToGenerator2 = require('babel-runtime/helpers/asyncToGenerator');
29
30var _asyncToGenerator3 = _interopRequireDefault(_asyncToGenerator2);
31
32/**
33 * Render route into JS.
34 */
35
36var renderRoute = exports.renderRoute = function () {
37 var ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee(route, options) {
38 var _options$publicPath, publicPath, node;
39
40 return _regenerator2.default.wrap(function _callee$(_context) {
41 while (1) {
42 switch (_context.prev = _context.next) {
43 case 0:
44 (0, _bluebird.promisifyAll)(options.fs);
45 _options$publicPath = options.publicPath;
46 publicPath = _options$publicPath === undefined ? '/' : _options$publicPath;
47 _context.next = 5;
48 return renderRoutePage(route, publicPath, [''], options);
49
50 case 5:
51 node = _context.sent;
52 return _context.abrupt('return', node);
53
54 case 7:
55 case 'end':
56 return _context.stop();
57 }
58 }
59 }, _callee, this);
60 }));
61 return function renderRoute(_x, _x2) {
62 return ref.apply(this, arguments);
63 };
64}();
65
66var renderRoutePage = function () {
67 var ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee2(route, path, trace, options) {
68 var indexRoute, childRoutes, key, getComponent, _getComponent;
69
70 return _regenerator2.default.wrap(function _callee2$(_context2) {
71 while (1) {
72 switch (_context2.prev = _context2.next) {
73 case 0:
74 if (!route.route) {
75 _context2.next = 23;
76 break;
77 }
78
79 indexRoute = void 0;
80 childRoutes = [];
81 _context2.t0 = _regenerator2.default.keys(route.route);
82
83 case 4:
84 if ((_context2.t1 = _context2.t0()).done) {
85 _context2.next = 19;
86 break;
87 }
88
89 key = _context2.t1.value;
90
91 if (!(key === 'index')) {
92 _context2.next = 12;
93 break;
94 }
95
96 _context2.next = 9;
97 return renderRoutePage(route.route[key], null, trace, options);
98
99 case 9:
100 indexRoute = _context2.sent;
101 _context2.next = 17;
102 break;
103
104 case 12:
105 _context2.t2 = childRoutes;
106 _context2.next = 15;
107 return renderRoutePage(route.route[key], key, trace.concat(key), options);
108
109 case 15:
110 _context2.t3 = _context2.sent;
111
112 _context2.t2.push.call(_context2.t2, _context2.t3);
113
114 case 17:
115 _context2.next = 4;
116 break;
117
118 case 19:
119 getComponent = renderGetComponent(route.page, {
120 split: options.split !== undefined ? options.split : route.split,
121 chunkName: trace.join('/')
122 });
123 return _context2.abrupt('return', renderRouteSpec({
124 path: path,
125 getComponent: getComponent,
126 indexRoute: indexRoute,
127 childRoutes: childRoutes
128 }));
129
130 case 23:
131 if (!route.collection) {
132 _context2.next = 27;
133 break;
134 }
135
136 return _context2.abrupt('return', renderRouteCollection(route, path, trace, options));
137
138 case 27:
139 _getComponent = renderGetComponent(route.page, {
140 split: options.split !== undefined ? options.split : route.split,
141 chunkName: trace.join('/')
142 });
143 return _context2.abrupt('return', renderRouteSpec({
144 path: path,
145 getComponent: _getComponent
146 }));
147
148 case 29:
149 case 'end':
150 return _context2.stop();
151 }
152 }
153 }, _callee2, this);
154 }));
155 return function renderRoutePage(_x3, _x4, _x5, _x6) {
156 return ref.apply(this, arguments);
157 };
158}();
159
160var renderRouteCollection = function () {
161 var ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee3(route, path, trace, options) {
162 var _route$collection, _route$collection$pag, directory, match, paginate, chunkName, name, items, chunks, childRoutes, getComponent, pageParam;
163
164 return _regenerator2.default.wrap(function _callee3$(_context3) {
165 while (1) {
166 switch (_context3.prev = _context3.next) {
167 case 0:
168 _route$collection = route.collection;
169 _route$collection$pag = _route$collection.page;
170 directory = _route$collection$pag.directory;
171 match = _route$collection$pag.match;
172 paginate = _route$collection.paginate;
173 chunkName = trace.join('/');
174 name = chunkName;
175 _context3.next = 9;
176 return walkDirectory(options.fs, directory);
177
178 case 9:
179 _context3.t0 = function (filename) {
180 return match(filename);
181 };
182
183 _context3.t1 = function (filename) {
184 return {
185 filename: filename,
186 path: stripExtension((0, _path.relative)(directory, filename))
187 };
188 };
189
190 items = _context3.sent.filter(_context3.t0).map(_context3.t1);
191 chunks = [items];
192
193 if (paginate) {
194 chunks = (0, _lodash.chunk)(items, paginate.size);
195 }
196
197 chunks = chunks.map(function (chunk, idx) {
198 return renderGetCollectionChunk(chunk, {
199 page: route.page,
200 chunkNumber: idx,
201 chunkCount: chunks.length,
202 loader: META_LOADER,
203 chunkName: trace.concat('@page', idx + 1).join('/'),
204 split: options.split !== undefined ? options.split : true
205 });
206 });
207
208 childRoutes = items.map(function (item) {
209 return renderRouteSpec({
210 path: item.path,
211 getComponent: renderGetComponent(item.filename, {
212 split: options.split !== undefined ? options.split : true,
213 chunkName: trace.concat(item.path).join('/')
214 })
215 });
216 });
217
218 getComponent = function (_param) {
219 return {
220 "type": "FunctionExpression",
221 "id": {
222 "type": "Identifier",
223 "name": "getComponent"
224 },
225 "generator": false,
226 "expression": false,
227 "async": false,
228 "params": [{
229 "type": "Identifier",
230 "name": "nextState"
231 }, {
232 "type": "Identifier",
233 "name": "cb"
234 }],
235 "body": {
236 "type": "BlockStatement",
237 "body": [{
238 "type": "VariableDeclaration",
239 "declarations": [{
240 "type": "VariableDeclarator",
241 "id": {
242 "type": "Identifier",
243 "name": "pageNumber"
244 },
245 "init": {
246 "type": "CallExpression",
247 "callee": {
248 "type": "Identifier",
249 "name": "parseInt"
250 },
251 "arguments": [{
252 "type": "LogicalExpression",
253 "left": {
254 "type": "MemberExpression",
255 "object": {
256 "type": "MemberExpression",
257 "object": {
258 "type": "Identifier",
259 "name": "nextState"
260 },
261 "property": {
262 "type": "Identifier",
263 "name": "params"
264 },
265 "computed": false
266 },
267 "property": {
268 "type": "Identifier",
269 "name": "page"
270 },
271 "computed": false
272 },
273 "operator": "||",
274 "right": {
275 "type": "NumericLiteral",
276 "extra": {
277 "rawValue": 1,
278 "raw": "1"
279 },
280 "value": 1
281 }
282 }, {
283 "type": "NumericLiteral",
284 "extra": {
285 "rawValue": 10,
286 "raw": "10"
287 },
288 "value": 10
289 }]
290 }
291 }],
292 "kind": "var"
293 }, {
294 "type": "VariableDeclaration",
295 "declarations": [{
296 "type": "VariableDeclarator",
297 "id": {
298 "type": "Identifier",
299 "name": "chunkList"
300 },
301 "init": _param
302 }],
303 "kind": "var"
304 }, {
305 "type": "VariableDeclaration",
306 "declarations": [{
307 "type": "VariableDeclarator",
308 "id": {
309 "type": "Identifier",
310 "name": "getCollectionChunk"
311 },
312 "init": {
313 "type": "MemberExpression",
314 "object": {
315 "type": "Identifier",
316 "name": "chunkList"
317 },
318 "property": {
319 "type": "BinaryExpression",
320 "left": {
321 "type": "Identifier",
322 "name": "pageNumber"
323 },
324 "operator": "-",
325 "right": {
326 "type": "NumericLiteral",
327 "extra": {
328 "rawValue": 1,
329 "raw": "1"
330 },
331 "value": 1
332 }
333 },
334 "computed": true
335 }
336 }],
337 "kind": "var"
338 }, {
339 "type": "ExpressionStatement",
340 "expression": {
341 "type": "CallExpression",
342 "callee": {
343 "type": "Identifier",
344 "name": "getCollectionChunk"
345 },
346 "arguments": [{
347 "type": "Identifier",
348 "name": "nextState"
349 }, {
350 "type": "Identifier",
351 "name": "cb"
352 }]
353 }
354 }],
355 "directives": []
356 },
357 "extra": {
358 "parenthesized": true,
359 "parenStart": 0
360 }
361 };
362 }(types.arrayExpression(chunks));
363
364 pageParam = types.arrayExpression(chunks.map(function (_, idx) {
365 return types.numericLiteral(idx + 1);
366 }));
367
368 childRoutes = childRoutes.concat(renderRouteSpec({
369 path: '@page/:page',
370 name: name,
371 params: function (_param2) {
372 return {
373 "type": "ObjectExpression",
374 "properties": [{
375 "type": "ObjectProperty",
376 "method": false,
377 "shorthand": false,
378 "computed": false,
379 "key": {
380 "type": "Identifier",
381 "name": "page"
382 },
383 "value": _param2
384 }],
385 "extra": {
386 "parenthesized": true,
387 "parenStart": 0
388 }
389 };
390 }(pageParam)
391 }));
392
393 return _context3.abrupt('return', renderRouteSpec({
394 path: path,
395 getComponent: getComponent,
396 childRoutes: childRoutes,
397 name: name
398 }));
399
400 case 20:
401 case 'end':
402 return _context3.stop();
403 }
404 }
405 }, _callee3, this);
406 }));
407 return function renderRouteCollection(_x7, _x8, _x9, _x10) {
408 return ref.apply(this, arguments);
409 };
410}();
411
412/**
413 * Render a fetcher for a collection chunk.
414 */
415
416
417var walkDirectory = function () {
418 var ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee5(fs, directory) {
419 var _this = this;
420
421 var files;
422 return _regenerator2.default.wrap(function _callee5$(_context5) {
423 while (1) {
424 switch (_context5.prev = _context5.next) {
425 case 0:
426 _context5.next = 2;
427 return fs.readdirAsync(directory).then(function (segments) {
428 return _promise2.default.all(segments.map(function () {
429 var ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee4(segment) {
430 var filename, stat;
431 return _regenerator2.default.wrap(function _callee4$(_context4) {
432 while (1) {
433 switch (_context4.prev = _context4.next) {
434 case 0:
435 filename = (0, _path.join)(directory, segment);
436 _context4.next = 3;
437 return fs.statAsync(filename);
438
439 case 3:
440 stat = _context4.sent;
441
442 if (!stat.isDirectory()) {
443 _context4.next = 8;
444 break;
445 }
446
447 return _context4.abrupt('return', walkDirectory(fs, filename));
448
449 case 8:
450 return _context4.abrupt('return', [filename]);
451
452 case 9:
453 case 'end':
454 return _context4.stop();
455 }
456 }
457 }, _callee4, _this);
458 }));
459 return function (_x13) {
460 return ref.apply(this, arguments);
461 };
462 }()));
463 });
464
465 case 2:
466 files = _context5.sent;
467 return _context5.abrupt('return', (0, _lodash.flatten)(files));
468
469 case 4:
470 case 'end':
471 return _context5.stop();
472 }
473 }
474 }, _callee5, this);
475 }));
476 return function walkDirectory(_x11, _x12) {
477 return ref.apply(this, arguments);
478 };
479}();
480
481exports.forEach = forEach;
482exports.validate = validate;
483exports.forEachPath = forEachPath;
484
485var _path = require('path');
486
487var _bluebird = require('bluebird');
488
489var _minimatch = require('minimatch');
490
491var _minimatch2 = _interopRequireDefault(_minimatch);
492
493var _lodash = require('lodash');
494
495var _schema = require('validated/schema');
496
497var _object = require('validated/object');
498
499var _babelTypes = require('babel-types');
500
501var types = _interopRequireWildcard(_babelTypes);
502
503var _config = require('./config');
504
505function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } }
506
507function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
508
509/**
510 * @copyright 2016-present, Sitegen team
511 */
512
513var META_LOADER = require.resolve('./loader/meta');
514var CHUNK_LOADER = require.resolve('./loader/chunk');
515var REACT_HOT_LOADER = require.resolve('react-hot-loader/webpack');
516
517function forEach(route, func) {
518 func(route);
519 if (route.route) {
520 for (var k in route.route) {
521 forEach(route.route[k], func);
522 }
523 }
524}
525
526/**
527 * Validate route.
528 */
529function validate(route, _ref) {
530 var basedir = _ref.basedir;
531
532 var pattern = _schema.string.andThen(function (val) {
533 return parsePattern(val, { basedir: basedir });
534 });
535 var pageShortcut = _schema.string.andThen(function (page) {
536 return { page: page };
537 });
538 var collectionShortcut = pattern.andThen(function (page) {
539 return { page: page };
540 });
541 var schema = (0, _schema.ref)();
542
543 var routePage = (0, _schema.oneOf)(pageShortcut, (0, _schema.object)({
544 page: _schema.string,
545 split: (0, _schema.maybe)(_schema.boolean),
546 route: (0, _schema.maybe)((0, _schema.mapping)(schema)),
547 collection: (0, _schema.maybe)((0, _schema.oneOf)(collectionShortcut, (0, _schema.object)({
548 page: pattern,
549 paginate: (0, _schema.maybe)((0, _schema.object)({ size: _schema.number }))
550 })))
551 })).andThen(function (route, error) {
552 if (route.route && route.collection) {
553 error('Route cannot have both "route" and "collection" defined');
554 }
555 return route;
556 });
557
558 schema.set(routePage);
559
560 return (0, _object.validate)(schema, route);
561}function renderGetCollectionChunk(chunk, options) {
562 options = (0, _extends3.default)({
563 loader: META_LOADER,
564 chunkName: 'collectionPage',
565 split: false
566 }, options);
567
568 var query = {
569 chunk: chunk,
570 loader: options.loader,
571 chunkNumber: options.chunkNumber,
572 chunkCount: options.chunkCount
573 };
574
575 var req = (0, _config.moduleRequest)(options.page, { loader: CHUNK_LOADER, query: query },
576 // TODO: we don't really want to leak configuration here, think of a better
577 // way to inject this
578 REACT_HOT_LOADER);
579 return renderGetComponent(req, {
580 split: options.split,
581 chunkName: options.chunkName
582 });
583}
584
585/**
586 * Render React Router route spec.
587 */
588function renderRouteSpec(_ref2) {
589 var path = _ref2.path;
590 var getComponent = _ref2.getComponent;
591 var indexRoute = _ref2.indexRoute;
592 var childRoutes = _ref2.childRoutes;
593 var name = _ref2.name;
594 var params = _ref2.params;
595
596 path = path == null ? function () {
597 return {
598 "type": "Identifier",
599 "name": "undefined",
600 "extra": {
601 "parenthesized": true,
602 "parenStart": 0
603 }
604 };
605 }() : types.stringLiteral(path);
606 getComponent = getComponent == null ? function () {
607 return {
608 "type": "Identifier",
609 "name": "undefined",
610 "extra": {
611 "parenthesized": true,
612 "parenStart": 0
613 }
614 };
615 }() : getComponent;
616 indexRoute = indexRoute == null ? function () {
617 return {
618 "type": "Identifier",
619 "name": "undefined",
620 "extra": {
621 "parenthesized": true,
622 "parenStart": 0
623 }
624 };
625 }() : indexRoute;
626 childRoutes = types.arrayExpression(childRoutes || []);
627 name = name == null ? function () {
628 return {
629 "type": "Identifier",
630 "name": "undefined",
631 "extra": {
632 "parenthesized": true,
633 "parenStart": 0
634 }
635 };
636 }() : types.stringLiteral(name);
637 params = params == null ? function () {
638 return {
639 "type": "Identifier",
640 "name": "undefined",
641 "extra": {
642 "parenthesized": true,
643 "parenStart": 0
644 }
645 };
646 }() : params;
647 return function (_param3, _param4, _param5, _param6, _param7, _param8) {
648 return {
649 "type": "ObjectExpression",
650 "properties": [{
651 "type": "ObjectProperty",
652 "method": false,
653 "shorthand": false,
654 "computed": false,
655 "key": {
656 "type": "Identifier",
657 "name": "path"
658 },
659 "value": _param3
660 }, {
661 "type": "ObjectProperty",
662 "method": false,
663 "shorthand": false,
664 "computed": false,
665 "key": {
666 "type": "Identifier",
667 "name": "getComponent"
668 },
669 "value": _param4
670 }, {
671 "type": "ObjectProperty",
672 "method": false,
673 "shorthand": false,
674 "computed": false,
675 "key": {
676 "type": "Identifier",
677 "name": "indexRoute"
678 },
679 "value": _param5
680 }, {
681 "type": "ObjectProperty",
682 "method": false,
683 "shorthand": false,
684 "computed": false,
685 "key": {
686 "type": "Identifier",
687 "name": "childRoutes"
688 },
689 "value": _param6
690 }, {
691 "type": "ObjectProperty",
692 "method": false,
693 "shorthand": false,
694 "computed": false,
695 "key": {
696 "type": "Identifier",
697 "name": "name"
698 },
699 "value": _param7
700 }, {
701 "type": "ObjectProperty",
702 "method": false,
703 "shorthand": false,
704 "computed": false,
705 "key": {
706 "type": "Identifier",
707 "name": "params"
708 },
709 "value": _param8
710 }],
711 "extra": {
712 "parenthesized": true,
713 "parenStart": 0
714 }
715 };
716 }(path, getComponent, indexRoute, childRoutes, name, params);
717}
718
719/**
720 * Render component loader to load component from a given module.
721 */
722function renderGetComponent(id, options) {
723 options = (0, _extends3.default)({
724 split: false,
725 chunkName: 'component'
726 }, options);
727
728 var req = types.stringLiteral(id);
729
730 if (options.split) {
731 return function (_param9, _param10) {
732 return {
733 "type": "FunctionExpression",
734 "id": {
735 "type": "Identifier",
736 "name": "getComponentAsync"
737 },
738 "generator": false,
739 "expression": false,
740 "async": false,
741 "params": [{
742 "type": "Identifier",
743 "name": "_nextState"
744 }, {
745 "type": "Identifier",
746 "name": "cb"
747 }],
748 "body": {
749 "type": "BlockStatement",
750 "body": [{
751 "type": "ExpressionStatement",
752 "expression": {
753 "type": "CallExpression",
754 "callee": {
755 "type": "MemberExpression",
756 "object": {
757 "type": "Identifier",
758 "name": "require"
759 },
760 "property": {
761 "type": "Identifier",
762 "name": "ensure"
763 },
764 "computed": false
765 },
766 "arguments": [{
767 "type": "ArrayExpression",
768 "elements": []
769 }, {
770 "type": "FunctionExpression",
771 "id": null,
772 "generator": false,
773 "expression": false,
774 "async": false,
775 "params": [{
776 "type": "Identifier",
777 "name": "require"
778 }],
779 "body": {
780 "type": "BlockStatement",
781 "body": [{
782 "type": "ExpressionStatement",
783 "expression": {
784 "type": "CallExpression",
785 "callee": {
786 "type": "Identifier",
787 "name": "cb"
788 },
789 "arguments": [{
790 "type": "NullLiteral"
791 }, {
792 "type": "MemberExpression",
793 "object": {
794 "type": "CallExpression",
795 "callee": {
796 "type": "Identifier",
797 "name": "require"
798 },
799 "arguments": [_param9]
800 },
801 "property": {
802 "type": "Identifier",
803 "name": "default"
804 },
805 "computed": false
806 }]
807 }
808 }],
809 "directives": []
810 }
811 }, _param10]
812 }
813 }],
814 "directives": []
815 },
816 "extra": {
817 "parenthesized": true,
818 "parenStart": 0
819 }
820 };
821 }(req, types.stringLiteral(options.chunkName));
822 } else {
823 return function (_param11) {
824 return {
825 "type": "FunctionExpression",
826 "id": {
827 "type": "Identifier",
828 "name": "getComponentSync"
829 },
830 "generator": false,
831 "expression": false,
832 "async": false,
833 "params": [{
834 "type": "Identifier",
835 "name": "_nextState"
836 }, {
837 "type": "Identifier",
838 "name": "cb"
839 }],
840 "body": {
841 "type": "BlockStatement",
842 "body": [{
843 "type": "ExpressionStatement",
844 "expression": {
845 "type": "CallExpression",
846 "callee": {
847 "type": "Identifier",
848 "name": "cb"
849 },
850 "arguments": [{
851 "type": "NullLiteral"
852 }, {
853 "type": "MemberExpression",
854 "object": {
855 "type": "CallExpression",
856 "callee": {
857 "type": "Identifier",
858 "name": "require"
859 },
860 "arguments": [_param11]
861 },
862 "property": {
863 "type": "Identifier",
864 "name": "default"
865 },
866 "computed": false
867 }]
868 }
869 }],
870 "directives": []
871 },
872 "extra": {
873 "parenthesized": true,
874 "parenStart": 0
875 }
876 };
877 }(req);
878 }
879}
880
881function parsePattern(pattern, _ref3) {
882 var basedir = _ref3.basedir;
883
884 pattern = (0, _path.join)(basedir, pattern);
885 var matcher = new _minimatch2.default.Minimatch(pattern);
886 var directory = [];
887 for (var i = 0; i < matcher.set[0].length; i++) {
888 if (typeof matcher.set[0][i] === 'string') {
889 directory.push(matcher.set[0][i]);
890 } else {
891 break;
892 }
893 }
894 if (directory.length > 1) {
895 directory = directory.join('/');
896 }
897 return {
898 directory: directory,
899 match: matcher.match.bind(matcher)
900 };
901}
902
903function stripExtension(filename) {
904 return filename.substring(0, filename.length - (0, _path.extname)(filename).length);
905}
906
907function forEachPath(route, func) {
908 var trace = arguments.length <= 2 || arguments[2] === undefined ? [''] : arguments[2];
909
910 var path = '/' + trace.filter(Boolean).join('/');
911 if (route.params) {
912 (0, _keys2.default)(route.params).forEach(function (key) {
913 route.params[key].forEach(function (value) {
914 func(interpolateParams(path, (0, _defineProperty3.default)({}, key, value)));
915 });
916 });
917 } else {
918 func(path);
919 }
920 if (route.childRoutes) {
921 route.childRoutes.forEach(function (route) {
922 return forEachPath(route, func, trace.concat(route.path));
923 });
924 }
925}
926
927function interpolateParams(path, params) {
928 return path.replace(/:([a-zA-Z_\-]+)/g, function (_, key) {
929 return params[key];
930 });
931}
932//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/andreypopp/Workspace/sitegen/src/route.js"],"names":[],"mappings":"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","file":"route.js","sourcesContent":["/**\n * @copyright 2016-present, Sitegen team\n */\n\nimport {join, extname, relative} from 'path';\nimport {promisifyAll} from 'bluebird';\nimport minimatch from 'minimatch';\nimport {flatten, chunk} from 'lodash';\n\nimport {string, number, ref, maybe, object, mapping, oneOf, boolean} from 'validated/schema';\nimport {validate as validateObject} from 'validated/object';\n\nimport * as types from 'babel-types';\nimport {moduleRequest} from './config';\n\nconst META_LOADER = require.resolve('./loader/meta');\nconst CHUNK_LOADER = require.resolve('./loader/chunk');\nconst REACT_HOT_LOADER = require.resolve('react-hot-loader/webpack');\n\nexport function forEach(route, func) {\n  func(route);\n  if (route.route) {\n    for (let k in route.route) {\n      forEach(route.route[k], func);\n    }\n  }\n}\n\n/**\n * Validate route.\n */\nexport function validate(route, {basedir}) {\n  let pattern = string.andThen(val => parsePattern(val, {basedir}));\n  let pageShortcut = string.andThen(page => ({page}));\n  let collectionShortcut = pattern.andThen(page => ({page}));\n  let schema = ref();\n\n  let routePage = oneOf(\n    pageShortcut,\n    object({\n      page: string,\n      split: maybe(boolean),\n      route: maybe(mapping(schema)),\n      collection: maybe(oneOf(\n        collectionShortcut,\n        object({\n          page: pattern,\n          paginate: maybe(object({size: number})),\n        })\n      )),\n    })\n  ).andThen((route, error) => {\n    if (route.route && route.collection) {\n      error('Route cannot have both \"route\" and \"collection\" defined');\n    }\n    return route;\n  });\n\n  schema.set(routePage);\n\n  return validateObject(schema, route);\n}\n\n/**\n * Render route into JS.\n */\nexport async function renderRoute(route, options) {\n  promisifyAll(options.fs);\n  let {publicPath = '/'} = options;\n  let node = await renderRoutePage(route, publicPath, [''], options);\n  return node;\n}\n\nasync function renderRoutePage(route, path, trace, options) {\n  if (route.route) {\n    let indexRoute;\n    let childRoutes = [];\n    for (let key in route.route) {\n      if (key === 'index') {\n        indexRoute = await renderRoutePage(route.route[key], null, trace, options);\n      } else {\n        childRoutes.push(\n          await renderRoutePage(route.route[key], key, trace.concat(key), options)\n        );\n      }\n    }\n\n    let getComponent = renderGetComponent(route.page, {\n      split: options.split !== undefined ? options.split : route.split,\n      chunkName: trace.join('/'),\n    });\n    return renderRouteSpec({\n      path,\n      getComponent,\n      indexRoute,\n      childRoutes\n    });\n  } else if (route.collection) {\n    return renderRouteCollection(route, path, trace, options);\n  } else {\n    let getComponent = renderGetComponent(route.page, {\n      split: options.split !== undefined ? options.split : route.split,\n      chunkName: trace.join('/'),\n    });\n    return renderRouteSpec({\n      path,\n      getComponent,\n    });\n  }\n}\n\nasync function renderRouteCollection(route, path, trace, options) {\n  let {page: {directory, match}, paginate} = route.collection;\n\n  let chunkName = trace.join('/');\n  let name = chunkName;\n\n  let items = (await walkDirectory(options.fs, directory))\n    .filter(filename => match(filename))\n    .map(filename => ({\n      filename,\n      path: stripExtension(relative(directory, filename))\n    }));\n\n  let chunks = [items];\n  if (paginate) {\n    chunks = chunk(items, paginate.size);\n  }\n\n  chunks = chunks.map((chunk, idx) => renderGetCollectionChunk(chunk, {\n    page: route.page,\n    chunkNumber: idx,\n    chunkCount: chunks.length,\n    loader: META_LOADER,\n    chunkName: trace.concat('@page', idx + 1).join('/'),\n    split: options.split !== undefined ? options.split : true,\n  }));\n\n  let childRoutes = items.map(item => renderRouteSpec({\n    path: item.path,\n    getComponent: renderGetComponent(item.filename, {\n      split: options.split !== undefined ? options.split : true,\n      chunkName: trace.concat(item.path).join('/'),\n    }),\n  }));\n\n  let getComponent = expr`function getComponent(nextState, cb) {\n    var pageNumber = parseInt(nextState.params.page || 1, 10);\n    var chunkList = ${types.arrayExpression(chunks)};\n    var getCollectionChunk = chunkList[pageNumber - 1];\n    getCollectionChunk(nextState, cb);\n  }`;\n\n  let pageParam = types.arrayExpression(chunks.map((_, idx) => types.numericLiteral(idx + 1)));\n  childRoutes = childRoutes.concat(renderRouteSpec({\n    path: '@page/:page',\n    name,\n    params: expr`{page: ${pageParam}}`,\n  }));\n\n  return renderRouteSpec({\n    path,\n    getComponent,\n    childRoutes,\n    name,\n  });\n}\n\n/**\n * Render a fetcher for a collection chunk.\n */\nfunction renderGetCollectionChunk(chunk, options) {\n  options = {\n    loader: META_LOADER,\n    chunkName: 'collectionPage',\n    split: false,\n    ...options,\n  };\n\n  let query = {\n    chunk,\n    loader: options.loader,\n    chunkNumber: options.chunkNumber,\n    chunkCount: options.chunkCount,\n  };\n\n  let req = moduleRequest(\n    options.page,\n    {loader: CHUNK_LOADER, query},\n    // TODO: we don't really want to leak configuration here, think of a better\n    // way to inject this\n    REACT_HOT_LOADER\n  );\n  return renderGetComponent(req, {\n    split: options.split,\n    chunkName: options.chunkName,\n  });\n}\n\n/**\n * Render React Router route spec.\n */\nfunction renderRouteSpec({path, getComponent, indexRoute, childRoutes, name, params}) {\n  path = path == null ? expr`undefined` : types.stringLiteral(path);\n  getComponent = getComponent == null ? expr`undefined` : getComponent;\n  indexRoute = indexRoute == null ? expr`undefined` : indexRoute;\n  childRoutes = types.arrayExpression(childRoutes || []);\n  name = name == null ? expr`undefined` : types.stringLiteral(name);\n  params = params == null ? expr`undefined` : params;\n  return expr`{\n    path: ${path},\n    getComponent: ${getComponent},\n    indexRoute: ${indexRoute},\n    childRoutes: ${childRoutes},\n    name: ${name},\n    params: ${params},\n  }`;\n}\n\n/**\n * Render component loader to load component from a given module.\n */\nfunction renderGetComponent(id, options) {\n  options = {\n    split: false,\n    chunkName: 'component',\n    ...options,\n  };\n\n  let req = types.stringLiteral(id);\n\n  if (options.split) {\n    return expr`\n      function getComponentAsync(_nextState, cb) {\n        require.ensure([], function(require) {\n          cb(null, require(${req}).default);\n        }, ${types.stringLiteral(options.chunkName)});\n      }\n    `;\n  } else {\n    return expr`\n      function getComponentSync(_nextState, cb) {\n        cb(null, require(\"${req}\").default);\n      }\n    `;\n  }\n}\n\nfunction parsePattern(pattern, {basedir}) {\n  pattern = join(basedir, pattern);\n  let matcher = new minimatch.Minimatch(pattern);\n  let directory = [];\n  for (let i = 0; i < matcher.set[0].length; i++) {\n    if (typeof matcher.set[0][i] === 'string') {\n      directory.push(matcher.set[0][i]);\n    } else {\n      break;\n    }\n  }\n  if (directory.length > 1) {\n    directory = directory.join('/');\n  }\n  return {\n    directory,\n    match: matcher.match.bind(matcher),\n  };\n}\n\nfunction stripExtension(filename) {\n  return filename.substring(0, filename.length - extname(filename).length);\n}\n\nasync function walkDirectory(fs, directory) {\n  let files = await fs.readdirAsync(directory).then(segments =>\n    Promise.all(segments.map(async segment => {\n      let filename = join(directory, segment);\n      let stat = await fs.statAsync(filename);\n      if (stat.isDirectory()) {\n        return walkDirectory(fs, filename);\n      } else {\n        return [filename];\n      }\n    }))\n  );\n  return flatten(files);\n}\n\nexport function forEachPath(route, func, trace = ['']) {\n  let path = '/' + trace.filter(Boolean).join('/');\n  if (route.params) {\n    Object.keys(route.params).forEach(key => {\n      route.params[key].forEach(value => {\n        func(interpolateParams(path, {[key]: value}));\n      });\n    });\n  } else {\n    func(path);\n  }\n  if (route.childRoutes) {\n    route.childRoutes.forEach(route =>\n      forEachPath(route, func, trace.concat(route.path)));\n  }\n}\n\nfunction interpolateParams(path, params) {\n  return path.replace(/:([a-zA-Z_\\-]+)/g, (_, key) => params[key]);\n}\n"]}
\No newline at end of file