Private GIT

Skip to content
Snippets Groups Projects
Select Git revision
  • 212cba93ee4d4e84c35a9578b9906d7064cca9dc
  • master default protected
  • fix_nzb_cat
  • develop
  • guessit2-minimal
  • ssl_warning
  • UHD-qualities
  • fix_providers8
  • !
  • tvvault
  • provider_alpharatio
  • v5.1.1
  • v5.1
  • v5.0.3
  • v5.0.2
  • v5.0.1
  • v5.0
  • v4.2.1.07
  • v4.2.1.06
  • v4.2.1.05
  • v4.2.1.04
  • v4.2.1.03
  • v4.2.1.02
  • v4.2.1.01
  • v4.2.1.0
  • v4.2.0.6
  • v4.2.0.5
  • v4.2.0.4
  • v4.2.0.3
  • v4.2.0.2
  • v4.2.0.1
31 results

Gruntfile.js

Blame
  • Gruntfile.js 19.46 KiB
    'use strict';
    
    module.exports = function(grunt) {
        const isTravis = Boolean(process.env.TRAVIS);
    
        grunt.registerTask('default', [
            'clean',
            'bower',
            'bower_concat',
            'copy',
            'uglify',
            'cssmin'
        ]);
    
        grunt.registerTask('auto_update_trans', 'Update translations on master and push to master & develop', function() {
            if (!isTravis) {
                grunt.fatal('This task is only for Travis-CI!');
                return false;
            }
            grunt.log.writeln('Running grunt and updating translations...'.magenta);
            grunt.task.run([
                'exec:git:checkout:master',
                'default', // Run default task
                'update_trans', // Update translations
                'exec:commit_changed_files:yes', // Determine what we need to commit if needed, stop if nothing to commit.
                'exec:git:reset --hard', // Reset unstaged changes (to allow for a rebase)
                'exec:git:checkout:develop', 'exec:git:rebase:master', // FF develop to the updated master
                'exec:git_push:origin:master develop' // Push master and develop
            ]);
        });
    
        grunt.registerTask('update_trans', 'Update translations', function() {
            grunt.log.writeln('Updating translations...'.magenta);
            var tasks = [
                'exec:babel_extract',
                'exec:babel_update',
                // + crowdin
                'exec:babel_compile',
                'po2json'
            ];
            if (process.env.CROWDIN_API_KEY) {
                tasks.splice(2, 0, 'exec:crowdin_upload', 'exec:crowdin_download'); // insert items at index 2
            } else {
                grunt.log.warn('Environment variable `CROWDIN_API_KEY` is not set, not syncing with Crowdin.'.bold);
            }
    
            grunt.task.run(tasks);
        });
    
        /****************************************
        *  Admin only tasks                     *
        ****************************************/
        grunt.registerTask('publish', 'ADMIN: Create a new release tag and generate new CHANGES.md', [
            'ci',
            'newrelease', // Pull and merge develop to master, create and push a new release
            'genchanges' // Update CHANGES.md
        ]);
    
        grunt.registerTask('newrelease', "Pull and merge develop to master, create and push a new release", [
            'exec:git:checkout:develop', 'exec:git:pull', // Pull develop
            'exec:git:checkout:master', 'exec:git:pull', // Pull master
            'exec:git:merge:develop', // Merge develop into master
            'exec:git_get_last_tag', 'exec:git_list_changes', // List changes from since last tag
            '_get_next_tag', 'exec:git_tag_new', // Create new release tag
            'exec:git_push:origin:master:tags', // Push master + tags
            'exec:git:checkout:develop' // Go back to develop
        ]);
    
        grunt.registerTask('genchanges', "Generate CHANGES.md file", function() {
            var file = grunt.option('file'); // --file=path/to/sickrage.github.io/sickrage-news/CHANGES.md
            if (!file) {
                file = process.env.SICKRAGE_CHANGES_FILE;
            }
            if (file && grunt.file.exists(file)) {
                grunt.config('changesmd_file', file.replace(/\\/g, '/')); // Use forward slashes only.
            } else {
                grunt.fatal('\tYou must provide a path to CHANGES.md to generate changes.\n' +
                    '\t\tUse --file=path/to/sickrage.github.io/sickrage-news/CHANGES.md\n' +
                    '\t\tor set the path in SICKRAGE_CHANGES_FILE (environment variable)');
            }
            grunt.task.run(['exec:git_list_tags', '_genchanges', 'exec:commit_changelog']);
        });
    
        /****************************************
        *  Task configurations                  *
        ****************************************/
        require('load-grunt-tasks')(grunt); // loads all grunt tasks matching the ['grunt-*', '@*/grunt-*'] patterns
        grunt.initConfig({
            clean: {
                dist: './dist/',
                'bower_components': './bower_components',
                fonts: './gui/slick/css/*.ttf',
                options: {
                    force: true
                }
            },
            bower: {
                install: {
                    options: {
                        copy: false
                    }
                }
            },
            'bower_concat': {
                all: {
                    dest: {
                        js: './dist/bower.js',
                        css: './dist/bower.css'
                    },
                    exclude: [],
                    dependencies: {},
                    mainFiles: {
                        'tablesorter': [
                            'dist/js/jquery.tablesorter.combined.js',
                            'dist/js/widgets/widget-columnSelector.min.js',
                            'dist/js/widgets/widget-stickyHeaders.min.js',
                            'dist/css/theme.blue.min.css'
                        ],
                        'bootstrap': [
                            'dist/css/bootstrap.min.css',
                            'dist/js/bootstrap.min.js'
                        ],
                        'bootstrap-formhelpers': [
                            'dist/js/bootstrap-formhelpers.min.js'
                        ],
                        'isotope': [
                            "dist/isotope.pkgd.min.js"
                        ],
                        "outlayer": [
                            "item.js",
                            "outlayer.js"
                        ],
                        "openSans": [
                            "*.ttf", "*.css"
                        ]
                    },
                    bowerOptions: {
                        relative: false
                    }
                }
            },
            copy: {
                openSans: {
                    files: [{
                        expand: true,
                        dot: true,
                        cwd: 'bower_components/openSans',
                        src: [
                            '*.ttf'
                        ],
                        dest: './gui/slick/css/'
                    }]
                },
                glyphicon: {
                    files: [{
                        expand: true,
                        dot: true,
                        cwd: 'bower_components/bootstrap/fonts',
                        src: [
                            '*.eot',
                            '*.svg',
                            '*.ttf',
                            '*.woff',
                            '*.woff2'
                        ],
                        dest: './gui/slick/fonts/'
                    }]
                }
            },
            uglify: {
                bower: {
                    files: {
                        './gui/slick/js/vender.min.js': ['./dist/bower.js']
                    }
                },
                core: {
                    files: {
                        './gui/slick/js/core.min.js': ['./gui/slick/js/core.js']
                    }
                }
            },
            cssmin: {
                options: {
                    shorthandCompacting: false,
                    roundingPrecision: -1
                },
                bower: {
                    files: {
                        './gui/slick/css/vender.min.css': ['./dist/bower.css']
                    }
                },
                core: {
                    files: {
                        './gui/slick/css/core.min.css': ['./dist/core.css']
                    }
                }
            },
            po2json: {
                messages: {
                    options: {
                        format: 'jed',
                        singleFile: true
                    },
                    files: [{
                        expand: true,
                        src: './locale/*/LC_MESSAGES/messages.po',
                        dest: '',
                        ext: '' // workaround for relative files
                    }]
                }
            },
            exec: {
                // Translations
                'babel_extract': {cmd: 'python setup.py extract_messages'},
                'babel_update': {cmd: 'python setup.py update_catalog'},
                'crowdin_upload': {cmd: 'crowdin-cli-py upload sources'},
                'crowdin_download': {cmd: 'crowdin-cli-py download'},
                'babel_compile': {cmd: 'python setup.py compile_catalog'},
    
                // Publish/Releases
                'git': {
                    cmd: function (cmd, branch) {
                        branch = branch ? ' ' + branch : '';
                        return 'git ' + cmd + branch;
                    }
                },
                'commit_changed_files': { // Choose what to commit.
                    cmd: function(travis) {
                        grunt.config('stop_no_changes', Boolean(travis));
                        return 'git status -s -- locale/ gui/';
                    },
                    stdout: false,
                    callback: function(err, stdout) {
                        stdout = stdout.trim();
                        if (!stdout.length) {
                            grunt.fatal('No changes to commit.', 0);
                        }
    
                        var commitMsg = [];
                        var commitPaths = [];
                        if (stdout.match(/gui\/.*(vender|core)\.min\.(js|css)$/gm)) {
                            commitMsg.push('Grunt');
                            commitPaths.push('gui/**/vender.min.*');
                            commitPaths.push('gui/**/core.min.*');
                        }
                        if (stdout.match(/locale\/.*(pot|po|mo|json)$/gm)) {
                            commitMsg.push('Update translations');
                            commitPaths.push('locale/');
                        }
    
                        if (!commitMsg.length || !commitPaths.length) {
                            if (grunt.config('stop_no_changes')) {
                                grunt.fatal('Nothing to commit, aborting', 0);
                            } else {
                                grunt.log.ok('No extra changes to commit'.green);
                            }
                        } else {
                            commitMsg = commitMsg.join(', ');
                            if (process.env.TRAVIS_BUILD_NUMBER) {
                                commitMsg += ' (build ' + process.env.TRAVIS_BUILD_NUMBER + ') [skip ci]';
                            }
                            grunt.config('commit_msg', commitMsg);
                            grunt.config('commit_paths', commitPaths.join(' '));
                            grunt.task.run('exec:commit_combined');
                        }
                    }
                },
                'commit_combined': {
                    cmd: function() {
                        var message = grunt.config('commit_msg');
                        var paths = grunt.config('commit_paths');
                        if (!message || !paths) {
                            grunt.fatal('Call exec:commit_changed_files instead!');
                        }
                        return 'git add -- ' + paths;
                    },
                    callback: function(err) {
                        if (!err) {
                            if (!isTravis) {
                                grunt.task.run('exec:git:commit:-m "' + grunt.config('commit_msg') + '"');
                            } else { // Workaround for Travis (with -m "text" the quotes are within the message)
                                var msgFilePath = 'commit-msg.txt';
                                grunt.file.write(msgFilePath, grunt.config('commit_msg'));
                                grunt.task.run('exec:git:commit:-F ' + msgFilePath);
                            }
                        }
                    }
                },
                'git_get_last_tag': {
                    cmd: 'git for-each-ref --sort=-refname --count=1 --format "%(refname:short)" refs/tags/v20[0-9][0-9]*',
                    stdout: false,
                    callback: function(err, stdout) {
                        stdout = stdout.trim();
                        if (/^v\d{4}.\d{1,2}.\d{1,2}.\d+$/.test(stdout)) {
                            grunt.config('last_tag', stdout);
                        } else {
                            grunt.fatal('Could not get the last tag name. We got: ' + stdout);
                        }
                    }
                },
                'git_list_changes': {
                    cmd: function() { return 'git log --oneline --pretty=format:%s ' + grunt.config('last_tag') + '..HEAD'; },
                    stdout: false,
                    callback: function(err, stdout) {
                        var commits = stdout.trim()
                            .replace(/`/gm, '').replace(/^\([\w\d\s,.\-+_/>]+\)\s/gm, '');  // removes ` and tag information
                        if (commits) {
                            grunt.config('commits', commits);
                        } else {
                            grunt.fatal('Getting new commit list failed!');
                        }
                    }
                },
                'git_tag_new': {
                    cmd: function (sign) {
                        sign = sign !== "true" ? '' : '-s ';
                        return 'git tag ' + sign + grunt.config('next_tag') + ' -m "' + grunt.config('commits') + '"';
                    },
                    stdout: false
                },
                'git_push': {
                    cmd: function (remote, branch, tags) {
                        var pushCmd = 'git push ' + remote + ' ' + branch;
                        if (tags) {
                            pushCmd += ' --tags';
                        }
                        if (grunt.option('no-push')) {
                            grunt.log.warn('Pushing with --dry-run ...'.magenta);
                            pushCmd += ' --dry-run';
                        }
                        return pushCmd;
                    },
                    stderr: false,
                    callback: function(err, stdout, stderr) {
                        grunt.log.write(stderr.replace(process.env.GH_CRED, '[censored]'));
                    }
                },
                'git_list_tags': {
                    cmd: 'git for-each-ref --sort=refname ' +
                            '--format="%(refname:short)|||%(objectname)|||%(contents)\xB6\xB6\xB6" ' +
                            'refs/tags/v20[0-9][0-9]*',
                    stdout: false,
                    callback: function(err, stdout) {
                        if (!stdout) {
                            grunt.fatal('Git command returned no data.');
                        }
                        if (err) {
                            grunt.fatal('Git command failed to execute.');
                        }
                        var allTags = stdout
                            .replace(/-{5}BEGIN PGP SIGNATURE-{5}(.*\n)+?-{5}END PGP SIGNATURE-{5}\n/g, '')
                            .split('\xB6\xB6\xB6');
                        var foundTags = [];
                        allTags.forEach(function(curTag) {
                            if (curTag.length) {
                                var explode = curTag.split('|||');
                                if (explode[0] && explode[1] && explode[2]) {
                                    foundTags.push({
                                        tag: explode[0].trim(),
                                        hash: explode[1].trim(),
                                        message: explode[2].trim().split('\n'),
                                        previous: (foundTags.length ? foundTags.slice(-1)[0].tag : null)
                                    });
                                }
                            }
                        });
                        if (foundTags.length) {
                            grunt.config('all_tags', foundTags.reverse()); // LIFO
                        } else {
                            grunt.fatal('Could not get existing tags information');
                        }
                    }
                },
                'commit_changelog': {
                    cmd: function() {
                        var file = grunt.config('changesmd_file');
                        if (!file) {
                            grunt.fatal('Missing file path.');
                        }
                        var path = file.slice(0, -24); // slices 'sickrage-news/CHANGES.md' (len=24)
                        if (!path) {
                            grunt.fatal('path = "' + path + '"');
                        }
                        var pushCmd = 'git push origin master';
                        if (grunt.option('no-push')) {
                            grunt.log.warn('Pushing with --dry-run ...'.magenta);
                            pushCmd += ' --dry-run';
                        }
                        return ['cd ' + path, 'git commit -asm "Update changelog"', 'git fetch origin', 'git rebase',
                            pushCmd].join(' && ');
                    },
                    stdout: true
                }
            }
        });
    
        /****************************************
        *  Internal tasks                       *
        *****************************************/
        grunt.registerTask('_get_next_tag', '(internal) do not run', function() {
            grunt.config.requires('last_tag');
            var lastTag = grunt.config('last_tag');
    
            var lastPatch = lastTag.match(/[0-9]+$/)[0];
            lastTag = grunt.template.date(lastTag.replace(/^v|-[0-9]*-?$/g, ''), 'yyyy.mm.dd');
            var today = grunt.template.today('yyyy.mm.dd');
            var patch = lastTag === today ? (parseInt(lastPatch) + 1).toString() : '1';
    
            var nextTag = 'v' + today + '-' + patch;
            grunt.log.ok(('Creating tag ' + nextTag).green);
            grunt.config('next_tag', nextTag);
        });
    
        grunt.registerTask('_genchanges', "(internal) do not run", function() {
            // actual generate changes
            var allTags = grunt.config('all_tags');
            if (!allTags) {
                grunt.fatal('No tags information was received.');
            }
    
            var file = grunt.config('changesmd_file'); // --file=path/to/sickrage.github.io/sickrage-news/CHANGES.md
            if (!file) {
                grunt.fatal('Missing file path.');
            }
    
            var contents = "";
            allTags.forEach(function(tag) {
                contents += '### ' + tag.tag + '\n';
                contents += '\n';
                if (tag.previous) {
                    contents += '[full changelog](https://github.com/SickRage/SickRage/compare/' +
                        tag.previous + '...' + tag.tag + ')\n';
                }
                contents += '\n';
                tag.message.forEach(function (row) {
                    contents += row
                        // link issue numbers, style links of issues and pull requests
                        .replace(/([\w\d\-.]+\/[\w\d\-.]+)?#(\d+)|https?:\/\/github.com\/([\w\d\-.]+\/[\w\d\-.]+)\/(issues|pull)\/(\d+)/gm,
                            function(all, repoL, numL, repoR, typeR, numR) {
                                if (numL) { // repoL, numL = user/repo#1234 style
                                    return '[' + (repoL ? repoL : '') + '#' + numL + '](https://github.com/' +
                                    (repoL ? repoL : 'SickRage/SickRage') + '/issues/' + numL + ')';
                                } else if (numR) { // repoR, type, numR = https://github/user/repo/issues/1234 style
                                    return '[#' + numR + ']' + '(https://github.com/' +
                                        repoR + '/' + typeR + '/' + numR + ')';
                                }
                        })
                        // shorten and link commit hashes
                        .replace(/([a-f0-9]{40}(?![a-f0-9]))/g, function(sha1) {
                            return '[' + sha1.substr(0, 7) + '](https://github.com/SickRage/SickRage/commit/' + sha1 + ')';
                        })
                        // remove tag information
                        .replace(/^\([\w\d\s,.\-+/>]+\)\s/gm, '')
                        // remove commit hashes from start
                        .replace(/^[a-f0-9]{7} /gm, '')
                        // style messages that contain lists
                        .replace(/( {3,}\*)(?!\*)/g, '\n  -')
                        // escapes markdown __ tags
                        .replace(/__/gm, '\\_\\_')
                        // add * to the first line only
                        .replace(/^(\w)/, '* $1');
    
                    contents += '\n';
                });
                contents += '\n';
            });
    
            if (contents) {
                grunt.file.write(file, contents);
                return true;
            } else {
                grunt.fatal('Received no contents to write to file, aborting');
            }
        });
    };