Как я могу использовать grunt-respecte с grunt-contrib-coffee для компиляции измененных файлов. Coffee?

В моем проекте более 300 файлов CoffeeScript, поэтому для перекомпиляции требуется несколько секунд. Я хотел бы только перекомпилировать измененные файлы CoffeeScript.

Здесь ближайший я пришел до сих пор, но структура папки "frontend-src/coffeescript" копируется из каталога src в каталог dest.

coffee: {
  changed: {
    expand: true,
    cwd: './',
    src: ['<%= grunt.regarde.changed %>'],
    dest: 'public/js/',
    ext: '.js'
  }
},
regarde: {
  coffee: {
    files: 'frontend-src/coffeescript/**/*.coffee',
    tasks: ['coffee:changed', 'livereload']
  }
}

Все это с Grunt 0.4.0. Любая помощь будет принята с благодарностью!

Ответ 1

У меня была такая же проблема. Я решил это, используя событие regarde:file.

Сначала я слушаю измененные файлы с помощью события regarde:file. Это обеспечит конфигурацию для двух задач: clean:coffee, если файлы в исходном местоположении были удалены, и coffee:refresh, если файлы были изменены/добавлены.

Затем задача regarde будет запускать свои задачи, которые запустит refresh:coffee (чтобы не ошибиться с coffee:refresh). Эта задача проверяет, добавлена ​​ли конфигурация для clean:coffee и/или для coffee:refresh, и при необходимости запускайте эти задачи (через функцию grunt.task.run). Если также будет reset флаг, который заставит следующее принятое событие regarde:file снова очистить конфигурацию.

Подробное объяснение:

Прежде всего, regarde config:

 // watch for changed coffeescript files
 coffee: {
    files: 'path/to/coffee/**/*.coffee',
    tasks: ['refresh:coffee', 'livereload']
 },

Затем я слушаю событие regarde:file, где я обновляю списки файлов clean:coffee и coffee:refresh в их конфигурации.

Подайте конфигурацию на основе события regarde:file:

grunt.event.on('regarde:file', function (status, target, filepath) {
   if (resetFlag) {
      // clean file list from previous cycle, so clean clean:coffee and coffee:refresh
      // file lists
      ...

      resetFlag = false;
   } 
   if (status === 'deleted') {
        if (filepath) {
            // calculate filepath destination and  
            // add it to clean:coffee filelist
        }
    } else {
        if (!grunt.file.isDir(filepath)) {
            // add filepath to coffee:refresh filelist
        }
    }
}

Легко обновить конфигурацию с помощью функции grunt.config(). Ниже фрагментов кода для подачи coffee:refresh и clean:coffee.

Добавление конфигурации в coffee:refresh:

var config = grunt.config('coffee') || {};
var value = config.refresh || {};
value.files = value.files || [];
...
var cwd = path.dirname(filepath),
    src = path.basename(filepath),
    dest = cwd.replace('path/to/source', 'path/to/dest');
    value.files.push({
       expand:true,
       src:src,
       dest:dest,
       cwd:cwd,
       ext:'.js'
    });
grunt.config('coffee', config);

Добавление конфигурации в clean:coffee:

    var cwd = path.dirname(filepath),
        src = path.basename(filepath),
        dest = cwd.replace('path/to/source', 'path/to/dest');
        value.src.push(path.join(dest, src.replace('coffee', 'js')));
    // clean only what has been removed
        config = grunt.config('clean') || {};

    config.coffee = value;

    grunt.config('clean', config);

Задача refresh:coffee запускается:

    grunt.registerMultiTask('refresh', 'refreshing the changed file(s)', function () {
        this.requires('regarde');

        var tasks = [];
        var clean = grunt.config('clean');

        // check if there is clean:refresh config available
        if (clean && clean[this.target]) {
            tasks.push('clean:' + this.target);
        }
        var config = grunt.config(this.target);

        // check if there is coffee:refresh config available
        if (config && config.refresh) {
            tasks.push(this.target + ':refresh');
        }
        // run the tasks
        grunt.task.run(tasks);

        // set the resetFlag back to true
        resetFlag = true;
    });

Ответ 2

У меня была эта проблема, и я смог придумать решение для нее, вдохновленное комментариями по этой проблеме: https://github.com/gruntjs/grunt-contrib-watch/issues/14

На самом деле это плагин для grunt-contrib-watch, но он также должен работать на grunt-respecte, поскольку он имеет похожие события.

Идея заключается в привязке обратного вызова к событию watch, в которое вы добавляете новую задачу в конфигурацию grunt с измененным пути к файлу, а затем запустите ее.

Из моего Gruntfile.coffee:

coffee:
    app:
        expand: true
        cwd: 'app/'
        src: ['*.coffee',"**/*.coffee"]
        dest: './public/temp'
        ext: '.js'
watch: 
    coffee:
        files: ['app/**/*.coffee']
        tasks: ['livereload']
        options:
            nospawn: true

grunt.event.on 'watch', (action, filepath) ->       
    cwd = 'app/'
    filepath = filepath.replace(cwd,'')
    grunt.config.set('coffee',
        changed:
            expand: true
            cwd: cwd
            src: filepath
            dest: './public/temp'
            ext: '.js'
    )
    grunt.task.run('coffee:changed')

Носпаунт важен для задачи часов, поэтому он запускает новую задачу перед задачей, связанной с загрузкой. Я почти уверен, что по умолчанию дочерние процессы не порождают.

Ответ 3

grunt.regarde.changed - правильный массив?

Должно src: ['<%= grunt.regarde.changed %>']

be src: '<%= grunt.regarde.changed %>'

Я просмотрел источник grunt-contrib-coffee на секунду, чтобы понять, не может ли он правильно обрабатывать все, что вы ему даете. Похоже, что строковый массив, который вы ему даете, не попадает и не обрабатывается.

Я думаю, что вы случайно проходите: src: [ '[path1, path2, path3, etc]' ]

Если я ухожу из базы, оставьте комментарий, и я удалю этот ответ.