Hugo picking up file changes via saved file in editor, but not via gulp (macOS)

Hi,

I have a simple gulp file that builds CSS from Sass. I’m running this on macOS 10.13.6. Just in case it’s relevant, this is the part of the gulpfile that takes care of that.

// Clean the build and dist directories
gulp.task('clean', function() {
  return del(['build/**/*'],{
    // Remove hidden (dot) files
    dot: true,
    force: true
  });
});


// Compile Sass into CSS
gulp.task('build:sass', ['clean'], function() {
    return gulp.src('src/scss/*.scss')
            .on('error', gutil.log)
        .pipe(sass({
            includePaths: [
                'node_modules'
            ]
        }))
            .on('error', function(err) {
                sass.logError.bind(this)(err);
            })
        .pipe(gulp.dest('build/css'));
});

This works fine and I can verify that changing Sass files have an impact on the final CSS.

However, when I run hugo server --ignoreCache --noHTTPCache --forceSyncStatic, Hugo does not pick up changes to the resulting CSS file.

If however, I open my editor (I’m using VSCode, but I doubt that should matter) and hit Cmd + S, Hugo picks the change up immediately and reloads the browser as expected.

Another test you can do to replicate this behaviour is to open up a CSS file in vi, change some content, then write the file, without quitting vi (:w) – here the file on disk will be written just fine, but Hugo will not pick it up. However, if you write and quit vi (:wq!) Hugo will pick up the change immediately.

I’m pretty positive this used to work fine before (it’s been a couple of months since I’ve used Hugo), naturally I updated tot he latest version before reporting this. Here are the versions of what I’m running (I have managed to reproduce this behaviour on two machines running the same versions of Hugo):

hugo version
Hugo Static Site Generator v0.46/extended darwin/amd64 BuildDate: unknown

Just in case it’s relevant:

node -v
v10.7.0
gulp -v
[13:09:33] CLI version 2.0.1

Any help would be much appreciated.

Thanks,
Ian

Did you try

.pipe(gulp.dest('static/css'));

Hugo server is picking up changed files from there.

1 Like

Hey @Leo_Merkel thanks for your reply. Sorry, I forgot to mention that I have another Gulp task moving those files to static/assets:

// Copy static assets
gulp.task('build', ['clean',
                    'build:sass',
                    'build:js',
                    'build:img',
                    'build:ico',
                    'build:fnt'], function() {
    return gulp.src('build/**/*')
        .on('error', gutil.log)
    .pipe(gulp.dest('static/assets'));
});

Hmm, that’s weird. I used to generate my CSS with Gulp through Sass but nowadays using the new Hugo pipeline. I got rid of gulp completely. But even if I used Gulp it was working with Hugo. Do you have a link to a repo?

Yeah it might just be the excuse I need to try out Hugo’s pipelines :laughing:. Unfortunately, no this site is not yet online, nor is the repo public.

The thing that is puzzling me is that it’s also happening with vi. Really not sure what’s going on. Not sure if perhaps I’m hitting some edge case.

Here’s my gulpfile which I used to work with before I switched to Hugo’s pipeline. Perhaps this helps:

import gulp from 'gulp';
import autoprefixer from 'gulp-autoprefixer';
import cleanCss from 'gulp-clean-css';
import {spawn} from 'child_process';
import hugoBin from 'hugo-bin';
import BrowserSync from 'browser-sync';
import concat from 'gulp-concat';
import hash from 'gulp-hash';
import htmlmin from 'gulp-htmlmin';
import imagemin from 'gulp-imagemin';
import del from 'del';
import notify from 'gulp-notify';
import rename from 'gulp-rename';
import sass from 'gulp-sass';
import sourcemaps from 'gulp-sourcemaps';
import uglify from 'gulp-uglify';
import gzip from 'gulp-gzip';
import environments from 'gulp-environments';

import confPlugins from './gulp/gulp-plugins.json';
import jsModules   from './src/assets/js-user/modules.js';

/***************************************************************************************
** Constants
***************************************************************************************/
// https://www.npmjs.com/package/gulp-environments
const development = environments.development;
const staging     = environments.make("staging");
const production  = environments.production;

const browserSync = BrowserSync.create();

const hugo = {
  src: './src/assets',
  site: './site',
  static: './site/static',
  build: './build',
}

const paths = {
  styles: {
    src: hugo.src + '/scss',
    dest: hugo.static + '/css',
  },
  scripts: {
    src: hugo.src + '/js-user',
    dest: hugo.static + '/js',
  },
  images: {
    src: [
      hugo.src + '/images/*.{jpg,png,gif,svg}',
      hugo.src + '/img/*.{jpg,png,gif,svg}',
    ],
    dest: hugo.static + '/images/',
    optim: hugo.static + '/images/*',
  },
  html: {
    src: hugo.build + '/public/**/*.html',
    dest: hugo.build + '/public',
  },
  partials: hugo.site + '/layouts/partials/',
}

const sassPaths = [
  // Checkbox
  './node_modules/checkbox.css/dist/scss',

  // Owl Carousel
  './node_modules/owl.carousel/src/scss',
  
  // Animate
  './node_modules/animate.css',
  
  // Slider Pro
  // Einzelne Dateien können importiert werden:
  // ./node_modules/slider-pro/src/css
  './node_modules/slider-pro/dist/css',
  
  // Foundation 6 Pro
  './node_modules/foundation-sites/scss',
  './node_modules/motion-ui/src',

  // Hover
  'node_modules/hover.css/scss',
  
  // Vendor
  'src/assets/scss/vendor',
  'src/assets/scss/vendor/foundation-blocks',
];

const breakpoints = {
  small: 0, // or wider
  medium: 640, // or wider
  large: 1024, // or wider
}


/***************************************************************************************
** Minimize HTML
***************************************************************************************/
function html () {
  return gulp
    .src(paths.html.src)
    .pipe(htmlmin(confPlugins.htmlmin))
    .pipe(gulp.dest(paths.html.dest))
}

gulp.task(
  'html-min',
  gulp.series(html)
)


/***************************************************************************************
** GZIP
***************************************************************************************/
function allgzip () {
  return gulp.src(hugo.build + '/public/**/*.{html,xml,json,css,js}')
    .pipe(gzip())
    .pipe(gulp.dest(hugo.build + '/public'));
}

gulp.task(
  'prod-gzip',
  gulp.series(allgzip)
)


/***************************************************************************************
** CSS
***************************************************************************************/
function css () {
  return gulp
    .src(paths.styles.src + '/main.scss')
    .pipe(development(sourcemaps.init()))
    .pipe(sass({
      includePaths: sassPaths
      })
      .on('error', sass.logError))
    // Autoprefixer only in production mode
    .pipe(production(autoprefixer({ browsers: ['>1%'] })))
    // CleanCSS only in production mode
    .pipe(production(cleanCss({compatibility: 'ie8'})))
    .pipe(development(sourcemaps.write('.')))
    // Hashes only in production mode
    .pipe(production(hash()))
    .pipe(gulp.dest(paths.styles.dest))
    .pipe(browserSync.stream())
    // Hashes only in production mode
    .pipe(production(hash.manifest('css.json')))
    // Store Map in data folder
    .pipe(production(gulp.dest('./site/data/hashes')))
  }

gulp.task(
  'scss',
  gulp.series(css)
)


/***************************************************************************************
** JavaScript
***************************************************************************************/
function js (cb) {
  return gulp
  .src(jsModules.js)
  .pipe(concat('user.js'))
  // Uglify only in production mode
  .pipe(production(uglify()))
  // Hashes only in production mode
  .pipe(production(hash()))
  .pipe(gulp.dest(paths.scripts.dest))
  .pipe(browserSync.stream())
  // only in production mode
  .pipe(production(hash.manifest('user.json')))
  // Store Map in data folder
  .pipe(production(gulp.dest('./site/data/hashes')))
}

function jsvendor (cb) {
  return gulp
  .src(jsModules.libs)
  .pipe(concat('vendor.js'))
  .pipe(production(uglify()))
  .pipe(production(hash()))
  .pipe(gulp.dest(paths.scripts.dest))
  .pipe(browserSync.stream())
  .pipe(production(hash.manifest('vendor.json')))
  .pipe(production(gulp.dest('./site/data/hashes')))
}

gulp.task(
  'js-all',
  gulp.series(js, jsvendor)
)


/***************************************************************************************
** Image Copy
***************************************************************************************/
function imgcopy (done) {
  return gulp
  .src(paths.images.src)
  .pipe(gulp.dest(paths.images.dest))
  done()
}

gulp.task(
  'img-copy',
  gulp.series(imgcopy)
)


/***************************************************************************************
** Hugo
***************************************************************************************/
// Hugo arguments
const hugoArgsDefault = ["-d", "../build/dev", "-s", "site", "-v"];
const hugoArgsPreview = ["--buildDrafts", "--buildFuture"];
const hugoArgsPublic  = ["-d", "../build/public", "--config", "site/config.toml,site/config_public.toml"];
const hugoArgsStage   = ["-d", "../build/stage", "--config", "site/config.toml,site/config_stage.toml"];
const hugoArgsLocal   = ["-d", "../build/local", "--config", "site/config.toml,site/config_local.toml"];

// Development Tasks
gulp.task('hugo-development', (cb) => buildSite(cb));
gulp.task('hugo-development-preview', (cb) => buildSite(cb, hugoArgsPreview));

// Production Task
gulp.task('hugo-production', (cb) => buildSite(cb, hugoArgsPublic));

// Stage Task
gulp.task('hugo-stage', (cb) => buildSite(cb, hugoArgsStage));

// Local Task
gulp.task('hugo-local', (cb) => buildSite(cb, hugoArgsLocal));

// Dev Server with BrowserSync
gulp.task('server', () => {
  browserSync.init({
    server: {
      baseDir: "./build/dev"
    },
    open: false
  });
  gulp.watch(paths.scripts.src + "/**/*.js", gulp.series("js-all"));
  gulp.watch(paths.styles.src + "/**/*.scss", gulp.series("scss"));
  gulp.watch(paths.images.src, gulp.series("img-copy"));
  gulp.watch(hugo.site + "/**/*", gulp.series("hugo-development-preview"));
});

// Start Hugo and create site
function buildSite(cb, options, environment = "development") {
  const args = options ? hugoArgsDefault.concat(options) : hugoArgsDefault;

  process.env.NODE_ENV = environment;

  return spawn(hugoBin, args, {stdio: "inherit"}).on("close", (code) => {
    if (code === 0) {
      browserSync.reload();
      cb();
    } else {
      browserSync.notify("Hugo build failed :(");
      cb("Hugo build failed");
    }
  })
};

Now I use the following code in my baseof.html to make use of Hugo’s execellent asset pipeline:

{{/* <!-- CSS --> */}}
{{ $sassIncludes := (slice "node_modules" "assets/scss/vendor" "assets/scss/components") }}
{{ $target := "styles/main.css" }}
{{ if .Site.IsServer }}
    {{ $cssOpts := (dict "targetPath" $target "enableSourceMap" true "includePaths" $sassIncludes ) }}
    {{ $styles := resources.Get "scss/main.scss" | toCSS $cssOpts }}
    <link rel="stylesheet" href="{{ $styles.Permalink }}" media="screen">
{{ else }}
    {{ $cssOpts := (dict "targetPath" $target "includePaths" $sassIncludes ) }}
    {{ $styles := resources.Get "scss/main.scss" | toCSS $cssOpts | postCSS | minify | fingerprint }}
    <link rel="stylesheet" href="{{ $styles.Permalink }}" integrity="{{ $styles.Data.Integrity }}" media="screen">
{{ end }}
{{/* <!-- JavaScript --> */}}
{{/* <!-- NOTE: Vendor Scripts --> */}}
{{ $jquery := resources.Get "/js/jquery/dist/jquery.js" }}
{{ $bootstrap := resources.Get "/js/bootstrap/dist/js/bootstrap.js" }}
{{ $popper := resources.Get "/js/popper/popper.js" }}
{{/* <!-- NOTE: User Scripts --> */}}
{{ $user := resources.Get "/js/user.js" }}
{{ $scripts := resources.Get "/js/scripts.js" }}

{{ if .Site.IsServer }}
    {{ $script := slice $jquery $bootstrap $popper $scripts $user | resources.Concat "/js/bundle.js" }}
    <script src="{{ $script.Permalink }}"></script>
{{ else }}
    {{ $script := slice $jquery $bootstrap $popper $scripts $user | resources.Concat "/js/bundle.js" | minify | fingerprint }}
    <script src="{{ $script.Permalink }}"></script>
{{ end }}
2 Likes