¿QUÉ ES GULP.JS?

Gulp.js es un build system(sistema de construcción) que permite automatizar tareas comunes de desarrollo, tales como la minificación de código JavaScript, recarga del navegador, compresión de imágenes, validación de sintaxis de código y un sin fin de tareas más.
Como verás no importa si eres un desarrollador Front-End, Back-End ó los dos a la vez. Si hoy en día no quieres perder tiempo realizando tareas comunes “manualmente”, es momento de que aprendas a usar un automatizador como Gulp.js.
Adicionalmente Gulp.js está construído con Javascript, funciona sobre Node.js y es Open Source, así que su código fuente lo puedes encontrar en github.

¿POR QUÉ TE DEBERÍA INTERESAR USAR GULP.JS?

Porque hoy en día el flujo de trabajo de un desarrollador se ha vuelto más complejo, usamos muchas herramientas de desarrollo, por lo cual configurar cada tarea y ejecutarla “manualmente” y por separado requiere demasiado tiempo.
Porque Gulp.js no sólo soluciona este problema sino que le aporta mejoras, convirtiendose en una herramienta que tiene prácticamente todo en uno, así mismo nos permite administrar y controlar todas esas tareas en un solo lugar.
Porque Gulp.js prefiere el código sobre la configuración, esto no sólo lo hace muy fácil para escribir tareas, sino también lo hace más simple de leer y mantener.
Porque Gulp.js tiene directrices estrictas para la creación de sus plugins, lo cual asegura que estos sean simples y que funcionen como se espera.
Solo por mencionarlo, una de las directrices es la siguiente: “Un plugin sólo debe hacer una cosa y hacerla bien”. Esta directriz ha sido muy positiva para los usuarios de Gulp.js, lo comprobaremos más adelante cuando veamos como configurar una tarea con Gulp.js.

¿CÓMO FUNCIONA GULP.JS?

Gulp.js utiliza el módulo Stream de Node.js, lo cual lo hace más rápido para construir, a diferencia de Grunt.js.
Gulp.js no necesita escribir archivos y/o carpetas temporales en el disco duro, lo cual supone que realizará las mismas tareas que Grunt.js pero en menor tiempo.
Gulp.js utiliza el método pipe(), este método obtiene todos los datos de un Stream legible(readable) y lo escribe en destino que le indiquemos ó de lo contrario lo entrega o sirve hacia otro pipe.
En la siguiente imagen veremos como Grunt.js manipula los archivos al realizar sus tareas:
grunt-file-manipulation
Y en esta veremos como Gulp.js manipula los archivos al realizar sus tareas:
gulp-file-manipulation
Como podemos ver, aunque los 2 hicieron la misma tarea Gulp.js no escribió archivos temporales en el disco duro. Gulp.js realizó 4 operaciones y en cambio Grunt.js realizó 8 operaciones.
Gulp.js utiliza el poder del paquete Node.js vinyl-fs para leer y escribir Streams.
Gulp.js también utiliza un paquete Node.js para la secuenciación, ejecución de tareas y dependencias en máxima concurrencia, llamado Orchestrator.
Más adelante veremos con mayor detalle como trabajan los Streams de Node.js.

¿CÓMO INSTALAR GULP.JS?

Para instalar Gulp.js el único requisito es tener instalado Node.js. si usas Linux Ubuntu y aún no lo tienes instalado este post te puede servir.

INSTALANDO GULP.JS DE FORMA GLOBAL EN NUESTRO SISTEMA


npm install -g gulp

Si estás usando Linux o Mac, tal vez necesites anteponer la palabra sudo para poder ejecutar este comando con los permisos de administrador, así:

sudo npm install -g gulp

Verificamos que Gulp.js ha sido instalado correctamente.

gulp -v

Si lo tenemos instalado correctamente, nos mostrará lo siguiente:

CLI version 3.8.6
Local version undefined

Donde CLI version es la versión de Gulp.js instalada en nuestro sistema y Local version es la versión de Gulp.js instalada en nuestro proyecto local, pero como aún no hemos creado nuestro proyecto nos saldrá undefined.

¿CÓMO USAR GULP.JS?

Una vez instalado en nuestro sistema estamos listos para crear nuestro primer proyecto usando Gulp.js, nuestro pequeño proyecto concatenará dos archivos .js en uno solo y luego lo minificará. Así que configuraremos 2 tareas(concatenar y minificar), todo esto contenido en una tarea llamada “demo”.
Creamos una carpeta llamada: gulp-primeros-pasos, ingresamos a esa carpeta mediante terminal.
Luego allí dentro creamos nuestro archivo: gulpfile.js, que es el archivo que Gulp.js necesita para saber que tareas realizará y de momento no le podremos ningún contenido.
Luego escribimos lo siguiente(en este punto suponemos que tenemos instalado Node.js):

npm init

Npm nos pedirá los datos de nuestro proyecto, ya que en esta ocasión sólo estamos haciendo un demo. Simplemente presionaremos Enter a todas las preguntas.
Con esto, Npm nos debe haber creado un archivo llamado: package.json, que contendrá algo parecido a lo siguiente:

{
  "name": "gulp-primeros-pasos",
  "version": "0.0.1",
  "description": "Gulp: Primeros pasos",
  "main": "gulpfile.js",
  "author": "jansanchez",
  "license": "MIT"
}

Ahora agregaremos las dependencias de desarrollo a nuestro proyecto, la primera a instalar será: gulp, así que escribimos lo siguiente en nuestra terminal:

npm install --save-dev gulp

Luego instalamos: gulp-concat

npm install --save-dev gulp-concat

Y finalmente instalamos: gulp-uglify

npm install --save-dev gulp-uglify

Tengamos en cuenta que sí no agregamos el parámetro: –save-dev, entonces Npm no agregará este paquete como una dependencia de desarrollo de nuestro proyecto y mucho menos lo agregará a nuestro archivo package.json.
Como podremos observar, nuestro archivo package.json a cambiado y debería contener algo parecido a lo siguiente:

{
  "name": "gulp-primeros-pasos",
  "version": "0.0.1",
  "description": "Gulp: Primeros pasos",
  "main": "gulpfile.js",
  "author": "jansanchez",
  "license": "MIT",
  "devDependencies": {
    "gulp": "^3.8.7",
    "gulp-concat": "^2.3.4",
    "gulp-uglify": "^0.3.1"
  }
}

Como vemos, se agregó la clave devDependencies y en su interior se comienzan a guardar nuestras dependencias de desarrollo y la versión que hemos instalado localmente.
Luego vamos a crear las siguientes carpetas y archivos:
Creamos la carpeta js y dentro de esta carpeta crearemos la carpeta source.
Dentro de la carpeta source crearemos el archivo 1.js y le agregaremos el siguiente contenido:

// contenido del archivo 1.js

var sumar = function (a, b){
  return a + b;
};

Nuevamente dentro de la carpeta source crearemos el archivo 2.js y le agregaremos el siguiente contenido:

// contenido del archivo 2.js

var restar = function (a, b){
  return a - b;
};

Ahora vamos a poner el siguiente contenido a nuestro archivo gulpfile.js:

/*
* Dependencias
*/
var gulp = require('gulp'),
  concat = require('gulp-concat'),
  uglify = require('gulp-uglify');

/*
* Configuración de la tarea 'demo'
*/
gulp.task('demo', function () {
  gulp.src('js/source/*.js')
  .pipe(concat('todo.js'))
  .pipe(uglify())
  .pipe(gulp.dest('js/build/'))
});

Con esto ya tenemos todo configurado, así que para ponerlo a prueba en nuestra terminal escribimos lo siguiente:

gulp demo

Y si todo anda bien, nos dará el siguiente mensaje:

[11:23:09] Using gulpfile ~/htdocs/gulp-primeros-pasos/gulpfile.js
[11:23:09] Starting 'demo'...
[11:23:09] Finished 'demo' after 9 ms

El cual nos indica que la tarea demo se ejecutó con éxito en 9 milisegundos.
Para comprobar si se ejecutaron las 2 tareas requeridas, nos dirigimos a la carpeta js/build y abrimos el archivo todo.js y nos debe mostrar el siguiente contenido:

var sumar=function(r,n){return r+n},restar=function(r,n){return r-n};

Como vemos, con unas simples y limpias lineas de código hemos realizado 2 tareas de desarrollo comunes(concatenar y minificar archivos .js).
Para esto usamos 2 plugins de Gulp.js, la buena noticia es que actualmente existen más de 1800 plugins para Gulp.js, así que imagínate todo lo que puedes automatizar en tu flujo de trabajo con Gulp.js.

ANALIZANDO EL GULPFILE.JS

Ahora vamos a analizar el código que escribimos en nuestro gulpfile.js para entender un
poco más como funciona Gulp.js.
Primero para llevar a cabo las tareas que deseamos, requerimos los siguientes paquetes: gulp, gulp-concat y gulp-uglify, así:

/*
* Dependencias
*/
var gulp = require('gulp'),
  concat = require('gulp-concat'),
  uglify = require('gulp-uglify');

API

Gulp.js tiene una pequeña API, esto te permitirá aprender Gulp.js rápidamente.

GULP.TASK()

Con el método gulp.task() definimos una tarea, este método toma 3 argumentos: el nombre de la tareala ó las tareas de las que depende esta tarea y la función que ejecutará al llamar esta tarea.
En nuestro ejemplo sólo usamos 2 parámetros: el nombre y la función, así:

/*
* Configuración de la tarea 'demo'
*/
gulp.task('demo', function () {
  // Contenido de la tarea 'demo'
});

Con lo cual declaramos “demo” como nombre de la tarea y dentro escribimos lo que deseamos que haga nuestra tarea.
Así que si queremos llamar esta tarea tan solo escribimos en nuestra terminal:

gulp demo

LISTA DE TAREAS

Una tarea también puede actuar como una lista de tareas, supongamos que queremos definir una tarea que corra otras 3 tareas por ejemplo: imágenescss y js. Entonces escribiríamos lo siguiente:

gulp.task('estaticos', ['imagenes', 'css', 'js']);

Lo que quiere decir que al ejecutar la tarea “estaticos” con el comando gulp estaticos se ejecutarán estas 3 tareas.
El detalle es que estas tareas correran asíncronamente, osea que correrán todas juntas al mismo tiempo sin ningún orden de ejecución.

TAREAS COMO DEPENDENCIA

Si deseamos que una tarea se ejecute sí y solo sí otra tarea haya terminado antes, entonces podemos hacer lo siguiente:

gulp.task('css', ['imagenes'], function () {
  /*
  * Aquí iría el contenido de mi tarea 'css'
  * Que se ejecutará solo cuando la tarea
  * 'imagenes' haya terminado.
  */
});

Entonces, cuando corramos la tarea “css”, Gulp.js ejecutará primero la tarea “imagenes”, esperará a que esta tarea termine y luego recién ejecutará la tarea “css”.

TAREA POR DEFECTO(DEFAULT)

Gulp.js nos permite definir una tarea por defecto, que corra tan solo al escribir el comando gulp. Esto se puede hacer tan solo poniendole a la tarea el nombre de default, así:

gulp.task('default', function () {
  /*
  * Código de nuestra tarea por defecto.
  */
});

Y claro, también puedes hacer que tu tarea por defecto sea una lista de tareas, así:

gulp.task('default', ['css', 'js']);

Esta tarea ejecutará las tarea ‘css’ y ‘js’, tan solo escribiendo en nuestra terminal:

gulp

GULP.SRC()

El método gulp.src() toma como parámetro un valor glob es decir, una cadena que coincida con uno o más archivos usando los patrones que usa el intérprete de comandos de unix(shell) y retorna un stream que puede ser “pipeado” a un plugin adicional ó hacia el método gulp.dest().
Este parámetro puede ser una cadena o una colección(Array) de valores glob.
Gulp.js usa el paquete de Node.js node-glob para obtener los archivos especificados en él ó los globs ingresados.

EJEMPLOS DE GLOBS

  • js/source/1.js coincide exactamente con el archivo.
  • js/source/*.js coincide con los archivos que terminen en .js dentro de la carpeta js/source.
  • js/**/*.js coincide con los archivos que terminen en .js dentro de la carpeta js y dentro de todas sus sub-carpetas.
  • !js/source/3.js Excluye especificamente el archivo 3.js.
  • static/*.+(js|css) coincide con los archivos que terminen en .js ó .css dentro de la carpetastatic/
Existen más patrones, los puedes revisar desde la documentación de la librería minimatch.
Así que tienes la oportunidad de realizar todas las combinaciones posibles, según lo necesites.
Como en nuestra demo, necesitabamos encontrar todos los archivos que terminen en .js dentro de la carpeta js/source, así:

gulp.src('js/source/*.js')

Cada vez que Gulp.js encuentre un archivo que coincida con nuestro patrón, lo irá metiendo dentro de un Stream, que será como una colección de archivos. Claro, respetando las propiedades de cada archivo(ruta, etc).
Entonces podemos decir que tendremos todos esos archivos con sus respectivas propiedades dentro de un Stream, Este Stream puede ser manipulado por Gulp.js.

EL MÉTODO PIPE() DE NODE.JS

El método pipe() puede leer, ayudar a transformar y grabar los datos de un Stream.
Es por eso que en nuestro ejemplo usamos el método pipe() 3 veces.
La primera vez lo usamos para leer el Stream y se lo pasamos al plugin “concat” para que este realize la concatenación y así transforme los datos del Stream, así:

.pipe(concat('todo.js'))

La segunda vez lo usamos para leer los datos actuales(js concatenado) y se lo pasamos al plugin “uglify”, para que realize la minificación del archivo concatenado. Todo esto sin escribir en el disco ningún archivo temporal, así:

.pipe(uglify())

La tercera vez se lo pasamos a el método gulp.dest(), así que veamos que hace este método.

GULP.DEST()

Canaliza y escribe archivos desde un Stream, por lo que puede canalizar a varias carpetas. Creará las carpetas que no existan y retornará el Stream, por si deseamos realizar alguna acción más.
En pocas palabras, sirve para escribir los datos actuales de un Stream.
Y en nuestro ejemplo lo usamos así:

.pipe(gulp.dest('js/build/'))

Con lo cual escribimos los datos resultantes del Stream dentro de la carpeta js/build/.
El código final nos quedó así:

/*
* Dependencias
*/
var gulp = require('gulp'),
  concat = require('gulp-concat'),
  uglify = require('gulp-uglify');

/*
* Configuración de la tarea 'demo'
*/
gulp.task('demo', function () {
  gulp.src('js/source/*.js')
  .pipe(concat('todo.js'))
  .pipe(uglify())
  .pipe(gulp.dest('js/build/'))
});

Así como realizé 2 tareas consecutivas, con Gulp.js se pueden realizar muchas más.

GULP.WATCH()

Ver archivos y hacer algo cuando se modifique un archivo. Esto siempre devuelve un EventEmitter que emite los eventos de cambio.
Tiene 2 formas de usar:
gulp.watch(glob, tareas) ó gulp.watch(glob, callback).

gulp.watch('js/source/*.js', ['js']);

Con lo cual, cada vez que se modifique un archivo .js que se encuentre dentro de la carpeta js/source/automáticamente se ejecutará la tarea js.

gulp.watch('js/source/*.js', function(){
  /*
  * Aquí iría el código de la acción que deseas realizar,
  * Cuando hayan cambios en dichos archivos.
  *
  * También podrías ejecutar una tarea mediante el método
  * gulp.start('js')
  *
  * Pero este método no es oficial, le pertenece al
  * paquete 'Orchestrator' ya que Gulp hereda los 
  * métodos de 'Orchestrator'.
  */  
});

Con esto ya podemos usar el API de Gulp.js con mucha facilidad.

PLUGINS

Como dije anteriormente Gulp.js tiene alrededor de 1800 plugins “oficiales” y otros cuantos no-oficiales, los plugins “oficiales” los puedes encontrar en la página de plugins de Gulp.js o buscando con la palabra clave gulpplugin en npm.
Actualmente existen plugins para casí todo lo que se necesita al desarrollar, si deseas puedes pasar por nuestro post sobre los plugins más usados de Gulp.js y sus configuraciones básicas.
La mayoría de los plugins son bastante fáciles de usar, tienen una buena documentación y se ejecutan de la misma forma (mediante la canalización de un Stream de objetos de archivo). Estos plugins normalmente modifican los archivos (aunque algunos, como validadores, no lo harán) y devuelven los archivos nuevos que se pasarán al siguiente plugin en un Stream.
Por ejemplo digamos que en ejemplo que teníamos quiero agregarle la funcionalidad de validación de sintaxis, entonces haríamos lo siguiente:

npm install --save-dev gulp-jshint

Y luego nuestro gulpfile.js quedaría así:

/*
* Dependencias
*/
var gulp = require('gulp'),
  jshint = require('gulp-jshint'),
  concat = require('gulp-concat'),
  uglify = require('gulp-uglify');

/*
* Configuración de la tarea 'demo'
*/
gulp.task('demo', function () {
  gulp.src('js/source/*.js')
    .pipe(jshint())
    .pipe(jshint.reporter('default'))
    .pipe(concat('todo.js'))
    .pipe(uglify())
    .pipe(gulp.dest('js/build/'))
});

Solo agregé 3 lineas especificas a mi gulpfile.js y ya estoy realizando otra tarea más sobre los mismos archivos, es por eso que es más simple que Grunt.js, donde no puedes codificar sino que tienes que poner en el objeto JSON tu configuración, lo cual te limita mucho y hace que generes un archivo de configuración muy grande y siempre llega a verse muy desordenado.
Dejo en github el repositorio del ejemplo que trabajamos en este artículo.

SOBRE LOS PIPES EN UNIX

Para entender aun más el proceso Gulp.js a continuación veamos como funcionan los pipes en Unix.
En unix, los comandos bash pueden “pipear” o “canalizar” la salida de otro comando bash, permitiendo operar esa información antes de ser devuelta, veamos un ejemplo práctico.
Por ejemplo si deseamos averiguar nuestra MAC address en unix ejecutamos el siguiente comando:

ifconfig

Lo cual nos devuelve toda esta información:

eth0
  inet addr:192.111.77.27  Bcast:192.111.77.255
  Link encap:Ethernet  HWaddr 00:1b:00:bb:1c:ab
  Mask:255.255.255.0 Scope:Link
  collisions:0 txqueuelen:1000

Pero nosotros no queremos toda esa información, sino solamente nuestra MAC address, entonces aquí podemos utilizar el poder de los pipes para lograr esto con el siguiente código.

ifconfig | grep "HWaddr"

Ahora sí obtuvimos especificamente lo que necesitabamos:

HWaddr 00:1b:00:bb:1c:ab Link encap:Ethernet

ACLARACIONES

  • El comando ifconfig, nos devuelve toda la información de red en nuestro ordenador.
  • El comando grep, busca las líneas que contengan una coincidencia con un patrón que le pasemos.
  • pipe = |

EXPLICACIÓN DEL PROCESO

El proceso fue el siguiente:
  1. El comando ifconfig generó una respuesta.
  2. Antes que sea escrita “canalizamos” esa respuesta usando un pipe.
  3. A esa respuesta le aplicamos el comando grep con el patrón: “HWaddr”.
  4. Y finalmente nos devuelve sólo la MAC address.
Todo esto se hizo directamente sobre la respuesta inicial y sin almacenarla, ni escribirla en disco de manera temporal.
De la misma forma trabaja Gulp.js, canalizando el stream resultante de una función o plugin y pasando esa respuesta para ser operada por una nueva función y así sucesivamente.

¿EXISTEN OTROS SISTEMAS DE CONSTRUCCIÓN HECHOS CON JAVASCRIPT?

Claro que sí, a continuación listaré los que me parecen más interesantes:
Si ya se animaron a usar Gulp.js entonces les recomiendo leer nuestro artículo Plugins más usados de Gulp.js y sus configuraciones básicas. Para crear tareas en Gulp.js, comenzar a automatizar sus proyectos y empezar a ahorrar tiempo.
Espero que este artículo les haya servido para conocer Gulp.js, si tienen alguna duda, consulta ó aclaración solo háganla dejando sus comentarios, que estaremos prestos a ayudarlos.
Aquí en Frontend Labs estamos dispuestos a compartir el conocimiento, a enseñar y aprender!.
Si el artículo les gustó, por favor compartanlo en las redes sociales. De esa manera nos ayudan muchísimo! las opciones para compartir se encuentran aquí abajo, muchas gracias.
SHARE

Milan Tomic

Hi. I’m Designer of Blog Magic. I’m CEO/Founder of ThemeXpose. I’m Creative Art Director, Web Designer, UI/UX Designer, Interaction Designer, Industrial Designer, Web Developer, Business Enthusiast, StartUp Enthusiast, Speaker, Writer and Photographer. Inspired to make things looks better.

  • Image
  • Image
  • Image
  • Image
  • Image
    Blogger Comment
    Facebook Comment

0 comentarios:

Publicar un comentario