Issue to node-sass and Docker

25,917

Solution 1

The support for Node.js 7 (for Linux and OSX) seems to have been added in node-sass v3.7.0. Make sure you use a version equal to or newer than this.

Either you can update your Dockerfile:

FROM node:7.2.1
RUN apt-get update -qq && apt-get install -y build-essential
RUN apt-get install -y libpq-dev postgresql-client
ENV APP_HOME /my_app
RUN mkdir $APP_HOME
WORKDIR $APP_HOME
ADD package.json .

# Add the two entries below
RUN mkdir -p node_modules/node-sass/vendor/linux-x64-51
RUN curl -L https://github.com/sass/node-sass/releases/download/v4.5.0/linux-x64-51_binding.node -o node_modules/node-sass/vendor/linux-x64-51/binding.node

RUN npm install
RUN npm rebuild node-sass
ADD . .
CMD [ "npm", "start" ]
EXPOSE 3000

Or you can download the binding locally and then build from the Dockerfile without any modification:

cd /path/to/node_app/node_modules
mkdir -p node-sass/vendor/linux-x64-51
curl -L https://github.com/sass/node-sass/releases/download/v4.5.0/linux-x64-51_binding.node -o node-sass/vendor/linux-x64-51/binding.node

Keep an eye out for different versions for the pre-compiled native bindings at: https://github.com/sass/node-sass/releases

Solution 2

I have encountered this problem a few times when running my apps within Docker. I believe the problem arises when mounting my local app directory with Docker as this includes the node_modules folder which have been built in a different environment.

My local setup is Mac OSX while Docker is running in linux.

The solution for me was to add a .dockerignore file in the root of my app and add node_modules as per this suggestion https://stackoverflow.com/a/55657576/3258059

I then needed to trigger yarn install to run again in my docker container and also ran the node-sass rebuild command with the docker container: docker-compose run web npm rebuild node-sass where web is the name of my docker container.

I suspect my Dockerfile may have a bit of bloat but I will add in case it helps someone:

FROM ruby:2.3.7
ENV BUNDLER_VERSION=1.17.3
RUN gem install bundler -v "$BUNDLER_VERSION" --no-document
RUN curl -sL https://deb.nodesource.com/setup_8.x | bash -  
RUN curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | apt-key add -   
RUN echo "deb https://dl.yarnpkg.com/debian/ stable main" | tee /etc/apt/sources.list.d/yarn.list   
RUN apt-get update -qq && apt-get install -y nodejs yarn build-essential libpq-dev postgresql-client
RUN mkdir /myapp
WORKDIR /myapp
COPY Gemfile /myapp/Gemfile
COPY Gemfile.lock /myapp/Gemfile.lock
ENV RAILS_ENV docker    
ENV WEBPACKER_DEV_SERVER_PUBLIC localhost:3035  
ENV WEBPACKER_DEV_SERVER_HOST localhost 
COPY package.json *yarn* ./
RUN bundle install
RUN yarn install
COPY . /myapp

# Add a script to be executed every time the container starts.
COPY entrypoint.sh /usr/bin/
RUN chmod +x /usr/bin/entrypoint.sh
ENTRYPOINT ["entrypoint.sh"]
EXPOSE 3000

# Start the main process.
CMD ["rails", "server", "-b", "0.0.0.0"]

and my docker-compose.yml

version: '3'
services:
  db:
    image: postgres
    ports: 
      - "5433:5432"
    volumes:
      - ./tmp/db:/var/lib/postgresql/data
  web:
    build: .
    environment: 
      - RAILS_ENV=development
      - NODE_ENV=development
      - WEBPACKER_DEV_SERVER_PUBLIC=localhost:3035
      - WEBPACKER_DEV_SERVER_HOST=localhost
    command: bash -c "rm -f tmp/pids/server.pid && bundle exec rails s -p 3000 -b '0.0.0.0'"
    volumes:
      - .:/myapp
    ports:
      - "3000:3000"
    depends_on:
      - db
      - redis
      #- elasticsearch
  redis:
    image: redis:alpine
Share:
25,917
jubeless
Author by

jubeless

Updated on October 07, 2021

Comments

  • jubeless
    jubeless over 2 years

    I'm attempting to dockerise my node application. My current application is a nodejs express server with postgresql. ExpressJS uses node-sass-middleware to handle the sass assets. When I run node and postgresql locally on my OSX machine everything works fine. When I try to run the app with docker-compose I get a "Missing Binding error"

    Here is my Dockerfile:

    FROM node:7.2.1
    RUN apt-get update -qq && apt-get install -y build-essential
    RUN apt-get install -y libpq-dev postgresql-client
    ENV APP_HOME /my_app
    RUN mkdir $APP_HOME
    WORKDIR $APP_HOME
    ADD package.json .
    RUN npm install
    RUN npm rebuild node-sass
    ADD . .
    CMD [ "npm", "start" ]
    EXPOSE 3000
    

    Here is my docker-compose.yml file:

    version: '2'
    services:
        db:
          image: postgres:9.6.1
          ports:
            - '5432:5432'
        web:
          build: . # use the Dockerfile next to this file
          volumes:
            - .:/my_app
          ports:
            - "3000:3000"
          depends_on:
            - db
    

    When I run docker-compose up I still get the following error:

    web_1  | [nodemon] 1.11.0
    web_1  | [nodemon] to restart at any time, enter `rs`
    web_1  | [nodemon] watching: *.*
    web_1  | [nodemon] starting `node ./bin/www`
    web_1  | /my_app/node_modules/node-sass/lib/binding.js:15
    web_1  |       throw new Error(errors.missingBinary());
    web_1  |       ^
    web_1  |
    web_1  | Error: Missing binding /my_app/node_modules/node-sass/vendor/linux-x64-51/binding.node
    web_1  | Node Sass could not find a binding for your current environment: Linux 64-bit with Node.js 7.x
    web_1  |
    web_1  | Found bindings for the following environments:
    web_1  |   - OS X 64-bit with Node.js 7.x
    web_1  |
    web_1  | This usually happens because your environment has changed since running `npm install`.
    web_1  | Run `npm rebuild node-sass` to build the binding for your current environment.
    web_1  |     at module.exports (/my_app/node_modules/node-sass/lib/binding.js:15:13)
    web_1  |     at Object.<anonymous> (/my_app/node_modules/node-sass/lib/index.js:14:35)
    web_1  |     at Module._compile (module.js:571:32)
    web_1  |     at Object.Module._extensions..js (module.js:580:10)
    web_1  |     at Module.load (module.js:488:32)
    web_1  |     at tryModuleLoad (module.js:447:12)
    web_1  |     at Function.Module._load (module.js:439:3)
    web_1  |     at Module.require (module.js:498:17)
    web_1  |     at require (internal/module.js:20:19)
    web_1  |     at Object.<anonymous> (/my_app/node_modules/node-sass-middleware/middleware.js:3:12)
    web_1  |     at Module._compile (module.js:571:32)
    web_1  |     at Object.Module._extensions..js (module.js:580:10)
    web_1  |     at Module.load (module.js:488:32)
    web_1  |     at tryModuleLoad (module.js:447:12)
    web_1  |     at Function.Module._load (module.js:439:3)
    web_1  |     at Module.require (module.js:498:17)
    web_1  | [nodemon] app crashed - waiting for file changes before starting...
    

    I though by adding RUN npm rebuild node-sass to the Dockerfile, it would build the correct binding for the OS in the docker container... But it does not seem to work.

    Any thoughts?