From af1344fa03ac6598f3611cf9d0e90ae64d9bf494 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Leon=20A=2E=20Okida=20Gon=C3=A7alves?= <laog19@inf.ufpr.br>
Date: Fri, 3 Mar 2023 11:09:46 -0300
Subject: [PATCH] Create v2 version, adapt app.js to use both versions

---
 src/libs/app.js                               |    8 +-
 src/libs/{routes => routes_v1}/api.js         |   70 +-
 src/libs/{routes => routes_v1}/auxiliar.js    |    0
 src/libs/{routes => routes_v1}/city.js        |    0
 src/libs/{routes => routes_v1}/class.js       |    0
 src/libs/{routes => routes_v1}/classCount.js  |    0
 src/libs/{routes => routes_v1}/classroom.js   |    0
 .../{routes => routes_v1}/classroomCount.js   |    0
 src/libs/{routes => routes_v1}/courseCount.js |    0
 .../{routes => routes_v1}/courseStudents.js   |    0
 src/libs/{routes => routes_v1}/cub.js         |    0
 .../dailyChargeAmount.js                      |    0
 src/libs/{routes => routes_v1}/disciplines.js |    0
 .../distributionFactor.js                     |    0
 src/libs/{routes => routes_v1}/downloads.js   |    0
 .../{routes => routes_v1}/educationYears.js   |    0
 .../educationalBudget.js                      |    0
 src/libs/{routes => routes_v1}/employees.js   |    0
 src/libs/{routes => routes_v1}/enrollment.js  |    0
 .../enrollmentProjection.js                   |    0
 src/libs/{routes => routes_v1}/financial.js   |    0
 .../glossEnrollmentRatio.js                   |    0
 src/libs/{routes => routes_v1}/idhm.js        |    0
 src/libs/{routes => routes_v1}/idhme.js       |    0
 src/libs/{routes => routes_v1}/idhml.js       |    0
 src/libs/{routes => routes_v1}/idhmr.js       |    0
 .../{routes => routes_v1}/infrastructure.js   |    0
 .../liquidEnrollmentRatio.js                  |    0
 src/libs/{routes => routes_v1}/location.js    |    0
 src/libs/{routes => routes_v1}/mesoregion.js  |    0
 src/libs/{routes => routes_v1}/message.js     |    0
 src/libs/{routes => routes_v1}/microregion.js |    0
 src/libs/{routes => routes_v1}/outOfSchool.js |    0
 .../{routes => routes_v1}/pibpercapita.js     |    0
 src/libs/{routes => routes_v1}/population.js  |    0
 src/libs/{routes => routes_v1}/portalMec.js   |    0
 .../{routes => routes_v1}/portalMecInep.js    |    0
 src/libs/{routes => routes_v1}/rateSchool.js  |    0
 src/libs/{routes => routes_v1}/region.js      |    0
 src/libs/{routes => routes_v1}/resetToken.js  |    0
 src/libs/{routes => routes_v1}/school.js      |    0
 .../schoolInfrastructure.js                   |    0
 .../{routes => routes_v1}/schoolLocation.js   |    0
 src/libs/{routes => routes_v1}/simulation.js  |    0
 src/libs/{routes => routes_v1}/siope.js       |    0
 src/libs/{routes => routes_v1}/spatial.js     |    0
 src/libs/{routes => routes_v1}/state.js       |    0
 src/libs/{routes => routes_v1}/studentsAee.js |    0
 src/libs/{routes => routes_v1}/teacher.js     |    0
 src/libs/{routes => routes_v1}/transport.js   |    0
 src/libs/{routes => routes_v1}/university.js  |    0
 .../universityEnrollment.js                   |    0
 .../universityLocalOffer.js                   |    0
 .../universityTeacher.js                      |    0
 src/libs/{routes => routes_v1}/user.js        |    0
 src/libs/{routes => routes_v1}/verifyToken.js |    0
 src/libs/routes_v2/api.js                     |  196 +++
 src/libs/routes_v2/auxiliar.js                |  397 ++++++
 src/libs/routes_v2/city.js                    |  151 +++
 src/libs/routes_v2/class.js                   |  402 ++++++
 src/libs/routes_v2/classCount.js              |  455 +++++++
 src/libs/routes_v2/classroom.js               |  277 ++++
 src/libs/routes_v2/classroomCount.js          | 1172 ++++++++++++++++
 src/libs/routes_v2/courseCount.js             |  814 +++++++++++
 src/libs/routes_v2/courseStudents.js          |  201 +++
 src/libs/routes_v2/cub.js                     |  227 ++++
 src/libs/routes_v2/dailyChargeAmount.js       |  467 +++++++
 src/libs/routes_v2/disciplines.js             |  682 ++++++++++
 src/libs/routes_v2/distributionFactor.js      |  210 +++
 src/libs/routes_v2/downloads.js               |   54 +
 src/libs/routes_v2/educationYears.js          |   46 +
 src/libs/routes_v2/educationalBudget.js       |  325 +++++
 src/libs/routes_v2/employees.js               |  656 +++++++++
 src/libs/routes_v2/enrollment.js              | 1204 +++++++++++++++++
 src/libs/routes_v2/enrollmentProjection.js    |  239 ++++
 src/libs/routes_v2/financial.js               |  186 +++
 src/libs/routes_v2/glossEnrollmentRatio.js    |  390 ++++++
 src/libs/routes_v2/idhm.js                    |  212 +++
 src/libs/routes_v2/idhme.js                   |  185 +++
 src/libs/routes_v2/idhml.js                   |  185 +++
 src/libs/routes_v2/idhmr.js                   |  188 +++
 src/libs/routes_v2/infrastructure.js          |  585 ++++++++
 src/libs/routes_v2/liquidEnrollmentRatio.js   |  434 ++++++
 src/libs/routes_v2/location.js                |  135 ++
 src/libs/routes_v2/mesoregion.js              |   86 ++
 src/libs/routes_v2/message.js                 |   55 +
 src/libs/routes_v2/microregion.js             |   99 ++
 src/libs/routes_v2/outOfSchool.js             |  377 ++++++
 src/libs/routes_v2/pibpercapita.js            |  250 ++++
 src/libs/routes_v2/population.js              |  163 +++
 src/libs/routes_v2/portalMec.js               |  136 ++
 src/libs/routes_v2/portalMecInep.js           |   60 +
 src/libs/routes_v2/rateSchool.js              |  337 +++++
 src/libs/routes_v2/region.js                  |   72 +
 src/libs/routes_v2/resetToken.js              |   81 ++
 src/libs/routes_v2/school.js                  |  663 +++++++++
 src/libs/routes_v2/schoolInfrastructure.js    |  741 ++++++++++
 src/libs/routes_v2/schoolLocation.js          |  114 ++
 src/libs/routes_v2/simulation.js              |  167 +++
 src/libs/routes_v2/siope.js                   |  186 +++
 src/libs/routes_v2/spatial.js                 |  373 +++++
 src/libs/routes_v2/state.js                   |  108 ++
 src/libs/routes_v2/studentsAee.js             |  219 +++
 src/libs/routes_v2/teacher.js                 |  576 ++++++++
 src/libs/routes_v2/transport.js               |  400 ++++++
 src/libs/routes_v2/university.js              |  347 +++++
 src/libs/routes_v2/universityEnrollment.js    |  848 ++++++++++++
 src/libs/routes_v2/universityLocalOffer.js    |  170 +++
 src/libs/routes_v2/universityTeacher.js       |  517 +++++++
 src/libs/routes_v2/user.js                    |  305 +++++
 src/libs/routes_v2/verifyToken.js             |   52 +
 111 files changed, 18218 insertions(+), 37 deletions(-)
 rename src/libs/{routes => routes_v1}/api.js (63%)
 rename src/libs/{routes => routes_v1}/auxiliar.js (100%)
 rename src/libs/{routes => routes_v1}/city.js (100%)
 rename src/libs/{routes => routes_v1}/class.js (100%)
 rename src/libs/{routes => routes_v1}/classCount.js (100%)
 rename src/libs/{routes => routes_v1}/classroom.js (100%)
 rename src/libs/{routes => routes_v1}/classroomCount.js (100%)
 rename src/libs/{routes => routes_v1}/courseCount.js (100%)
 rename src/libs/{routes => routes_v1}/courseStudents.js (100%)
 rename src/libs/{routes => routes_v1}/cub.js (100%)
 rename src/libs/{routes => routes_v1}/dailyChargeAmount.js (100%)
 rename src/libs/{routes => routes_v1}/disciplines.js (100%)
 rename src/libs/{routes => routes_v1}/distributionFactor.js (100%)
 rename src/libs/{routes => routes_v1}/downloads.js (100%)
 rename src/libs/{routes => routes_v1}/educationYears.js (100%)
 rename src/libs/{routes => routes_v1}/educationalBudget.js (100%)
 rename src/libs/{routes => routes_v1}/employees.js (100%)
 rename src/libs/{routes => routes_v1}/enrollment.js (100%)
 rename src/libs/{routes => routes_v1}/enrollmentProjection.js (100%)
 rename src/libs/{routes => routes_v1}/financial.js (100%)
 rename src/libs/{routes => routes_v1}/glossEnrollmentRatio.js (100%)
 rename src/libs/{routes => routes_v1}/idhm.js (100%)
 rename src/libs/{routes => routes_v1}/idhme.js (100%)
 rename src/libs/{routes => routes_v1}/idhml.js (100%)
 rename src/libs/{routes => routes_v1}/idhmr.js (100%)
 rename src/libs/{routes => routes_v1}/infrastructure.js (100%)
 rename src/libs/{routes => routes_v1}/liquidEnrollmentRatio.js (100%)
 rename src/libs/{routes => routes_v1}/location.js (100%)
 rename src/libs/{routes => routes_v1}/mesoregion.js (100%)
 rename src/libs/{routes => routes_v1}/message.js (100%)
 rename src/libs/{routes => routes_v1}/microregion.js (100%)
 rename src/libs/{routes => routes_v1}/outOfSchool.js (100%)
 rename src/libs/{routes => routes_v1}/pibpercapita.js (100%)
 rename src/libs/{routes => routes_v1}/population.js (100%)
 rename src/libs/{routes => routes_v1}/portalMec.js (100%)
 rename src/libs/{routes => routes_v1}/portalMecInep.js (100%)
 rename src/libs/{routes => routes_v1}/rateSchool.js (100%)
 rename src/libs/{routes => routes_v1}/region.js (100%)
 rename src/libs/{routes => routes_v1}/resetToken.js (100%)
 rename src/libs/{routes => routes_v1}/school.js (100%)
 rename src/libs/{routes => routes_v1}/schoolInfrastructure.js (100%)
 rename src/libs/{routes => routes_v1}/schoolLocation.js (100%)
 rename src/libs/{routes => routes_v1}/simulation.js (100%)
 rename src/libs/{routes => routes_v1}/siope.js (100%)
 rename src/libs/{routes => routes_v1}/spatial.js (100%)
 rename src/libs/{routes => routes_v1}/state.js (100%)
 rename src/libs/{routes => routes_v1}/studentsAee.js (100%)
 rename src/libs/{routes => routes_v1}/teacher.js (100%)
 rename src/libs/{routes => routes_v1}/transport.js (100%)
 rename src/libs/{routes => routes_v1}/university.js (100%)
 rename src/libs/{routes => routes_v1}/universityEnrollment.js (100%)
 rename src/libs/{routes => routes_v1}/universityLocalOffer.js (100%)
 rename src/libs/{routes => routes_v1}/universityTeacher.js (100%)
 rename src/libs/{routes => routes_v1}/user.js (100%)
 rename src/libs/{routes => routes_v1}/verifyToken.js (100%)
 create mode 100644 src/libs/routes_v2/api.js
 create mode 100644 src/libs/routes_v2/auxiliar.js
 create mode 100644 src/libs/routes_v2/city.js
 create mode 100644 src/libs/routes_v2/class.js
 create mode 100644 src/libs/routes_v2/classCount.js
 create mode 100644 src/libs/routes_v2/classroom.js
 create mode 100644 src/libs/routes_v2/classroomCount.js
 create mode 100644 src/libs/routes_v2/courseCount.js
 create mode 100644 src/libs/routes_v2/courseStudents.js
 create mode 100644 src/libs/routes_v2/cub.js
 create mode 100644 src/libs/routes_v2/dailyChargeAmount.js
 create mode 100644 src/libs/routes_v2/disciplines.js
 create mode 100644 src/libs/routes_v2/distributionFactor.js
 create mode 100644 src/libs/routes_v2/downloads.js
 create mode 100644 src/libs/routes_v2/educationYears.js
 create mode 100644 src/libs/routes_v2/educationalBudget.js
 create mode 100644 src/libs/routes_v2/employees.js
 create mode 100644 src/libs/routes_v2/enrollment.js
 create mode 100644 src/libs/routes_v2/enrollmentProjection.js
 create mode 100644 src/libs/routes_v2/financial.js
 create mode 100644 src/libs/routes_v2/glossEnrollmentRatio.js
 create mode 100644 src/libs/routes_v2/idhm.js
 create mode 100644 src/libs/routes_v2/idhme.js
 create mode 100644 src/libs/routes_v2/idhml.js
 create mode 100644 src/libs/routes_v2/idhmr.js
 create mode 100644 src/libs/routes_v2/infrastructure.js
 create mode 100644 src/libs/routes_v2/liquidEnrollmentRatio.js
 create mode 100644 src/libs/routes_v2/location.js
 create mode 100644 src/libs/routes_v2/mesoregion.js
 create mode 100644 src/libs/routes_v2/message.js
 create mode 100644 src/libs/routes_v2/microregion.js
 create mode 100644 src/libs/routes_v2/outOfSchool.js
 create mode 100644 src/libs/routes_v2/pibpercapita.js
 create mode 100644 src/libs/routes_v2/population.js
 create mode 100644 src/libs/routes_v2/portalMec.js
 create mode 100644 src/libs/routes_v2/portalMecInep.js
 create mode 100644 src/libs/routes_v2/rateSchool.js
 create mode 100644 src/libs/routes_v2/region.js
 create mode 100644 src/libs/routes_v2/resetToken.js
 create mode 100644 src/libs/routes_v2/school.js
 create mode 100644 src/libs/routes_v2/schoolInfrastructure.js
 create mode 100644 src/libs/routes_v2/schoolLocation.js
 create mode 100644 src/libs/routes_v2/simulation.js
 create mode 100644 src/libs/routes_v2/siope.js
 create mode 100644 src/libs/routes_v2/spatial.js
 create mode 100644 src/libs/routes_v2/state.js
 create mode 100644 src/libs/routes_v2/studentsAee.js
 create mode 100644 src/libs/routes_v2/teacher.js
 create mode 100644 src/libs/routes_v2/transport.js
 create mode 100644 src/libs/routes_v2/university.js
 create mode 100644 src/libs/routes_v2/universityEnrollment.js
 create mode 100644 src/libs/routes_v2/universityLocalOffer.js
 create mode 100644 src/libs/routes_v2/universityTeacher.js
 create mode 100644 src/libs/routes_v2/user.js
 create mode 100644 src/libs/routes_v2/verifyToken.js

diff --git a/src/libs/app.js b/src/libs/app.js
index b8b953c1..9f185105 100644
--- a/src/libs/app.js
+++ b/src/libs/app.js
@@ -14,7 +14,8 @@ const config = require(`${libs}/config`);
 
 const app = express();
 
-const api = require('./routes/api');
+const api_v1 = require('./routes_v1/api');
+const api_v2 = require('./routes_v2/api');
 
 const passport = require('passport');
 
@@ -65,7 +66,10 @@ app.use((req, res, next) => {
     next();
 });
 // Mounts all API routes under /api/v1
-app.use('/api/v1', api);
+app.use('/api/v1', api_v1);
+
+// Mounts all API routes under /api/v2
+app.use('/api/v2', api_v2);
 
 // Catch 404 and forward to error handler
 app.use((req, res, next) => {
diff --git a/src/libs/routes/api.js b/src/libs/routes_v1/api.js
similarity index 63%
rename from src/libs/routes/api.js
rename to src/libs/routes_v1/api.js
index f6715eaa..4b6a5630 100644
--- a/src/libs/routes/api.js
+++ b/src/libs/routes_v1/api.js
@@ -66,79 +66,79 @@ const idhml = require('./idhml');
 
 const oauth2 = require(`${libs}/middlewares/oauth2`);
 
-const verifyToken = require(`${libs}/routes/verifyToken`);
+const verifyToken = require(`${libs}/routes_v1/verifyToken`);
 
-const resetToken = require(`${libs}/routes/resetToken`);
+const resetToken = require(`${libs}/routes_v1/resetToken`);
 
-const educationYears = require(`${libs}/routes/educationYears`);
+const educationYears = require(`${libs}/routes_v1/educationYears`);
 
-const downloads = require(`${libs}/routes/downloads`);
+const downloads = require(`${libs}/routes_v1/downloads`);
 
-const infrastructure = require(`${libs}/routes/infrastructure`);
+const infrastructure = require(`${libs}/routes_v1/infrastructure`);
 
-const schoolInfrastructure = require(`${libs}/routes/schoolInfrastructure`);
+const schoolInfrastructure = require(`${libs}/routes_v1/schoolInfrastructure`);
 
-const distributionFactor = require(`${libs}/routes/distributionFactor`);
+const distributionFactor = require(`${libs}/routes_v1/distributionFactor`);
 
-const siope = require(`${libs}/routes/siope`);
+const siope = require(`${libs}/routes_v1/siope`);
 
-const verifyTeacher = require(`${libs}/routes/portalMec`);
+const verifyTeacher = require(`${libs}/routes_v1/portalMec`);
 
-const outOfSchool = require(`${libs}/routes/outOfSchool`);
+const outOfSchool = require(`${libs}/routes_v1/outOfSchool`);
 
-const classroomCount = require(`${libs}/routes/classroomCount`);
+const classroomCount = require(`${libs}/routes_v1/classroomCount`);
 
 const transport = require(`./transport`);
 
-const auxiliar = require(`${libs}/routes/auxiliar`);
+const auxiliar = require(`${libs}/routes_v1/auxiliar`);
 
-const dailyChargeAmount = require(`${libs}/routes/dailyChargeAmount`);
+const dailyChargeAmount = require(`${libs}/routes_v1/dailyChargeAmount`);
 
-const cub = require(`${libs}/routes/cub`);
+const cub = require(`${libs}/routes_v1/cub`);
 
-const classCount = require(`${libs}/routes/classCount`);
+const classCount = require(`${libs}/routes_v1/classCount`);
 
-const portalMecInep = require(`${libs}/routes/portalMecInep`);
+const portalMecInep = require(`${libs}/routes_v1/portalMecInep`);
 
-const enrollmentProjection = require(`${libs}/routes/enrollmentProjection`);
+const enrollmentProjection = require(`${libs}/routes_v1/enrollmentProjection`);
 
-const employees = require(`${libs}/routes/employees`);
+const employees = require(`${libs}/routes_v1/employees`);
 
-const financial = require(`${libs}/routes/financial`);
+const financial = require(`${libs}/routes_v1/financial`);
 
-const universityEnrollment = require(`${libs}/routes/universityEnrollment`);
+const universityEnrollment = require(`${libs}/routes_v1/universityEnrollment`);
 
-const courseCount = require(`${libs}/routes/courseCount`);
+const courseCount = require(`${libs}/routes_v1/courseCount`);
 
-const university = require(`${libs}/routes/university`);
+const university = require(`${libs}/routes_v1/university`);
 
-const universityTeacher = require(`${libs}/routes/universityTeacher`);
+const universityTeacher = require(`${libs}/routes_v1/universityTeacher`);
 
-const educationalBudget = require(`${libs}/routes/educationalBudget`);
+const educationalBudget = require(`${libs}/routes_v1/educationalBudget`);
 
-const schoolLocation = require(`${libs}/routes/schoolLocation`);
+const schoolLocation = require(`${libs}/routes_v1/schoolLocation`);
 
-const studentsAee = require(`${libs}/routes/studentsAee`);
+const studentsAee = require(`${libs}/routes_v1/studentsAee`);
 
-const mesoregion = require(`${libs}/routes/mesoregion`);
+const mesoregion = require(`${libs}/routes_v1/mesoregion`);
 
-const microregion = require(`${libs}/routes/microregion`);
+const microregion = require(`${libs}/routes_v1/microregion`);
 
-const location = require(`${libs}/routes/location`);
+const location = require(`${libs}/routes_v1/location`);
 
-const disciplines = require(`${libs}/routes/disciplines`);
+const disciplines = require(`${libs}/routes_v1/disciplines`);
 
-const universityLocalOffer = require(`${libs}/routes/universityLocalOffer`);
+const universityLocalOffer = require(`${libs}/routes_v1/universityLocalOffer`);
 
-const message = require(`${libs}/routes/message`);
+const message = require(`${libs}/routes_v1/message`);
 
-const courseStudents = require(`${libs}/routes/courseStudents`);
+const courseStudents = require(`${libs}/routes_v1/courseStudents`);
 
 api.get('/', (req, res) => {
-    res.json({ msg: 'SimCAQ API is running' });
+    res.json({ msg: 'SimCAQ API v1 is running' });
 });
 
-// mount API routes
+// mount API routes_v1
 api.use('/user', user);
 api.use('/simulation', simulation);
 api.use('/class', classes);
diff --git a/src/libs/routes/auxiliar.js b/src/libs/routes_v1/auxiliar.js
similarity index 100%
rename from src/libs/routes/auxiliar.js
rename to src/libs/routes_v1/auxiliar.js
diff --git a/src/libs/routes/city.js b/src/libs/routes_v1/city.js
similarity index 100%
rename from src/libs/routes/city.js
rename to src/libs/routes_v1/city.js
diff --git a/src/libs/routes/class.js b/src/libs/routes_v1/class.js
similarity index 100%
rename from src/libs/routes/class.js
rename to src/libs/routes_v1/class.js
diff --git a/src/libs/routes/classCount.js b/src/libs/routes_v1/classCount.js
similarity index 100%
rename from src/libs/routes/classCount.js
rename to src/libs/routes_v1/classCount.js
diff --git a/src/libs/routes/classroom.js b/src/libs/routes_v1/classroom.js
similarity index 100%
rename from src/libs/routes/classroom.js
rename to src/libs/routes_v1/classroom.js
diff --git a/src/libs/routes/classroomCount.js b/src/libs/routes_v1/classroomCount.js
similarity index 100%
rename from src/libs/routes/classroomCount.js
rename to src/libs/routes_v1/classroomCount.js
diff --git a/src/libs/routes/courseCount.js b/src/libs/routes_v1/courseCount.js
similarity index 100%
rename from src/libs/routes/courseCount.js
rename to src/libs/routes_v1/courseCount.js
diff --git a/src/libs/routes/courseStudents.js b/src/libs/routes_v1/courseStudents.js
similarity index 100%
rename from src/libs/routes/courseStudents.js
rename to src/libs/routes_v1/courseStudents.js
diff --git a/src/libs/routes/cub.js b/src/libs/routes_v1/cub.js
similarity index 100%
rename from src/libs/routes/cub.js
rename to src/libs/routes_v1/cub.js
diff --git a/src/libs/routes/dailyChargeAmount.js b/src/libs/routes_v1/dailyChargeAmount.js
similarity index 100%
rename from src/libs/routes/dailyChargeAmount.js
rename to src/libs/routes_v1/dailyChargeAmount.js
diff --git a/src/libs/routes/disciplines.js b/src/libs/routes_v1/disciplines.js
similarity index 100%
rename from src/libs/routes/disciplines.js
rename to src/libs/routes_v1/disciplines.js
diff --git a/src/libs/routes/distributionFactor.js b/src/libs/routes_v1/distributionFactor.js
similarity index 100%
rename from src/libs/routes/distributionFactor.js
rename to src/libs/routes_v1/distributionFactor.js
diff --git a/src/libs/routes/downloads.js b/src/libs/routes_v1/downloads.js
similarity index 100%
rename from src/libs/routes/downloads.js
rename to src/libs/routes_v1/downloads.js
diff --git a/src/libs/routes/educationYears.js b/src/libs/routes_v1/educationYears.js
similarity index 100%
rename from src/libs/routes/educationYears.js
rename to src/libs/routes_v1/educationYears.js
diff --git a/src/libs/routes/educationalBudget.js b/src/libs/routes_v1/educationalBudget.js
similarity index 100%
rename from src/libs/routes/educationalBudget.js
rename to src/libs/routes_v1/educationalBudget.js
diff --git a/src/libs/routes/employees.js b/src/libs/routes_v1/employees.js
similarity index 100%
rename from src/libs/routes/employees.js
rename to src/libs/routes_v1/employees.js
diff --git a/src/libs/routes/enrollment.js b/src/libs/routes_v1/enrollment.js
similarity index 100%
rename from src/libs/routes/enrollment.js
rename to src/libs/routes_v1/enrollment.js
diff --git a/src/libs/routes/enrollmentProjection.js b/src/libs/routes_v1/enrollmentProjection.js
similarity index 100%
rename from src/libs/routes/enrollmentProjection.js
rename to src/libs/routes_v1/enrollmentProjection.js
diff --git a/src/libs/routes/financial.js b/src/libs/routes_v1/financial.js
similarity index 100%
rename from src/libs/routes/financial.js
rename to src/libs/routes_v1/financial.js
diff --git a/src/libs/routes/glossEnrollmentRatio.js b/src/libs/routes_v1/glossEnrollmentRatio.js
similarity index 100%
rename from src/libs/routes/glossEnrollmentRatio.js
rename to src/libs/routes_v1/glossEnrollmentRatio.js
diff --git a/src/libs/routes/idhm.js b/src/libs/routes_v1/idhm.js
similarity index 100%
rename from src/libs/routes/idhm.js
rename to src/libs/routes_v1/idhm.js
diff --git a/src/libs/routes/idhme.js b/src/libs/routes_v1/idhme.js
similarity index 100%
rename from src/libs/routes/idhme.js
rename to src/libs/routes_v1/idhme.js
diff --git a/src/libs/routes/idhml.js b/src/libs/routes_v1/idhml.js
similarity index 100%
rename from src/libs/routes/idhml.js
rename to src/libs/routes_v1/idhml.js
diff --git a/src/libs/routes/idhmr.js b/src/libs/routes_v1/idhmr.js
similarity index 100%
rename from src/libs/routes/idhmr.js
rename to src/libs/routes_v1/idhmr.js
diff --git a/src/libs/routes/infrastructure.js b/src/libs/routes_v1/infrastructure.js
similarity index 100%
rename from src/libs/routes/infrastructure.js
rename to src/libs/routes_v1/infrastructure.js
diff --git a/src/libs/routes/liquidEnrollmentRatio.js b/src/libs/routes_v1/liquidEnrollmentRatio.js
similarity index 100%
rename from src/libs/routes/liquidEnrollmentRatio.js
rename to src/libs/routes_v1/liquidEnrollmentRatio.js
diff --git a/src/libs/routes/location.js b/src/libs/routes_v1/location.js
similarity index 100%
rename from src/libs/routes/location.js
rename to src/libs/routes_v1/location.js
diff --git a/src/libs/routes/mesoregion.js b/src/libs/routes_v1/mesoregion.js
similarity index 100%
rename from src/libs/routes/mesoregion.js
rename to src/libs/routes_v1/mesoregion.js
diff --git a/src/libs/routes/message.js b/src/libs/routes_v1/message.js
similarity index 100%
rename from src/libs/routes/message.js
rename to src/libs/routes_v1/message.js
diff --git a/src/libs/routes/microregion.js b/src/libs/routes_v1/microregion.js
similarity index 100%
rename from src/libs/routes/microregion.js
rename to src/libs/routes_v1/microregion.js
diff --git a/src/libs/routes/outOfSchool.js b/src/libs/routes_v1/outOfSchool.js
similarity index 100%
rename from src/libs/routes/outOfSchool.js
rename to src/libs/routes_v1/outOfSchool.js
diff --git a/src/libs/routes/pibpercapita.js b/src/libs/routes_v1/pibpercapita.js
similarity index 100%
rename from src/libs/routes/pibpercapita.js
rename to src/libs/routes_v1/pibpercapita.js
diff --git a/src/libs/routes/population.js b/src/libs/routes_v1/population.js
similarity index 100%
rename from src/libs/routes/population.js
rename to src/libs/routes_v1/population.js
diff --git a/src/libs/routes/portalMec.js b/src/libs/routes_v1/portalMec.js
similarity index 100%
rename from src/libs/routes/portalMec.js
rename to src/libs/routes_v1/portalMec.js
diff --git a/src/libs/routes/portalMecInep.js b/src/libs/routes_v1/portalMecInep.js
similarity index 100%
rename from src/libs/routes/portalMecInep.js
rename to src/libs/routes_v1/portalMecInep.js
diff --git a/src/libs/routes/rateSchool.js b/src/libs/routes_v1/rateSchool.js
similarity index 100%
rename from src/libs/routes/rateSchool.js
rename to src/libs/routes_v1/rateSchool.js
diff --git a/src/libs/routes/region.js b/src/libs/routes_v1/region.js
similarity index 100%
rename from src/libs/routes/region.js
rename to src/libs/routes_v1/region.js
diff --git a/src/libs/routes/resetToken.js b/src/libs/routes_v1/resetToken.js
similarity index 100%
rename from src/libs/routes/resetToken.js
rename to src/libs/routes_v1/resetToken.js
diff --git a/src/libs/routes/school.js b/src/libs/routes_v1/school.js
similarity index 100%
rename from src/libs/routes/school.js
rename to src/libs/routes_v1/school.js
diff --git a/src/libs/routes/schoolInfrastructure.js b/src/libs/routes_v1/schoolInfrastructure.js
similarity index 100%
rename from src/libs/routes/schoolInfrastructure.js
rename to src/libs/routes_v1/schoolInfrastructure.js
diff --git a/src/libs/routes/schoolLocation.js b/src/libs/routes_v1/schoolLocation.js
similarity index 100%
rename from src/libs/routes/schoolLocation.js
rename to src/libs/routes_v1/schoolLocation.js
diff --git a/src/libs/routes/simulation.js b/src/libs/routes_v1/simulation.js
similarity index 100%
rename from src/libs/routes/simulation.js
rename to src/libs/routes_v1/simulation.js
diff --git a/src/libs/routes/siope.js b/src/libs/routes_v1/siope.js
similarity index 100%
rename from src/libs/routes/siope.js
rename to src/libs/routes_v1/siope.js
diff --git a/src/libs/routes/spatial.js b/src/libs/routes_v1/spatial.js
similarity index 100%
rename from src/libs/routes/spatial.js
rename to src/libs/routes_v1/spatial.js
diff --git a/src/libs/routes/state.js b/src/libs/routes_v1/state.js
similarity index 100%
rename from src/libs/routes/state.js
rename to src/libs/routes_v1/state.js
diff --git a/src/libs/routes/studentsAee.js b/src/libs/routes_v1/studentsAee.js
similarity index 100%
rename from src/libs/routes/studentsAee.js
rename to src/libs/routes_v1/studentsAee.js
diff --git a/src/libs/routes/teacher.js b/src/libs/routes_v1/teacher.js
similarity index 100%
rename from src/libs/routes/teacher.js
rename to src/libs/routes_v1/teacher.js
diff --git a/src/libs/routes/transport.js b/src/libs/routes_v1/transport.js
similarity index 100%
rename from src/libs/routes/transport.js
rename to src/libs/routes_v1/transport.js
diff --git a/src/libs/routes/university.js b/src/libs/routes_v1/university.js
similarity index 100%
rename from src/libs/routes/university.js
rename to src/libs/routes_v1/university.js
diff --git a/src/libs/routes/universityEnrollment.js b/src/libs/routes_v1/universityEnrollment.js
similarity index 100%
rename from src/libs/routes/universityEnrollment.js
rename to src/libs/routes_v1/universityEnrollment.js
diff --git a/src/libs/routes/universityLocalOffer.js b/src/libs/routes_v1/universityLocalOffer.js
similarity index 100%
rename from src/libs/routes/universityLocalOffer.js
rename to src/libs/routes_v1/universityLocalOffer.js
diff --git a/src/libs/routes/universityTeacher.js b/src/libs/routes_v1/universityTeacher.js
similarity index 100%
rename from src/libs/routes/universityTeacher.js
rename to src/libs/routes_v1/universityTeacher.js
diff --git a/src/libs/routes/user.js b/src/libs/routes_v1/user.js
similarity index 100%
rename from src/libs/routes/user.js
rename to src/libs/routes_v1/user.js
diff --git a/src/libs/routes/verifyToken.js b/src/libs/routes_v1/verifyToken.js
similarity index 100%
rename from src/libs/routes/verifyToken.js
rename to src/libs/routes_v1/verifyToken.js
diff --git a/src/libs/routes_v2/api.js b/src/libs/routes_v2/api.js
new file mode 100644
index 00000000..906af9cf
--- /dev/null
+++ b/src/libs/routes_v2/api.js
@@ -0,0 +1,196 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const api = express();
+
+const libs = `${process.cwd()}/libs`;
+
+const config = require(`${libs}/config`);
+
+const classes = require('./class');
+
+const enrollment = require('./enrollment');
+
+const state = require('./state');
+
+const region = require('./region');
+
+const city = require('./city');
+
+const school = require('./school');
+
+const simulation = require('./simulation');
+
+const user = require('./user');
+
+const classroom = require('./classroom');
+
+const teacher = require('./teacher');
+
+const idhme = require('./idhme');
+
+const pibpercapita = require('./pibpercapita')
+
+const population = require('./population')
+
+const rateSchool = require('./rateSchool')
+
+const glossEnrollmentRatio = require('./glossEnrollmentRatio')
+
+const liquidEnrollmentRatio = require('./liquidEnrollmentRatio')
+
+const idhm = require('./idhm');
+
+const idhmr = require('./idhmr');
+
+const idhml = require('./idhml');
+
+const oauth2 = require(`${libs}/middlewares/oauth2`);
+
+const verifyToken = require(`${libs}/routes_v2/verifyToken`);
+
+const resetToken = require(`${libs}/routes_v2/resetToken`);
+
+const educationYears = require(`${libs}/routes_v2/educationYears`);
+
+const downloads = require(`${libs}/routes_v2/downloads`);
+
+const infrastructure = require(`${libs}/routes_v2/infrastructure`);
+
+const schoolInfrastructure = require(`${libs}/routes_v2/schoolInfrastructure`);
+
+const distributionFactor = require(`${libs}/routes_v2/distributionFactor`);
+
+const siope = require(`${libs}/routes_v2/siope`);
+
+const verifyTeacher = require(`${libs}/routes_v2/portalMec`);
+
+const outOfSchool = require(`${libs}/routes_v2/outOfSchool`);
+
+const classroomCount = require(`${libs}/routes_v2/classroomCount`);
+
+const transport = require(`./transport`);
+
+const auxiliar = require(`${libs}/routes_v2/auxiliar`);
+
+const dailyChargeAmount = require(`${libs}/routes_v2/dailyChargeAmount`);
+
+const cub = require(`${libs}/routes_v2/cub`);
+
+const classCount = require(`${libs}/routes_v2/classCount`);
+
+const portalMecInep = require(`${libs}/routes_v2/portalMecInep`);
+
+const enrollmentProjection = require(`${libs}/routes_v2/enrollmentProjection`);
+
+const employees = require(`${libs}/routes_v2/employees`);
+
+const financial = require(`${libs}/routes_v2/financial`);
+
+const universityEnrollment = require(`${libs}/routes_v2/universityEnrollment`);
+
+const courseCount = require(`${libs}/routes_v2/courseCount`);
+
+const university = require(`${libs}/routes_v2/university`);
+
+const universityTeacher = require(`${libs}/routes_v2/universityTeacher`);
+
+const educationalBudget = require(`${libs}/routes_v2/educationalBudget`);
+
+const schoolLocation = require(`${libs}/routes_v2/schoolLocation`);
+
+const studentsAee = require(`${libs}/routes_v2/studentsAee`);
+
+const mesoregion = require(`${libs}/routes_v2/mesoregion`);
+
+const microregion = require(`${libs}/routes_v2/microregion`);
+
+const location = require(`${libs}/routes_v2/location`);
+
+const disciplines = require(`${libs}/routes_v2/disciplines`);
+
+const universityLocalOffer = require(`${libs}/routes_v2/universityLocalOffer`);
+
+const message = require(`${libs}/routes_v2/message`);
+
+const courseStudents = require(`${libs}/routes_v2/courseStudents`);
+
+api.get('/', (req, res) => {
+    res.json({ msg: 'SimCAQ API v2 is running' });
+});
+
+// mount API routes_v2
+api.use('/user', user);
+api.use('/simulation', simulation);
+api.use('/class', classes);
+api.use('/enrollment', enrollment);
+api.use('/state', state);
+api.use('/region', region);
+api.use('/city', city);
+api.use('/school', school);
+api.use('/classroom', classroom);
+api.use('/teacher', teacher);
+api.use('/idhmr', idhmr);
+api.use('/idhm', idhm);
+api.use('/idhme', idhme);
+api.use('/pibpercapita', pibpercapita);
+api.use('/population', population);
+api.use('/rate_school', rateSchool);
+api.use('/gloss_enrollment_ratio', glossEnrollmentRatio);
+api.use('/liquid_enrollment_ratio', liquidEnrollmentRatio);
+api.use('/idhml', idhml);
+api.use('/auth/token', oauth2.token);
+api.use('/verify', verifyToken);
+api.use('/reset', resetToken);
+api.use('/education_years', educationYears);
+api.use('/downloads', downloads);
+api.use('/infrastructure', infrastructure);
+api.use('/school_infrastructure', schoolInfrastructure);
+api.use('/distribution_factor', distributionFactor);
+api.use('/siope', siope);
+api.use('/out_of_school', outOfSchool);
+api.use('/classroom_count', classroomCount);
+api.use('/daily_charge_amount', dailyChargeAmount);
+api.use('/transport', transport);
+api.use('/cub', cub);
+api.use('/auxiliar', auxiliar);
+api.use('/verify_teacher', verifyTeacher);
+api.use('/class_count', classCount);
+api.use('/portal_mec_inep', portalMecInep);
+api.use('/enrollment_projection', enrollmentProjection);
+api.use('/employees', employees);
+api.use('/financial', financial);
+api.use('/university_enrollment', universityEnrollment);
+api.use('/university', university);
+api.use('/university_teacher', universityTeacher);
+api.use('/course_count', courseCount);
+api.use('/school_location', schoolLocation);
+api.use('/students_aee', studentsAee);
+api.use('/mesoregion', mesoregion);
+api.use('/microregion', microregion);
+api.use('/location', location);
+api.use('/disciplines', disciplines);
+api.use('/universityLocalOffer', universityLocalOffer);
+api.use('/message', message);
+api.use('/course_students', courseStudents);
+
+module.exports = api;
diff --git a/src/libs/routes_v2/auxiliar.js b/src/libs/routes_v2/auxiliar.js
new file mode 100644
index 00000000..ef538961
--- /dev/null
+++ b/src/libs/routes_v2/auxiliar.js
@@ -0,0 +1,397 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const auxiliarApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]}  }).middleware;
+
+let rqf = new ReqQueryFields();
+
+auxiliarApp.use(cache('15 day'));
+
+auxiliarApp.get('/year_range', (req, res, next) => {
+    req.sql.from('docente')
+    .field('MIN(docente.ano_censo)', 'start_year')
+    .field('MAX(docente.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+auxiliarApp.get('/years', (req, res, next) => {
+    req.sql.from('docente').
+    field('DISTINCT docente.ano_censo', 'year')
+    next();
+}, query, response('years'));
+
+auxiliarApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'docente\'');
+    next();
+}, query, response('source'))
+
+auxiliarApp.get('/education_type', (req, res, next) => {
+    req.sql.from('docente')
+    .field('DISTINCT nivel_tipo_formacao', 'id')
+    .order('id');
+    next();
+}, query, (req, res, next) => {
+    req.result.forEach((result) => {
+        result.name = id2str.educationType(result.id);
+    });
+    next();
+}, response('education_type'));
+
+auxiliarApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 7; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+auxiliarApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+auxiliarApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+auxiliarApp.get('/diff_location', (req, res, next) => {
+    req.result = [
+        {id: 0, name: "A escola não está em localidade diferenciada"},
+        {id: 1, name: "Área de assentamento"},
+        {id: 2, name: "Terra indígena"},
+        {id: 3, name: "Terra remanescente de quilombos"},
+    ];
+    next();
+}, response('diff_location'));
+
+auxiliarApp.get('/education_level_mod', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 12; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.educationLevelMod(i)
+        });
+    }
+
+    req.result.push({
+        id: 99,
+        name: id2str.educationLevelMod(99)
+    });
+    next();
+}, response('education_level_mod'));
+
+auxiliarApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+auxiliarApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <=5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ethnicGroup(i)
+        });
+    }
+    next();
+}, response('ethnic_group'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'adm_dependency',
+    table: 'docente',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'docente',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'education_type',
+    table: 'docente',
+    tableField: 'nivel_tipo_formacao',
+    resultField: 'education_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'nivel_tipo_formacao'
+    }
+}).addValue({
+    name: 'education_level_mod',
+    table: 'docente',
+    tableField: 'etapas_mod_ensino_segmento_id',
+    resultField: 'education_level_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapas_mod_ensino_segmento_id'
+    }
+}).addValue({
+  name: 'region',
+  table: 'regiao',
+  tableField: ['nome', 'id'],
+  resultField: ['region_name', 'region_id'],
+  where: {
+      relation: '=',
+      type: 'integer',
+      field: 'id'
+  },
+  join: {
+      primary: 'id',
+      foreign: 'escola_regiao_id',
+      foreignTable: 'docente'
+  }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_estado_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'docente',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}, 'filter').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'docente'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'docente'
+    }
+}, 'filter').addValue({
+    name: 'location',
+    table: 'docente',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: 'docente',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'gender',
+    table: 'docente',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: 'docente',
+    tableField: 'cor_raca',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca'
+    }
+});
+
+// LDE
+auxiliarApp.get('/', rqf.parse(), (req, res, next) => {
+  req.sql.field('COUNT(DISTINCT docente.id_docente)', 'total')
+  .field("'Brasil'", 'name')
+  .field('docente.ano_censo', 'year')
+  .from('docente')
+  .join('turma', null, 'docente.turma_id=turma.id AND docente.ano_censo=turma.ano_censo')
+  .group('docente.ano_censo')
+  .order('docente.ano_censo')
+  .where('(docente.tipo_docente = 2) AND \
+    ((docente.tipo_turma_id >= 0 AND docente.tipo_turma_id <= 3 AND docente.tipo_turma_atendimento_id is NULL) \
+    OR ((docente.tipo_turma_atendimento_id = 1 OR docente.tipo_turma_atendimento_id = 2) AND docente.tipo_turma_id is NULL)) \
+    AND (docente.ano_censo <> 2009 or (docente.escola_estado_id <> 42 AND docente.escola_estado_id <> 43))');     // não devemos trazer SC em 2009.
+  next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('auxiliar'));
+
+// SimCAQ
+auxiliarApp.get('/count', rqf.parse(), (req, res, next) => {
+    req.sql.field('COUNT(DISTINCT docente.id_docente)', 'total')
+   .field("'Brasil'", 'name')
+   .field('docente.ano_censo', 'year')
+   .from('docente')
+   .group('docente.ano_censo')
+   .order('docente.ano_censo')
+   .where('(tipo_turma_id <= 3) AND (tipo_docente=2) AND (dependencia_adm_id = 1 OR dependencia_adm_id = 2 OR dependencia_adm_id = 3) AND  \
+         (etapas_mod_ensino_segmento_id=1 OR etapas_mod_ensino_segmento_id=2 OR etapas_mod_ensino_segmento_id=4 OR \
+         etapas_mod_ensino_segmento_id=5 OR etapas_mod_ensino_segmento_id=6 OR etapas_mod_ensino_segmento_id=8 OR etapas_mod_ensino_segmento_id=9)');
+
+  next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('auxiliar'));
+
+module.exports = auxiliarApp;
diff --git a/src/libs/routes_v2/city.js b/src/libs/routes_v2/city.js
new file mode 100644
index 00000000..fc697a9f
--- /dev/null
+++ b/src/libs/routes_v2/city.js
@@ -0,0 +1,151 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const cityApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+cityApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'id',
+    table: 'municipio',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'municipio'
+    }
+}).addValue({
+    name: 'state_not',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '<>',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'municipio'
+    }
+}).addField({
+    name: 'search',
+    field: false,
+    where: true
+}).addValueToField({
+    name: 'name',
+    table: 'municipio',
+    tableField: 'nome',
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome'
+    }
+}, 'search').addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: 'mesorregiao_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: 'microrregiao_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id'
+    }
+}).addValueToField({
+    name: 'region',
+    table: 'estado',
+    tableField: 'regiao_id',
+    resultField: 'region_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'regiao_id',
+        table: 'estado'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'municipio'
+    }
+}, 'filter');
+
+// Return all cities
+cityApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.from('municipio')
+    .field('municipio.nome', 'name')
+    .field('municipio.id')
+    .field('municipio.estado_id', 'state_id')
+    .field('municipio.longitude', 'longitude')
+    .field('municipio.latitude', 'latitude')
+    .field('municipio.mesorregiao_id', 'mesoregion_id')
+    .field('municipio.microrregiao_id', 'microregion_id');
+    next();
+}, query, response('city'));
+
+
+module.exports = cityApp;
diff --git a/src/libs/routes_v2/class.js b/src/libs/routes_v2/class.js
new file mode 100644
index 00000000..4eca8cc3
--- /dev/null
+++ b/src/libs/routes_v2/class.js
@@ -0,0 +1,402 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const classApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqfCount = new ReqQueryFields();
+
+classApp.use(cache('15 day'));
+
+// Complete range of the enrollments dataset.
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+classApp.get('/year_range', (req, res, next) => {
+    req.sql.from('turma')
+    .field('MIN(turma.ano_censo)', 'start_year')
+    .field('MAX(turma.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+classApp.get('/years', (req, res, next) => {
+    req.sql.from('turma')
+    .field('DISTINCT turma.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+classApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'turma\'');
+    next();
+}, query, response('source'));
+
+classApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+classApp.get('/diff_location', (req, res, next) => {
+    req.result = [
+        {id: 0, name: "A escola não está em localidade diferenciada"},
+        {id: 1, name: "Área de assentamento"},
+        {id: 2, name: "Terra indígena"},
+        {id: 3, name: "Terra remanescente de quilombos"},
+    ];
+    next();
+}, response('diff_location'));
+
+// Returns all adm dependencies
+classApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+classApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 8; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+// Returns all periods avaible
+classApp.get('/period', (req, res, next) => {
+    req.result = [];
+    for(let i=1; i <= 3; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.period(i)
+        });
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.period(99)
+    });
+    next();
+}, response('period'));
+
+// Returns integral-time avaible
+classApp.get('/integral_time', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.integralTime(i)
+        });
+    }
+    next();
+}, response('integral_time'));
+
+// Returns all educational levels avaible
+classApp.get('/education_level_mod', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <=12; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.educationLevelMod(i)
+        });
+    }
+    next();
+}, response('education_level_mod'));
+
+classApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Creche'},
+        {id: 2, name: 'Pré-Escola'},
+        {id: 3, name: 'Ensino Fundamental - anos iniciais'},
+        {id: 4, name: 'Ensino Fundamental - anos finais'},
+        {id: 5, name: 'Ensino Médio'},
+        {id: 6, name: 'EJA'},
+        {id: 7, name: 'EE exclusiva'}
+    ];
+    next();
+}, response('education_level_short'));
+
+rqfCount.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'turma'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'turma'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'turma'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'turma'
+    }
+}, 'filter').addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'turma'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'turma'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'turma'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'turma',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'turma',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name:'adm_dependency',
+    table: 'turma',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'location',
+    table: 'turma',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: 'turma',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'turma',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+  name:'education_level_mod',
+  table: 'turma',
+  tableField: 'etapas_mod_ensino_segmento_id',
+  resultField: 'education_level_mod_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'etapas_mod_ensino_segmento_id'
+  }
+}).addValue({
+    name: 'education_level_short',
+    table: 'turma',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'turma',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+  name:'period',
+  table: 'turma',
+  tableField: 'turma_turno_id',
+  resultField: 'period_id',
+  where: {
+      relation: '=',
+      type: 'integer',
+      field: 'turma_turno_id'
+  }
+}).addValue({
+  name:'integral_time',
+  table: 'turma',
+  tableField: 'tempo_integral',
+  resultField: 'integral_time_id',
+  where: {
+      relation: '=',
+      type: 'integer',
+      field: 'tempo_integral'
+  }
+}).addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'turma'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'turma'
+    }
+}, 'filter').addValueToField({
+    name: 'period_not',
+    table: 'turma',
+    tableField: 'turma_turno_id',
+    resultField: 'period_id',
+    where: {
+        relation: '<>',
+        type: 'integer',
+        field: 'turma_turno_id'
+    }
+}, 'filter');
+
+
+classApp.get('/', rqfCount.parse(), (req, res, next) => {
+    req.sql.field('COUNT(turma.id)', 'total')
+    .field("'Brasil'", 'name')
+    .field('turma.ano_censo', 'year')
+    .from('turma')
+    .group('turma.ano_censo')
+    .order('turma.ano_censo')
+    .where('(turma.tipo_turma_id >= 0 AND turma.tipo_turma_id <= 3 AND turma.tipo_atendimento_id is NULL) \
+        OR ((turma.tipo_atendimento_id = 1 OR turma.tipo_atendimento_id = 2) AND turma.tipo_turma_id is NULL) ');
+    next();
+}, rqfCount.build(), query, addMissing(rqfCount), id2str.transform(), response('class'));
+
+module.exports = classApp;
diff --git a/src/libs/routes_v2/classCount.js b/src/libs/routes_v2/classCount.js
new file mode 100644
index 00000000..ffdbe34f
--- /dev/null
+++ b/src/libs/routes_v2/classCount.js
@@ -0,0 +1,455 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const classCountApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+  name: 'filter',
+  field: false,
+  where: true
+}).addField({
+  name: 'dims',
+  field: true,
+  where: false
+}).addValueToField({
+  name: 'city',
+  table: 'municipio',
+  tableField: ['nome', 'id'],
+  resultField: ['city_name', 'city_id'],
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'municipio_id',
+    table: 'turma'
+  },
+  join: {
+    primary: 'id',
+    foreign: 'municipio_id',
+    foreignTable: 'turma'
+  }
+}, 'dims').addValueToField({
+  name: 'city',
+  table: 'municipio',
+  tableField: 'nome',
+  resultField: 'city_name',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'municipio_id',
+    table: 'turma'
+  },
+  join: {
+    primary: 'id',
+    foreign: 'municipio_id',
+    foreignTable: 'turma'
+  }
+}, 'filter')
+.addValue({
+  name: 'state',
+  table: 'estado',
+  tableField: ['nome', 'id'],
+  resultField: ['state_name', 'state_id'],
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'estado_id',
+    table: 'turma'
+  },
+  join: {
+    primary: 'id',
+    foreign: 'estado_id',
+    foreignTable: 'turma'
+  }
+}).addValue({
+  name: 'region',
+  table: 'regiao',
+  tableField: ['nome', 'id'],
+  resultField: ['region_name', 'region_id'],
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'id'
+  },
+  join: {
+    primary: 'id',
+    foreign: 'regiao_id',
+    foreignTable: 'turma'
+  }
+}).addValue({
+  name: 'min_year',
+  table: 'turma',
+  tableField: 'ano_censo',
+  resultField: 'year',
+  where: {
+    relation: '>=',
+    type: 'integer',
+    field: 'ano_censo'
+  }
+}).addValue({
+  name: 'max_year',
+  table: 'turma',
+  tableField: 'ano_censo',
+  resultField: 'year',
+  where: {
+    relation: '<=',
+    type: 'integer',
+    field: 'ano_censo'
+  }
+}).addValue({
+  name:'adm_dependency',
+  table: 'turma',
+  tableField: 'dependencia_adm_id',
+  resultField: 'adm_dependency_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'dependencia_adm_id'
+  }
+}).addValue({
+  name: 'location',
+  table: 'turma',
+  tableField: 'localizacao_id',
+  resultField: 'location_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'localizacao_id'
+  }
+}).addValue({
+  name: 'rural_location',
+  table: 'turma',
+  tableField: 'localidade_area_rural',
+  resultField: 'rural_location_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'localidade_area_rural'
+  }
+}).addValue({
+  name:'education_level_mod',
+  table: 'turma',
+  tableField: 'etapas_mod_ensino_segmento_id',
+  resultField: 'education_level_mod_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'etapas_mod_ensino_segmento_id'
+  }
+}).addValue({
+  name:'education_level_short',
+  table: 'turma',
+  tableField: 'etapa_resumida',
+  resultField: 'education_level_short_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'etapa_resumida'
+  }
+}).addValue({
+  name: 'adm_dependency_detailed',
+  table: 'turma',
+  tableField: 'dependencia_adm_priv',
+  resultField: 'adm_dependency_detailed_id',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'dependencia_adm_priv'
+  }
+}).addValueToField({
+  name: 'school',
+  table: 'escola',
+  tableField: ['nome_escola', 'id'],
+  resultField: ['school_name', 'school_id'],
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'id'
+  },
+  join: {
+    primary: ['id', 'ano_censo'],
+    foreign: ['escola_id', 'ano_censo'],
+    foreignTable: 'turma'
+  }
+}, 'dims').addValueToField({
+  name: 'school',
+  table: 'escola',
+  tableField: 'nome_escola',
+  resultField: 'school_name',
+  where: {
+    relation: '=',
+    type: 'integer',
+    field: 'id'
+  },
+  join: {
+    primary: ['id', 'ano_censo'],
+    foreign: ['escola_id', 'ano_censo'],
+    foreignTable: 'turma'
+  }
+}, 'filter');
+
+classCountApp.get('/year_range', (req, res, next) => {
+  req.sql.from('escola')
+  .field('MIN(escola.ano_censo)', 'start_year')
+  .field('MAX(escola.ano_censo)', 'end_year');
+  next();
+}, query, response('range'));
+
+classCountApp.get('/years', (req, res, next) => {
+  req.sql.from('escola')
+  .field('DISTINCT escola.ano_censo', 'year');
+  next();
+}, query, response('years'));
+
+classCountApp.get('/adm_dependency', (req, res, next) => {
+  req.result = [];
+  for(let i = 1; i <= 4; ++i) {
+    req.result.push({
+      id: i,
+      name: id2str.admDependency(i)
+    });
+  };
+  next();
+}, response('adm_dependency'));
+
+classCountApp.get('/adm_dependency_detailed', (req, res, next) => {
+  req.result = [];
+  for(let i = 1; i <= 6; ++i) {
+    req.result.push({
+      id: i,
+      name: id2str.admDependencyPriv(i)
+    });
+  };
+  next();
+}, response('adm_dependency_detailed'));
+
+classCountApp.get('/location', (req, res, next) => {
+  req.result = [
+    {id: 1, name: 'Urbana'},
+    {id: 2, name: 'Rural'}
+  ];
+  next();
+}, response('location'));
+
+classCountApp.get('/rural_location', (req, res, next) => {
+  req.result = [
+    {id: 1, name: "Urbana"},
+    {id: 2, name: "Rural"},
+    {id: 3, name: "Rural - Área de assentamento"},
+    {id: 4, name: "Rural - Terra indígena"},
+    {id: 5, name: "Rural - Área remanescente de quilombos"},
+    {id: 6, name: "Rural - Unidade de uso sustentável"}
+  ];
+  next();
+}, response('rural_location'));
+
+classCountApp.get('/education_level_mod', (req, res, next) => {
+  req.result = [];
+  for(let i = 1; i <= 10; ++i) {
+    req.result.push({
+      id: i,
+      name: id2str.educationLevelMod(i)
+    });
+  }
+  req.result.push({
+    id: 99,
+    name: id2str.educationLevelMod(99)
+  });
+  next();
+}, response('education_level_mod'));
+
+classCountApp.get('/education_level_short', (req, res, next) => {
+  req.result = [];
+  for(let i = 1; i <= 7; ++i) {
+    req.result.push({
+      id: i,
+      name: id2str.educationLevelShort(i)
+    });
+  }
+  req.result.push({
+    id: 99,
+    name: id2str.educationLevelShort(99)
+  });
+  next();
+}, response('education_level_short'));
+
+classCountApp.get('/source', (req, res, next) => {
+  req.sql.from('fonte')
+  .field('fonte', 'source')
+  .where('tabela = \'turma\'');
+  next();
+}, query, response('source'));
+
+// Se a dimensão obrigatória do LDE (etapa de ensino) possui alguma etapa sem nenhum valor, então é criado um objeto
+// com média 0.0 e é inserido no resultado. Usada para não quebrar a sequência de etapas na tabela do LDE.
+function addNullFields(result) {
+  const firstYear = result[0].year;
+  var obj = result.filter(res => res.year == firstYear);
+  var prevFirstDimId = obj[0];
+  obj.forEach((d) => {
+    if((d["education_level_mod_id"] > prevFirstDimId["education_level_mod_id"]) && (prevFirstDimId["education_level_mod_id"] != 10) &&
+       (d["education_level_mod_id"] != prevFirstDimId["education_level_mod_id"] + 1)) {
+      let newObj = {};
+      Object.keys(prevFirstDimId).forEach((key) => {
+        newObj[key] = prevFirstDimId[key];
+      });
+      newObj.education_level_mod_id = d["education_level_mod_id"] - 1;
+      newObj.education_level_mod_name = id2str.educationLevelMod(d["education_level_mod_id"] - 1);
+      newObj.average = 0.0;
+      result.splice(result.indexOf(prevFirstDimId) + 1, 0, newObj);
+    }
+    prevFirstDimId = d;
+  });
+}
+
+function addZeroFields(result) {
+    let i;
+    for (i=0; i < result.length; i++) {
+        let hasTotal = result[i].hasOwnProperty("total");
+        if (hasTotal == true) {
+            result[i].average = 0.0;
+            result[i].median = 0.0;
+            result[i].stddev = 0.0;
+            result[i].first_qt = 0.0;
+            result[i].third_qt = 0.0;
+        }
+    }
+    return result;
+}
+
+// SimCAQ
+classCountApp.get('/count', rqf.parse(), (req, res, next) => {
+  req.sql.field("'Brasil'", 'name')
+   .field('turma.ano_censo', 'year')
+   .field('AVG(turma.num_matricula)', 'average')
+   .field('MEDIAN(turma.num_matricula)', 'median')
+   .field('STDDEV_POP(turma.num_matricula)', 'stddev')
+   .field('QUANTILE(turma.num_matricula, 0.25)', 'first_qt')
+   .field('QUANTILE(turma.num_matricula, 0.75)', 'third_qt')
+   .from('turma')
+   .group('turma.ano_censo')
+   .order('turma.ano_censo')
+   .where('turma.local_turma = 0 AND turma.dependencia_adm_id <= 3 AND ((turma.etapa_resumida >= 1 AND turma.etapa_resumida <= 7) OR turma.etapa_resumida = 99) AND turma.turma_turno_id <> 99')
+   next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('class_count'));
+
+// LDE
+// Essa rota não esta sendo usada.
+// Ela está com defeito e quando é chamada cai o banco.
+/*
+classCountApp.get('/', rqf.parse(), (req, res, next) => {
+  // Faz a consulta do número de alunos pelas dimensões
+  if(("education_level_mod" in req.dims) || ("education_level_mod" in req.filter) || ("education_level_short" in req.dims)) {
+    req.sql.field("'Brasil'", 'name')
+     .field('turma.ano_censo', 'year')
+     .field('AVG(turma.num_matricula)', 'average')
+     .field('MEDIAN(turma.num_matricula)', 'median')
+     .field('STDDEV_POP(turma.num_matricula)', 'stddev')
+     .field('QUANTILE(turma.num_matricula, 0.25)', 'first_qt')
+     .field('QUANTILE(turma.num_matricula, 0.75)', 'third_qt')
+     .from('turma')
+     .group('turma.ano_censo')
+     .order('turma.ano_censo')
+     .where('turma.tipo_turma_id = 0 AND turma.etapas_mod_ensino_segmento_id >= 1 AND turma.etapas_mod_ensino_segmento_id <= 10');
+     next();
+  } else {
+    res.status(400);
+    next({
+      status: 400,
+      message: 'Wrong/No filter specified'
+    });
+  }
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.partial = [];
+
+    // Caso tenha apenas uma dimensão, o indicador possuirá uma linha de total
+    if((req.dims) && (req.dims.size == 1)) {
+      req.partial = req.result;
+
+      // A linha de total deverá conter o valor total do ano que está no banco de dados, então usa o mesmo filtro de anos da consulta anterior
+      let yearFilter = {};
+      if("min_year" in req.filter)
+        yearFilter.min_year = req.filter.min_year;
+      if("max_year" in req.filter)
+        yearFilter.max_year = req.filter.max_year;
+
+      // Faz a consulta sem dimensões, do total do(s) ano(s) escolhido(s)
+      req.resetSql();
+      req.dims = {};
+      req.filter = yearFilter;
+      req.sql.field("'Brasil'", 'name')
+       .field('turma.ano_censo', 'year')
+       .field('AVG(turma.num_matricula)', 'average')
+       .field('MEDIAN(turma.num_matricula)', 'median')
+       .field('STDDEV_POP(turma.num_matricula)', 'stddev')
+       .field('QUANTILE(turma.num_matricula, 0.25)', 'first_qt')
+       .field('QUANTILE(turma.num_matricula, 0.75)', 'third_qt')
+       .from('turma')
+       .group('turma.ano_censo')
+       .order('turma.ano_censo')
+       .where('turma.tipo_turma_id = 0 AND turma.etapas_mod_ensino_segmento_id >= 1 AND turma.etapas_mod_ensino_segmento_id <= 10');
+     }
+    next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), (req, res, next) => {
+  // Se tem apenas uma dimensão
+  if(req.partial.length > 0) {
+    const yearClassCount = req.result;
+    req.result = req.partial;
+
+    // Como a linha de total deve aparecer em um caso específico, ela é adicionada junto com a dimensão obrigatória
+    yearClassCount.forEach((result) => {
+      let obj = {};
+      obj = result;
+      obj.education_level_mod_name = "Total";
+      req.result.push(obj);
+    })
+  }
+  // Caso tenha mais de uma dimensão, retorna a consulta só pelas dimensões, sem linha de total
+  else {
+    addNullFields(req.result);
+    req.result = addZeroFields(req.result);
+  }
+
+  next();
+}, response('class_count'));
+*/
+
+module.exports = classCountApp;
diff --git a/src/libs/routes_v2/classroom.js b/src/libs/routes_v2/classroom.js
new file mode 100644
index 00000000..8e2b6a7e
--- /dev/null
+++ b/src/libs/routes_v2/classroom.js
@@ -0,0 +1,277 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const classroomApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+let rqfCount = new ReqQueryFields();
+
+//classroomApp.use(cache('15 day'));
+
+
+// Complete range of the enrollments dataset.
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+classroomApp.get('/year_range', (req, res, next) => {
+    req.sql.from('escola')
+    .field('MIN(escola.ano_censo)', 'start_year')
+    .field('MAX(escola.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+classroomApp.get('/years', (req, res, next) => {
+    req.sql.from('escola')
+    .field('DISTINCT escola.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+classroomApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'escola\'');
+    next();
+}, query, response('source'));
+
+classroomApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+classroomApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+classroomApp.get('/location', (req, res, next) => {
+    req.result = [
+		{id: 1, name: 'Urbana'},
+		{id: 2, name: 'Rural'}
+	];
+	next();
+}, response('location'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}, 'filter').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'filter').addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'adm_dependency',
+    table: 'escola',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'escola',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'location',
+    table: 'escola',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+});
+
+classroomApp.get('/', cache('15 day'), rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.from('escola')
+        .field('SUM(escola.qtde_salas_utilizadas_dentro)', 'total')
+        .field("'Brasil'", 'name')
+        .field('escola.ano_censo', 'year')
+        .group('escola.ano_censo')
+        .order('escola.ano_censo')
+        .where('escola.situacao_de_funcionamento = 1 AND escola.local_func_predio_escolar = 1')
+        .where('escola.ensino_regular = 1 OR escola.ensino_eja = 1 OR escola.educacao_profissional = 1');
+    next();
+}, query, addMissing(rqf), id2str.transform(), (req, res, next) => {
+    if (req.dims.location && req.result.length < 2) {    // Garantimos que conterá as duas localizações no resultado para o simCAQ
+        let result = [];
+        for (let i = 1; i <= 2; i++) {
+            if (i !== req.result[0].location_id) {
+                let newObj = Object.assign({}, req.result[0]);
+                newObj.location_id = i;
+                newObj.total = 0;
+                newObj.location_name = (i == 1) ? "Urbana" : "Rural";
+                result.push(newObj);
+            }
+            else {
+                result.push(req.result[0]);
+            }
+        }
+
+        req.result = result;
+    }
+
+    next();
+}, response('classroom'));
+
+module.exports = classroomApp;
diff --git a/src/libs/routes_v2/classroomCount.js b/src/libs/routes_v2/classroomCount.js
new file mode 100644
index 00000000..4f76bed3
--- /dev/null
+++ b/src/libs/routes_v2/classroomCount.js
@@ -0,0 +1,1172 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+
+const express = require('express');
+
+const classroomCountApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'filter').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'dims').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: 'nome',
+    resultField: 'state_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}, 'filter').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id',
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id',
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: '@'
+    }
+}, 'dims').addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'school_year',
+    table: '@',
+    tableField: 'serie_ano_id',
+    resultField: 'school_year_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'serie_ano_id'
+    }
+}).addValue({
+    name: 'location',
+    table: '@',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'period',
+    table: '@',
+    tableField: 'turma_turno_id',
+    resultField: 'period_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'turma_turno_id'
+    }
+}).addValue({
+    name: 'school_building',
+    table: 'escola',
+    tableField: 'local_func_predio_escolar',
+    resultField: 'school_building',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'local_func_predio_escolar'
+    }
+}).addValue({
+    name: 'night_time',
+    table: 'matricula',
+    tableField: 'periodo_noturno',
+    resultField: 'night_time',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'periodo_noturno'
+    }
+}).addValue({
+    name: 'formation_level',
+    table: 'docente_por_formacao',
+    tableField: 'tipo_formacao',
+    resultField: 'formation_level',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'tipo_formacao'
+    }
+}).addValueToField({
+    name: 'adm_dependency',
+    table: '@',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}, 'filter') .addValue({
+    name: 'integral_time',
+    table: 'matricula',
+    tableField: 'tempo_integral',
+    resultField: 'integral_time_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'tempo_integral'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'matricula',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+});
+
+classroomCountApp.post('/', rqf.parse(), (req, res, next) => {
+    let classSize = JSON.parse(req.body.class_size) || null;
+    let integralTime = JSON.parse(req.body.integral_time) || null;
+    let enrollmentProjection = (req.body.projections !== undefined) ? JSON.parse(req.body.projections) : null;
+
+    if(classSize == null || integralTime == null) {
+        res.statusCode = 400;
+        return res.json({err: {message: "There was an error processing class_size or integral_time. Check your JSON sintax and be sure you're sending both paramenters."}});
+    }
+    req.classSize = classSize;
+    req.integralTime = integralTime;
+    req.projections = (enrollmentProjection !== null) ? enrollmentProjection : false;
+
+    req.dims.state = true;
+    req.dims.city = true;
+    req.dims.school_year = true;
+    req.dims.location = true;
+
+    req.sql.field('sum(dia_total)', 'total_day')
+    .field('sum(noite_total)', 'total_night')
+    .field("'Brasil'", 'name')
+    .field('matricula_por_localizacao.ano_censo', 'year')
+    .from('matricula_por_localizacao')
+    .where('matricula_por_localizacao.serie_ano_id < 15')
+    .group('matricula_por_localizacao.ano_censo')
+    .order('matricula_por_localizacao.ano_censo')
+
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.enrollment = req.result;
+    
+    // Gera a relação etapa de ensino X ano escolar
+    req.educationSchoolYear = {};
+    for(let i = 1; i < 15; ++i) {
+        if(id2str.educationLevelSchoolYear(i) !== id2str.educationLevelSchoolYear(99)) {
+            let educationLevelId = (i > 10) ? Math.floor(i/10) : i;
+            
+            let classSize = req.classSize.find((el) => {return el.id === educationLevelId || el.id === i});
+            let integralTime = req.integralTime.find((el) => {return el.id === educationLevelId});
+            
+            let numberStudentClass = (typeof classSize !== 'undefined') ? classSize.numberStudentClass : null;
+            // Usa o offerGoal que é passado como parâmetro (Brasil inteiro). Atenção para isso.
+            let integralTimeOfferGoal = (typeof integralTime !== 'undefined') ? integralTime.offerGoal : null;
+            
+            let teacherByClass = (typeof classSize !== 'undefined') ? classSize.teacherByClass : null;
+
+            req.educationSchoolYear[i] = {
+                id: educationLevelId,
+                name: id2str.educationLevelShort(educationLevelId),
+                numberStudentClass,
+                integralTimeOfferGoal,
+                teacherByClass
+            };
+        }
+    }
+    
+    delete req.dims;
+    delete req.filter;
+    req.resetSql();
+    next();
+}, rqf.parse(), (req, res, next) => {
+
+    req.dims.state = true;
+    req.dims.city = true;
+    req.dims.location = true;
+    req.dims.school_building = true;
+
+    req.sql.field('SUM(escola.qtde_salas_utilizadas_dentro)', 'total')
+    .field("'Brasil'", 'name')
+    .field('escola.ano_censo', 'year')
+    .from('escola')
+    .group('escola.ano_censo')
+    .order('escola.ano_censo')
+    .where('escola.situacao_de_funcionamento = 1')
+    .where('escola.dependencia_adm_id < 4') 
+    .where('escola.ensino_regular = 1 OR escola.ensino_eja = 1 OR escola.educacao_profissional = 1');
+
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) =>  {
+    req.classroom = req.result;
+
+    delete req.dims;
+    delete req.filter;
+    req.resetSql();
+    next();
+}, rqf.parse(), (req, res, next) => {
+    if (req.body.teacher_journey !== undefined) {
+        // req.teacherJourney = JSON.parse(req.body.teacher_journey) || null;
+
+        let schoolDays = JSON.parse(req.body.school_days) || null;
+        let teacherFormation = JSON.parse(req.body.teacher_formation) || null;
+        let teachingHours = JSON.parse(req.body.teaching_hours) || null;
+
+        let schoolDaysParsed = [];
+        let teachingHoursParsed = [];
+        for(let i = 1; i <= 6; i++) {   // Adciona turnos faltantes, corrige ordem dos id's
+            schoolDaysParsed.push(schoolDays.find((el) => {return el.id === i}))
+
+            let schoolYear = teachingHours.find((el) => {return el.id === i})
+            let shifts = []
+            for (let j = 1; j <= 3; j++) {
+                let turno = schoolYear.shifts.find((el) => {return el.id === j})
+                if (turno === undefined) {
+                    turno = {"id": j, value: 0}
+                }
+
+                shifts.push(turno)
+            }
+            schoolYear.shifts = shifts
+            teachingHoursParsed.push(schoolYear)
+        }
+
+        req.schoolDays = schoolDaysParsed;
+        req.teachingHours = teachingHoursParsed;
+
+        let teacherFormationParsed = []
+        for (let i of [2, 4, 6, 7, 8]) {
+            teacherFormationParsed.push(teacherFormation.find((el) => {return el.idFormationLevel === i}))
+        }
+        req.teacherFormation = teacherFormationParsed;
+        
+        req.teacherCalc = true;
+    }
+
+    req.dims.state = true;
+    req.dims.city = true;
+    req.dims.formation_level = true;
+    req.sql.field('count(distinct docente_por_formacao.id_docente)', 'total')
+           .field("'Brasil'", 'name')
+           .field('docente_por_formacao.ano_censo', 'year')
+           .from('docente_por_formacao')
+           .group('docente_por_formacao.ano_censo')
+           .order('docente_por_formacao.ano_censo');
+
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) =>  {
+    req.teacher = req.result;
+
+    delete req.dims;
+    delete req.filter;
+    req.resetSql();
+    next();
+}, rqf.parse(), (req, res, next) => {
+    req.dims.state = true;
+    req.dims.city = true;
+    req.dims.education_level_short = true;
+    req.dims.integral_time = true;
+    req.sql.field('COUNT(*)', 'total')
+           .field('matricula.ano_censo', 'year')
+           .from('matricula')
+           .group('matricula.ano_censo')
+           .order('matricula.ano_censo')
+           .where('((matricula.tipo<=3 OR matricula.tipo IS NULL) AND (matricula.tipo_atendimento_turma IS NULL OR matricula.tipo_atendimento_turma <= 2) AND matricula.turma_turno_id <> 99)');
+    next();
+}, rqf.build() ,query, id2str.transform(), (req, res, next) => {
+    // constrói objeto de tempo integral e calcula diagnósticos
+    var integral_time_result = req.result
+    req.integral_time = {}
+    for (let i = 0; i < integral_time_result.length; ++i){
+        // Se cidade não foi criada, cria
+        let integral_time = integral_time_result[i];
+        let code = '' + integral_time.year + integral_time.city_id
+        if (req.dims.school) code = code + integral_time.school_id
+
+        let currentCodeObj = req.integral_time[code]
+
+        // Cria município/escola caso não exista
+        if (currentCodeObj === undefined){
+            let obj = {
+                year: integral_time.year,
+                city_id: integral_time.city_id,
+                city_name: integral_time.city_name,
+                integral_time: {}
+            }
+            if (req.dims.school){
+                obj.school_id = integral_time.school_id
+                obj.school_name = integral_time.school_name
+            }
+            req.integral_time[code] = obj
+            currentCodeObj = obj;
+        }
+
+        // Pega objeto com dados do nível atual
+        var level = integral_time.education_level_short_id
+        var levelObj = currentCodeObj.integral_time[level]
+        if (levelObj === undefined){
+            levelObj = {
+                id: level,
+                integralTime: 0,
+                total: 0,
+            }
+        }
+
+        // Soma no total em integralTime, caso seja 1
+        levelObj.total += integral_time.total
+        if (integral_time.integral_time_id === 1){
+            levelObj.integralTime += integral_time.total
+        }
+
+        currentCodeObj.integral_time[level] = levelObj
+
+    }
+    for (let i in req.integral_time){
+        let curObj = req.integral_time[i]
+        // Adiciona diagnóstico aos objetos
+        for (let key in curObj.integral_time){
+            let cur = curObj.integral_time[key]
+            cur["diagnosis"] = cur.total ? (cur.integralTime/cur.total)*100.0 : 0.0;
+            curObj.integral_time[key] = cur
+        }
+    }
+
+    delete req.dims;
+    delete req.filter;
+    req.resetSql();
+    next()
+}, rqf.parse(), rqf.build(), (req, res, next) => {
+
+    // Talvez dê para remover todos os req.projections.
+    let enrollmentProjectionNight = [];
+    let enrollmentProjectionDay = [];
+    if (req.projections) {
+        for (let i = 0; i < req.projections.length; i++) {
+            if (req.projections[i].periods === 3) {
+                enrollmentProjectionNight.push(req.projections[i]);
+            } else {
+                enrollmentProjectionDay.push(req.projections[i]);
+            }
+        }
+    }
+    let qntYears = 1;
+    if (req.projections) {
+       qntYears = enrollmentProjectionDay[0].stagesEnrollments[0].seriesEnrollments[0].enrollments.length;
+    }
+    let result = [];
+
+    // Cria estrutura de resposta requisitada (itera por anos de projeção):
+    for(let yearCount = 0; yearCount < qntYears; yearCount++) {
+        let i = 0;
+        let j = 0;
+        let hashSet = new Set();
+        let enrollments = [...req.enrollment];
+
+        let ti = 0;     // index for teacher table
+
+        while (i < req.classroom.length) {
+            let classroom = req.classroom[i];
+            // Cria hash única para cada espacialidade, dado um ano
+            let hash = '' + yearCount + classroom.state_id + classroom.city_id; 
+            if (req.dims.school)
+                hash = hash + classroom.school_id
+            // Estrutura do objeto do resultado final
+            let obj = {
+                year: classroom.year + yearCount,
+                name: classroom.name,
+                state_id: classroom.state_id,
+                state_name: classroom.state_name,
+                city_id: classroom.city_id,
+                city_name: classroom.city_name,
+            };
+            if (req.dims.school){
+                obj.school_id = classroom.school_id,
+                obj.school_name = classroom.school_name
+            }
+            if (req.teacherCalc)
+                obj.percentage_teacher_career = [];
+            obj.locations = []
+
+            // Inserimos a localidade no array de locations da sala
+            let location = {
+                location_id: classroom.location_id,
+                location_name: classroom.location_name,
+                total_classroom: (classroom.school_building == 1) ? parseInt(classroom.total, 10) : 0,      // Conta apenas salas de prédios próprios
+                total_classroom_be_built: 0,
+                education_level: []
+            };
+            
+            // Ajusta estrutura de localização do último [município/escola] e inicializa variáveis para o atual
+            let currentClassroomObj = null;
+            if( !hashSet.has(hash) ) { // Nunca passou por esse município/escola
+                if (result[result.length - 1] !== undefined) {      // Após mudar de cidade, passamos pela cidade anterior e juntamos o valor to_be_built de localizações com o mesmo id
+                    let last_city = result[result.length - 1]
+                    let last_locations = result[result.length - 1].locations
+                    for (let i = 0; i < last_locations.length - 1; i++) {
+                        if (last_locations[i].location_id === last_locations[i+1].location_id) {
+                            last_locations[i].total_classroom_be_built += last_locations[i+1].total_classroom_be_built;
+                            last_locations[i].total_classroom += last_locations[i+1].total_classroom;
+                            last_locations.splice(i+1, 1);
+                        }
+                    }
+                    for (let i = 0; i < last_locations.length; i++) {   // Passamos agora para não deixar to_be_built < 0
+                        last_locations[i].total_classroom_be_built = (last_locations[i].total_classroom_be_built < 0) ? 0 : last_locations[i].total_classroom_be_built;
+                    }
+
+                    if (req.teacherCalc) executeTeacherCalc(last_city, ti);
+                }
+                hashSet.add(hash);
+                result.push(obj);
+                currentClassroomObj = obj;
+
+                var id_attribute = req.dims.school ? "school_id" : "city_id"
+                
+                var old_ti = ti;
+                while (ti < req.teacher.length && req.teacher[ti][id_attribute] !== classroom[id_attribute]) // match da tabela de professores.
+                    ti++;
+                
+                if (ti === req.teacher.length) {
+                    console.log(classroom[id_attribute], "not found")
+                    while (classroom[id_attribute] === enrollments[j][id_attribute])
+                        enrollments.splice(j, 1)
+                    ti = old_ti; 
+                    i++;
+                    continue;
+                }
+
+            } else { // Se a hash já existe, já temos a cidade nos resultados. Como está ordenado, é o último valor nos resultados
+                currentClassroomObj = result[result.length - 1];
+            }
+            currentClassroomObj.locations.push(location);
+
+            // Partimos para as etapas de ensino/anos escolares
+            let enrollmentMatch = true;
+            j = 0;
+            let educationLevelSet = new Set();
+            let schoolYearSet = new Set();
+            let enrollment;
+            while(enrollmentMatch && j < enrollments.length) {
+                enrollment = enrollments[j];
+
+                if(typeof enrollment === 'undefined') {
+                    ++j;
+                    continue;
+                }
+                
+                if (req.dims.school){
+                    if(classroom.school_id !== enrollment.school_id) { // Se as escolas não são iguais, já passamos do range
+                        enrollmentMatch = false;
+                        continue;
+                    } 
+                }
+                else{
+                    if(classroom.city_id !== enrollment.city_id) { // Se as cidades não são iguais, já passamos do range
+                        enrollmentMatch = false;
+                        continue;
+                    } 
+                }
+
+                if(enrollment.year != classroom.year || enrollment.location_id != classroom.location_id) { // Se ano ou localização são diferentes, passa para o próximo
+                    ++j;
+                    continue;
+                }
+
+
+                // Temos uma matrícula com cidade/escola, ano e localidades certos
+                // "Consome" a matrícula (remove do vetor de matrículas)
+                enrollments.splice(j, 1);
+
+                // Cria a etapa de ensino adequada
+                let enrollmentEducationLevel = req.educationSchoolYear[enrollment.school_year_id];
+                // Se não há um número de alunos por turna para a etapa de ensino, ignoramos a entrada
+
+                if(enrollmentEducationLevel.numberStudentClass == null) continue;
+                
+                // Adiciona nível de educação para município/escola
+                let educationLevel = null;
+                if(!educationLevelSet.has(enrollmentEducationLevel.id)) { // cria e insere ordenadamente novo education level
+                    educationLevelSet.add(enrollmentEducationLevel.id);
+                    
+                    let itHash = '' + enrollment.year + enrollment.city_id
+                    if (req.dims.school) itHash += enrollment.school_id
+    
+                    let cur_education_level_short = enrollmentEducationLevel.id
+    
+                    let level_diagnosis = 0
+                    let integral_time = 0;
+                    let integral_time_total = 0;
+                    if (req.integral_time[itHash].integral_time[cur_education_level_short]){
+                        level_diagnosis = parseFloat( (req.integral_time[itHash].integral_time[cur_education_level_short].diagnosis).toFixed(2) )
+                        integral_time = req.integral_time[itHash].integral_time[cur_education_level_short].integralTime
+                        integral_time_total = req.integral_time[itHash].integral_time[cur_education_level_short].total
+                    }
+
+                    educationLevel = {
+                        education_level_short_id: enrollmentEducationLevel.id,
+                        education_level_short_name: enrollmentEducationLevel.name,
+                        enrollment: {
+                            integral_percentage: req.dims.school ? level_diagnosis : Math.max(level_diagnosis, enrollmentEducationLevel.integralTimeOfferGoal),
+                            integral_time: req.dims.school ? integral_time : Math.round(integral_time_total * Math.max(level_diagnosis, enrollmentEducationLevel.integralTimeOfferGoal)/100),
+                            integral_time_total: integral_time_total,
+                            total_enrollment_day: 0,
+                            total_enrollment_night: 0,
+                            full_period_classes: 0,
+                            day_classes: 0,
+                            night_classes: 0,
+                            total_classrooms_needed: 0
+                        }
+                    };
+
+                    if(enrollmentEducationLevel.id == 1) {
+                        educationLevel.classes_school_year = [];
+                    }
+
+                    // Para manter a ordem da etapa de ensino (insertion sort)
+                    if (location.education_level.length == 0) {
+                        location.education_level.push(educationLevel);
+                    } else {
+                        let k = location.education_level.length - 1;
+                        let el = location.education_level[k];
+                        while (k >= 0) {
+                            if(educationLevel.education_level_short_id < el.education_level_short_id) {
+                                --k;
+                                if(k>=0) el = location.education_level[k];
+                            } else break;
+                        }
+                        k++;
+                        location.education_level.splice(k, 0, educationLevel);
+                    }
+                } else { // pega o objeto de education level já existente
+                    let k = 0;
+                    let el = location.education_level[k];
+                    while(k < location.education_level.length) {
+                        if(el.education_level_short_id != enrollmentEducationLevel.id) {
+                            ++k;
+                            if(k<location.education_level.length) el = location.education_level[k];
+                        } else break;
+                    }
+                    if(k >= location.education_level.length) --k;
+                    educationLevel = location.education_level[k];
+                }
+
+                // Adiciona as séries da creche
+                let currentSchoolYear = null;
+                if(enrollmentEducationLevel.id == 1){
+                    let schoolYearHash = '' + enrollment.year + enrollment.city_id + enrollment.location_id + enrollment.school_year_id;
+                    if (req.dims.shool) schoolYearHash = schoolYearHash + enrollment.shcool_id
+
+                    if(schoolYearSet.has(schoolYearHash)) { // Busca a série escolar
+                        let k = 0;
+                        let el = educationLevel.classes_school_year[k];
+                        while(k < educationLevel.classes_school_year.length) {
+                            if(el.school_year_id != enrollment.school_year_id) {
+                                ++k;
+                                if(k < educationLevel.classes_school_year.length) el = educationLevel.classes_school_year[k];
+                            } else break;
+                        }
+                        if(k >= educationLevel.classes_school_year.length) --k;
+                        currentSchoolYear = educationLevel.classes_school_year[k];
+                    } else { // Adiciona uma nova série escolar
+                        let school_year = {
+                            school_year_id: enrollment.school_year_id,
+                            school_year_name: enrollment.school_year_name,
+                            total_enrollment_day: 0,
+                            total_enrollment_night: 0,
+                            full_period_classes: 0,
+                            day_classes: 0,
+                            night_classes: 0,
+                            total_classrooms_needed: 0
+                        }
+                        schoolYearSet.add(schoolYearHash);
+                        // Busca a posição para inserir
+                        let k = 0;
+                        let el = educationLevel.classes_school_year[k];
+                        while((typeof el !== 'undefined') && school_year.school_year_id > el.school_year_id) {
+                            el = educationLevel.classes_school_year[++k];
+                        }
+                        // educationLevel.classes_school_year.push(school_year);
+                        educationLevel.classes_school_year.splice(k, 0, school_year);
+                        currentSchoolYear = school_year;
+                    }
+                }
+
+                let currentIntegralOfferGoal = educationLevel.enrollment.integral_percentage;
+                
+                let currentNumberStudentClass = (enrollment.location_id == 1) ? enrollmentEducationLevel.numberStudentClass.urban : enrollmentEducationLevel.numberStudentClass.country;
+                
+                // Soma os totais de matrícula da etapa de ensino
+                educationLevel.enrollment.total_enrollment_day += enrollment.total_day;
+                educationLevel.enrollment.total_enrollment_night += (educationLevel.education_level_short_id > 2) ? enrollment.total_night : 0;      //Não contamos matrículas noturnos de pré-escola e creche
+
+                // Calcula o número de turmas parcial
+                // Turmas de período integral
+                educationLevel.enrollment.full_period_classes = parseFloat(((educationLevel.enrollment.total_enrollment_day * (currentIntegralOfferGoal/100)) / currentNumberStudentClass).toFixed(2));
+
+                // Turmas diurnas (matrículas diurnas - matrículas integrais)
+                educationLevel.enrollment.day_classes = parseFloat(((educationLevel.enrollment.total_enrollment_day * (1 - currentIntegralOfferGoal/100)) / currentNumberStudentClass).toFixed(2));
+
+                // Turmas noturnas
+                educationLevel.enrollment.night_classes = parseFloat((educationLevel.enrollment.total_enrollment_night / currentNumberStudentClass).toFixed(2)) || 0;
+
+                // Total de salas
+                educationLevel.enrollment.total_classrooms_needed = (educationLevel.enrollment.full_period_classes + educationLevel.enrollment.day_classes/2);
+
+                if(educationLevel.enrollment.night_classes > (educationLevel.enrollment.day_classes/2)) educationLevel.enrollment.total_classrooms_needed += (educationLevel.enrollment.night_classes - (educationLevel.enrollment.day_classes/2));
+
+                educationLevel.enrollment.total_classrooms_needed = parseFloat(educationLevel.enrollment.total_classrooms_needed.toFixed(2));
+
+                
+                let currentEnrollmentOfferDay;
+                let currentEnrollmentOfferNight;
+                if (req.projections) {
+                    currentEnrollmentOfferDay = enrollmentProjectionDay[enrollment.location_id - 1].stagesEnrollments[educationLevel.education_level_short_id - 1];
+                    currentEnrollmentOfferNight = enrollmentProjectionNight[enrollment.location_id - 1].stagesEnrollments[educationLevel.education_level_short_id - 1];
+                }
+
+                // Faz os mesmos cálculos para a série escolar
+                if(currentSchoolYear) {
+                    // Totais de matrícula
+                    currentSchoolYear.total_enrollment_day += enrollment.total_day;
+
+                    // Número de turmas parcial
+                    currentSchoolYear.full_period_classes = parseFloat(((currentSchoolYear.total_enrollment_day * (currentIntegralOfferGoal/100)) / currentNumberStudentClass).toFixed(2));
+
+                    currentSchoolYear.day_classes = parseFloat(((currentSchoolYear.total_enrollment_day * (1 - currentIntegralOfferGoal/100)) / currentNumberStudentClass).toFixed(2));
+
+
+                    // Total de salas
+                    currentSchoolYear.total_classrooms_needed = (currentSchoolYear.full_period_classes + currentSchoolYear.day_classes/2);
+
+                    currentSchoolYear.total_classrooms_needed = parseFloat(currentSchoolYear.total_classrooms_needed.toFixed(2));
+
+                    function reducer(key) { return (sum, elem) => sum + elem[key]}
+                    educationLevel.enrollment.total_enrollment_day = educationLevel.classes_school_year.reduce(reducer('total_enrollment_day'), 0);
+                    educationLevel.enrollment.full_period_classes = educationLevel.classes_school_year.reduce(reducer('full_period_classes'), 0);
+                    educationLevel.enrollment.day_classes = educationLevel.classes_school_year.reduce(reducer('day_classes'), 0);
+                    educationLevel.enrollment.total_classrooms_needed = educationLevel.classes_school_year.reduce(reducer('total_classrooms_needed'), 0);
+                }
+
+                enrollment = enrollments[j];
+            }
+
+            // Calculamos o total classroom be built para o município usando reduce
+            location.total_classroom_be_built = location.education_level.reduce((total, atual) => {
+                return total + atual.enrollment.total_classrooms_needed;
+            }, 0) - location.total_classroom;
+            ++i;
+        }
+
+        // Tratamento do último resultado, para remover double location, tirar negativo do to_be_built.
+        if (result[result.length - 1] !== undefined) {      // Após mudar de cidade, passamos pela cidade anterior e juntamos o valor to_be_built de localizações com o mesmo id
+            let last_city = result[result.length - 1]
+            let last_locations = result[result.length - 1].locations
+            for (let i = 0; i < last_locations.length - 1; i++) {
+                if (last_locations[i].location_id === last_locations[i+1].location_id) {
+                    last_locations[i].total_classroom_be_built += last_locations[i+1].total_classroom_be_built;
+                    last_locations[i].total_classroom += last_locations[i+1].total_classroom;
+                    last_locations.splice(i+1, 1);
+                }
+            }
+            for (let i = 0; i < last_locations.length; i++) {   // Passamos agora para não deixar to_be_built < 0
+                last_locations[i].total_classroom_be_built = (last_locations[i].total_classroom_be_built < 0) ? 0 : last_locations[i].total_classroom_be_built;
+            }
+
+            if (req.teacherCalc) executeTeacherCalc(last_city, ti);
+        }
+    }
+
+    // Agregar por estado e brasil
+    let reduction = null;
+    
+    if(req.dims.state || (!req.dims.city && !req.dims.school)) { // Se um dos dois acontecer, sabemos que devemos agregar
+        let i = 0;
+        reduction = [];
+        let reductionSet = new Set();
+        while (i < result.length) {
+            let city = result[i];
+            let obj = {
+                year: city.year,
+                name: city.name,
+                count: 1
+            }
+
+            if(req.dims.state) {
+                obj.state_id = city.state_id;
+                obj.state_name = city.state_name;
+            }
+            obj.percentage_teacher_career = []
+            obj.locations = [];
+            
+            let hash = '' + city.year;
+            if(req.dims.state) hash += '' + city.state_id;
+
+            let currentObj = null;
+            if(!reductionSet.has(hash)) {
+                reductionSet.add(hash);
+                reduction.push(obj);
+                currentObj = obj;
+            } else { // Está ordenado, podemos pegar o último
+                currentObj = reduction[reduction.length - 1];
+                currentObj.count++;
+            }
+            
+            if (currentObj.count == 1)
+                currentObj.percentage_teacher_career = city.percentage_teacher_career
+            else { 
+                // Incrementa valores percentuais de cada nivel de carreira no objeto atual
+                currentObj.percentage_teacher_career.forEach((item, ind, thisArr) => {
+                    thisArr[ind].percentage += city.percentage_teacher_career[ind].percentage
+                })
+            }
+
+            // Fazer "merge" do array locations da cidade com o da agregação
+            if(currentObj.locations.length == 0) {
+                // Pode ser que a cidade atual tenha menos localidades que o total (só urbana ou só rural)
+                currentObj.locations = [...city.locations];
+                if (currentObj.locations.length === 1) {        // Se a cidade só tinha uma, adcionamos a outra manualmente.
+                    currentObj.locations[1] = {
+                        location_id: (currentObj.locations[0].location_id === 1) ? 2 : 1,   // Oposto da adcionada
+                        location_name: (currentObj.locations[0].location_id === 1) ? 'Rural' : 'Urbana',    // Oposto da adcionada
+                        total_classroom: 0,
+                        total_classroom_be_built: 0,
+                        education_level: []
+                    }
+                }
+            } else {
+                let j = 0;
+                let k = 0;
+                let cityLocation = null;
+                let currentLocation = null;
+                while((typeof cityLocation !== 'undefined') && (typeof currentLocation !== 'undefined')) {
+                    cityLocation = city.locations[j];
+                    currentLocation = currentObj.locations[k];
+                    if(cityLocation.location_id < currentLocation.location_id) {
+                        ++j;
+                        cityLocation = city.locations[j];
+                        continue;
+                    } else if(cityLocation.location_id > currentLocation.location_id) {
+                        ++k;
+                        currentLocation = currentObj.locations[k];
+                        continue;
+                    }
+
+                    // Faz "merge" do array education_level
+                    // Se a localidade atual não tem o vetor
+                    if(currentLocation.education_level.length == 0) {
+                        currentLocation.education_level = [...cityLocation.education_level];
+                    } else { // Caso já tenha, atualiza os valores
+                        let l = 0;
+                        while(l < cityLocation.education_level.length) {
+                            let cityEducation = cityLocation.education_level[l];
+                            let m = 0;
+                            let currentEducation = currentLocation.education_level[m];
+                            while(m < currentLocation.education_level.length && cityEducation.education_level_short_id > currentEducation.education_level_short_id) {
+                                ++m;
+                                currentEducation = currentLocation.education_level[m];
+                            }
+                            if(m >= currentLocation.education_level.length) --m;
+                            currentEducation = currentLocation.education_level[m];
+
+                            if(currentEducation.education_level_short_id == cityEducation.education_level_short_id) {
+                                currentEducation.enrollment.integral_time += cityEducation.enrollment.integral_time;
+                                currentEducation.enrollment.integral_time_total += cityEducation.enrollment.integral_time_total;
+                                currentEducation.enrollment.total_enrollment_day += cityEducation.enrollment.total_enrollment_day;
+                                currentEducation.enrollment.total_enrollment_night += cityEducation.enrollment.total_enrollment_night;
+                                currentEducation.enrollment.full_period_classes += cityEducation.enrollment.full_period_classes;
+                                currentEducation.enrollment.day_classes += cityEducation.enrollment.day_classes;
+                                currentEducation.enrollment.night_classes += cityEducation.enrollment.night_classes;
+                                currentEducation.enrollment.total_classrooms_needed += cityEducation.enrollment.total_classrooms_needed;
+
+                                // Caso tenha feito o cálculo de professores, atualiza os valores
+                                if (req.teacherCalc) {
+                                    currentEducation.teacherNumber.total_teacher += cityEducation.teacherNumber.total_teacher;
+                                    currentEducation.teacherNumber.total_teacher_full_period += cityEducation.teacherNumber.total_teacher_full_period;
+                                    currentEducation.teacherNumber.total_teacher_partial += cityEducation.teacherNumber.total_teacher_partial;
+
+                                    currentEducation.teacherNumber.careerLevels.forEach((currentLevel, i) => {
+                                        currentLevel.total_teacher_career += cityEducation.teacherNumber.careerLevels[i].total_teacher_career;
+                                        currentLevel.total_teacher_full_period_career += cityEducation.teacherNumber.careerLevels[i].total_teacher_full_period_career;
+                                        currentLevel.total_teacher_partial_career += cityEducation.teacherNumber.careerLevels[i].total_teacher_partial_career;
+                                    })
+                                }
+
+                                // Insere as séries escolares, se existirem
+                                if((typeof cityEducation.classes_school_year !== 'undefined') && (typeof currentEducation.classes_school_year !== 'undefined')) {
+                                    let n = 0;
+                                    let o = 0;
+                                    let currentClass = currentEducation.classes_school_year[n];
+                                    let cityClass = cityEducation.classes_school_year[o];
+                                    while((typeof cityClass !== 'undefined') && (typeof currentClass !== 'undefined')) {
+                                        currentClass = currentEducation.classes_school_year[n];
+                                        cityClass = cityEducation.classes_school_year[o];
+
+                                        // Se a série escolar é menor que a atual, ela não está no vetor, pois o vetor está ordenado e tem range limitado
+                                        if(cityClass.school_year_id < currentClass.school_year_id) {
+                                            currentEducation.classes_school_year.splice(n, 0, cityClass);
+                                            currentClass = currentEducation.classes_school_year[n];
+                                            cityClass = cityEducation.classes_school_year[++o];
+                                            continue;
+                                        } else if(cityClass.school_year_id > currentClass.school_year_id) {
+                                            currentClass = currentEducation.classes_school_year[++n];
+                                            // Se o ano escolar da cidade é  maior que do objeto atual E o vetor de ano escolar do objeto atual
+                                            // acaba, então este ano escolar falta no objeto atual, pois os anos escolares estão ordenados
+                                            if((typeof currentClass == 'undefined') && (typeof cityClass !== 'undefined')) {
+                                                currentEducation.classes_school_year[n] = cityClass;
+                                                currentClass = currentEducation.classes_school_year[n];
+                                            }
+                                            continue;
+                                        }
+
+                                        currentClass.total_enrollment_day += cityClass.total_enrollment_day;
+                                        currentClass.total_enrollment_night += cityClass.total_enrollment_night;
+                                        currentClass.full_period_classes += cityClass.full_period_classes;
+                                        currentClass.day_classes += cityClass.day_classes;
+                                        currentClass.night_classes += cityClass.night_classes;
+                                        currentClass.total_classrooms_needed += cityClass.total_classrooms_needed;
+
+                                        // Caso tenha feito o cálculo de professores, atualiza os valores
+                                        if (req.teacherCalc) {
+                                            currentClass.teacherNumber.total_teacher += cityClass.teacherNumber.total_teacher;
+                                            currentClass.teacherNumber.total_teacher_full_period += cityClass.teacherNumber.total_teacher_full_period;
+                                            currentClass.teacherNumber.total_teacher_partial += cityClass.teacherNumber.total_teacher_partial;
+
+                                            // currentClass.teacherNumber.careerLevels.forEach((currentLevel, i) => {
+                                            //     currentLevel.total_teacher_career += cityClass.teacherNumber.careerLevels[i].total_teacher_career;
+                                            //     currentLevel.total_teacher_full_period_career += cityClass.teacherNumber.careerLevels[i].total_teacher_full_period_career;
+                                            //     currentLevel.total_teacher_partial_career += cityClass.teacherNumber.careerLevels[i].total_teacher_partial_career;
+                                            // })
+                                        }
+    
+                                        cityClass = cityEducation.classes_school_year[++o];
+                                    }
+
+                                }
+                            } else {
+                                if(currentEducation.education_level_short_id < cityEducation.education_level_short_id) {
+                                    currentLocation.education_level.splice(++m, 0, cityEducation);
+                                } else {
+                                    currentLocation.education_level.splice(m, 0, cityEducation);
+                                }
+                            }
+                            ++l;
+                        }
+                    }
+                    // Reinicia vetor percentage_teacher_career e calcula porcentagens pelos totais
+
+                    currentLocation.total_classroom += cityLocation.total_classroom;
+                    currentLocation.total_classroom_be_built += cityLocation.total_classroom_be_built;
+
+                    ++j;
+                    cityLocation = city.locations[j];
+                }
+            }
+            ++i;
+        }
+        for (let state of reduction){
+            state.percentage_teacher_career.forEach((item, ind, thisArr) => {
+                thisArr[ind].percentage = Number((thisArr[ind].percentage / state.count).toFixed(2))
+            })
+            delete state.count
+            for (let location of state.locations){
+                for (let educationLevel of location.education_level){
+                    let total = educationLevel.enrollment.integral_time_total;
+                    let integralTime = educationLevel.enrollment.integral_time;
+                    educationLevel.enrollment.integral_percentage = total ? parseFloat( (100*(integralTime / total)).toFixed(2) ) : 0       
+                }
+            }
+        }
+    }
+
+    req.result = reduction || result;
+
+
+    function executeTeacherCalc(lastCity, index) {
+        let lastLocations = lastCity.locations
+        let teacherByFormation = [];        // Vetor com a porcentagem de professores por formação.
+        let teacherTotal = req.teacher[index].total;
+        
+        let i = index + 1;
+        for (let j of [2, 4, 6, 7, 8]) {       // Calcula a porcentagem de professores por formação.
+            if (req.teacher[i] === undefined || j !== req.teacher[i].formation_level) {         // Nesse caso a cidade não possuí professores com a formação atual, adcionamos 0.
+                teacherByFormation.push(0);
+            }
+            else {
+                teacherByFormation.push(parseFloat((req.teacher[i].total / teacherTotal).toFixed(3)));
+                i++;
+            }
+        }
+
+        // verifica se a soma de porcentagens vale 100.
+        let sum = 0;
+        for (let value of teacherByFormation) {
+            sum += value;
+        }
+        teacherByFormation[1] += teacherByFormation[0]
+        teacherByFormation[0] = 0
+
+        let diff = 1 - sum;
+        
+        // Se soma de porcentagens for menor/maior que 100, faz correção
+        if (Math.abs(diff) > 0.0001) {
+            // Garante que a porcentagem corrigida não ficará negativa
+            let indDiff = 1;
+            while (teacherByFormation[indDiff] + diff < 0) indDiff++;
+            teacherByFormation[indDiff] += diff;
+        }
+
+        // Cria vetor de porcentagens de carreira dos professores
+        req.teacherFormation.forEach((formation, i) => {
+            lastCity.percentage_teacher_career.push({
+                formation_level_id: formation.idFormationLevel,
+                percentage: Number((teacherByFormation[i]*100).toFixed(2)),
+            })
+        });
+
+        lastLocations.forEach((location) => {
+            location.education_level.forEach((educationLevel) => {
+                let educationLevelId = educationLevel.education_level_short_id;
+
+                let currentTeachingHours = req.teachingHours[educationLevelId-1].shifts;
+        
+                if(educationLevelId === 1) {        // Devido a divisão da creche é necessário tratá-la separadamente.
+                    educationLevel.classes_school_year.forEach((schoolYear) => {        // Aplicamos os cálculos para os anos da creche
+                        let teachingTimeFullPeriod = schoolYear.full_period_classes * currentTeachingHours[2].value * req.schoolDays[educationLevelId-1].value;
+                        let teachingTimeNight = schoolYear.night_classes * currentTeachingHours[1].value * req.schoolDays[educationLevelId-1].value;
+                        let teachingTimeDay = schoolYear.day_classes * currentTeachingHours[0].value * req.schoolDays[educationLevelId-1].value;
+
+                        let currentTeacherByClass = (location.location_id === 1) ? req.educationSchoolYear[schoolYear.school_year_id].teacherByClass.urban : req.educationSchoolYear[schoolYear.school_year_id].teacherByClass.country;
+                        
+                        let numberOfTeacherFullPeriod = 0;
+                        let numberOfTeacherNight = 0;
+                        let numberOfTeacherDay = 0;
+                        lastCity.percentage_teacher_career.forEach(career => {
+                            let journeyObj = req.teacherFormation.find(formation => formation.idFormationLevel === career.formation_level_id)
+                            let journey = journeyObj.journeyWithInteraction/100 * journeyObj.journeyTotal;
+                            numberOfTeacherFullPeriod += (teachingTimeFullPeriod / journey) * currentTeacherByClass * career.percentage/100;
+                            numberOfTeacherNight += (teachingTimeNight / journey) * currentTeacherByClass * career.percentage/100;
+                            numberOfTeacherDay += (teachingTimeDay / journey) * currentTeacherByClass * career.percentage/100;
+                        })
+                        numberOfTeacherFullPeriod = parseFloat(numberOfTeacherFullPeriod.toFixed(2));
+                        numberOfTeacherNight = parseFloat(numberOfTeacherNight.toFixed(2));
+                        numberOfTeacherDay = parseFloat(numberOfTeacherDay.toFixed(2));
+
+                        schoolYear.teacherNumber = {
+                            total_teacher : numberOfTeacherDay + numberOfTeacherNight + numberOfTeacherFullPeriod,
+                            total_teacher_full_period : numberOfTeacherFullPeriod,
+                            total_teacher_partial : numberOfTeacherNight + numberOfTeacherDay
+                        }
+                    })
+
+                    // Calculamos para o educationLevel usando reduce
+                    function reducer(key) { return (sum, elem) => sum + elem.teacherNumber[key]}
+                    educationLevel.teacherNumber = {
+                        total_teacher : educationLevel.classes_school_year.reduce(reducer('total_teacher'), 0),
+                        total_teacher_full_period : educationLevel.classes_school_year.reduce(reducer('total_teacher_full_period'), 0),
+                        total_teacher_partial : educationLevel.classes_school_year.reduce(reducer('total_teacher_partial'), 0)
+                    }
+
+                    // function reducerList(idx, key) { return (sum, elem) => sum + elem.teacherNumber.careerLevels[idx][key]}
+                    educationLevel.teacherNumber.careerLevels = [];
+                    req.teacherFormation.forEach((formation, i) => {
+                        let totalTeacherFullPeriodCareer = educationLevel.teacherNumber.total_teacher_full_period * teacherByFormation[i];
+                        let totalTeacherPartialCareer = educationLevel.teacherNumber.total_teacher_partial * teacherByFormation[i];
+
+                        educationLevel.teacherNumber.careerLevels.push({
+                            sequence: formation.sequence,
+                            denomination: formation.denomination,
+                            formation_level_id: formation.idFormationLevel,
+                            total_teacher_career: totalTeacherFullPeriodCareer + totalTeacherPartialCareer,
+                            total_teacher_full_period_career: totalTeacherFullPeriodCareer,
+                            total_teacher_partial_career:totalTeacherPartialCareer,
+                        })
+                    })
+                }
+
+                else {
+                    let teachingTimeFullPeriod = educationLevel.enrollment.full_period_classes * currentTeachingHours[2].value * req.schoolDays[educationLevelId-1].value;
+                    let teachingTimeNight = educationLevel.enrollment.night_classes * currentTeachingHours[1].value * req.schoolDays[educationLevelId-1].value;
+                    let teachingTimeDay = educationLevel.enrollment.day_classes * currentTeachingHours[0].value * req.schoolDays[educationLevelId-1].value;
+
+                    let currentTeacherByClass = (location.location_id === 1) ? req.educationSchoolYear[educationLevelId].teacherByClass.urban : req.educationSchoolYear[educationLevelId].teacherByClass.country;
+        
+                    let numberOfTeacherFullPeriod = 0;
+                    let numberOfTeacherNight = 0;
+                    let numberOfTeacherDay = 0;
+                    lastCity.percentage_teacher_career.forEach(career => {
+                        let journeyObj = req.teacherFormation.find(formation => formation.idFormationLevel === career.formation_level_id)
+                        let journey = journeyObj.journeyWithInteraction/100 * journeyObj.journeyTotal;
+                        numberOfTeacherFullPeriod += (teachingTimeFullPeriod / journey) * currentTeacherByClass * career.percentage/100;
+                        numberOfTeacherNight += (teachingTimeNight / journey) * currentTeacherByClass * career.percentage/100;
+                        numberOfTeacherDay += (teachingTimeDay / journey) * currentTeacherByClass * career.percentage/100;
+                    })
+                    numberOfTeacherFullPeriod = parseFloat(numberOfTeacherFullPeriod.toFixed(2));
+                    numberOfTeacherNight = parseFloat(numberOfTeacherNight.toFixed(2));
+                    numberOfTeacherDay = parseFloat(numberOfTeacherDay.toFixed(2));
+
+                    educationLevel.teacherNumber = {
+                        total_teacher : numberOfTeacherDay + numberOfTeacherNight + numberOfTeacherFullPeriod,
+                        total_teacher_full_period : numberOfTeacherFullPeriod,
+                        total_teacher_partial : numberOfTeacherNight + numberOfTeacherDay
+                    }
+
+                    educationLevel.teacherNumber.careerLevels = [];
+                    req.teacherFormation.forEach((formation, i) => {
+                        let totalTeacherFullPeriodCareer = educationLevel.teacherNumber.total_teacher_full_period * teacherByFormation[i];
+                        let totalTeacherPartialCareer = educationLevel.teacherNumber.total_teacher_partial * teacherByFormation[i];
+
+                        educationLevel.teacherNumber.careerLevels.push({
+                            sequence: formation.sequence,
+                            denomination: formation.denomination,
+                            formation_level_id: formation.idFormationLevel,
+                            total_teacher_career: totalTeacherFullPeriodCareer + totalTeacherPartialCareer,
+                            total_teacher_full_period_career: totalTeacherFullPeriodCareer,
+                            total_teacher_partial_career:totalTeacherPartialCareer,
+                        })
+                    })
+
+                }
+            })
+        })
+    }
+
+    next();
+}, response('classroom_count'));
+
+module.exports = classroomCountApp;
+
diff --git a/src/libs/routes_v2/courseCount.js b/src/libs/routes_v2/courseCount.js
new file mode 100644
index 00000000..404ed2eb
--- /dev/null
+++ b/src/libs/routes_v2/courseCount.js
@@ -0,0 +1,814 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+const { join } = require('lodash');
+
+const courseCountApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+let rqf = new ReqQueryFields();
+
+let rqfMapfor = new ReqQueryFields();
+
+courseCountApp.get('/upper_adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 7; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperAdmDependency(i)
+        });
+    };
+    next();
+}, response('upper_adm_dependency'));
+
+courseCountApp.get('/years', (req, res, next) => {
+    req.sql.from('curso_ens_superior')
+    .field('DISTINCT curso_ens_superior.ano_censo', 'year')
+    .where('curso_ens_superior.ano_censo > 2010');
+    next();
+}, query, response('years'));
+
+courseCountApp.get('/year_range', (req, res, next) => {
+    req.sql.from('curso_ens_superior')
+    .field('MIN(curso_ens_superior.ano_censo)', 'start_year')
+    .field('MAX(curso_ens_superior.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+courseCountApp.get('/academic_organization', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.academicOrganization(i)
+        });
+    };
+    next();
+}, response('academic_organization'));
+
+courseCountApp.get('/ocde_geral', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 8; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ocdeGeral(i)
+        });
+    };
+    next();
+}, response('ocde_geral'));
+
+courseCountApp.get('/ocde_specific', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 1; i <= 86; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.ocdeSpecific(i)
+        };
+        if (obj.name !== id2str.ocdeSpecific(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.ocdeSpecific(defaultCase)
+    });
+    next();
+}, response('ocde_specific'));
+
+courseCountApp.get('/ocde_detailed', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 142; i <= 863; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.ocdeDetailed(i)
+        };
+        if (obj.name !== id2str.ocdeDetailed(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.ocdeDetailed(defaultCase)
+    });
+    next();
+}, response('ocde_detailed'));
+
+courseCountApp.get('/cine_geral', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 10; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.cineGeral(i)
+        });
+    };
+    next();
+}, response('cine_geral'));
+
+courseCountApp.get('/cine_specific', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 1; i <= 104; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.cineSpecific(i)
+        };
+        if (obj.name !== id2str.cineSpecific(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.cineSpecific(defaultCase)
+    });
+    next();
+}, response('cine_specific'));
+
+courseCountApp.get('/cine_detailed', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 11; i <= 1041; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.cineDetailed(i)
+        };
+        if (obj.name !== id2str.cineDetailed(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.cineDetailed(defaultCase)
+    });
+    next();
+}, response('cine_detailed'));
+
+courseCountApp.get('/academic_level', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.academicLevel(i)
+        });
+    };
+    next();
+}, response('academic_level'));
+
+courseCountApp.get('/upper_education_mod', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperEducationMod(i)
+        });
+    };
+    next();
+}, response('upper_education_mod'));
+
+courseCountApp.get('/is_free', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Classificado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('is_free'));
+
+courseCountApp.get('/night_time', (req, res, next) => {
+    req.result = [{
+        id: 9, 
+        name: id2str.nightTime(9)
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.nightTime(i)
+        })
+    }
+    next();
+}, response('night_time'));
+
+courseCountApp.get('/university', (req, res, next) => {
+    req.sql.from('curso_ens_superior')
+    .field('DISTINCT curso_ens_superior.nome_ies', 'nome')
+    .field('curso_ens_superior.cod_ies', 'cod')
+    next();
+}, query, response('university'));
+
+courseCountApp.get('/localoffer', (req, res, next) => {
+    req.sql.from('localoferta_ens_superior', 'l')
+    .field('DISTINCT l.nome', 'localoffer_name')
+    .field('l.cod_local_oferta', 'localoffer_id');
+    next();
+}, query, response('localoffer'));
+
+
+rqfMapfor.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'state',
+    table: 'municipio',
+    tableField: 'estado_id',
+    resultField: 'state_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+	    table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio',
+        foreignTable: 'localoferta_ens_superior'
+    }
+}, 'filter').addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio',
+        foreignTable: 'localoferta_ens_superior'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio',
+        foreignTable: 'localoferta_ens_superior'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['id', 'nome'],
+    resultField: ['city_id', 'city_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio',
+        foreignTable: 'localoferta_ens_superior'
+    }
+}).addValue({
+    name:'academic_level',
+    table: 'curso_ens_superior',
+    tableField: 'cod_grau_academico',
+    resultField: 'academic_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_grau_academico'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'localoferta_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'localoferta_ens_superior',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'localoferta_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'localoferta_ens_superior',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name:'course',
+    table: 'curso_ens_superior',
+    tableField: 'nome_curso',
+    resultField: 'course_name',
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome_curso'
+    }
+}).addValueToField({
+    name: 'campi',
+    table: 'localoferta_ens_superior',
+    tableField: ['cod_local_oferta', 'nome'],
+    resultField: ['campi_id', 'campi_name'],
+    where: {
+	    relation: '=',
+	    type: 'integer',
+	    field: 'cod_local_oferta',
+	    table: 'localoferta_ens_superior'
+    }
+}, 'filter').addValue({
+    name:'upper_adm_dependency',
+    table: 'curso_ens_superior',
+    tableField: 'par_categoria_administrativa',
+    resultField: 'upper_adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_categoria_administrativa'
+    }
+}).addValue({
+    name:'upper_education_mod',
+    table: 'curso_ens_superior',
+    tableField: 'cod_modalidade_ensino',
+    resultField: 'upper_education_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_modalidade_ensino'
+    }
+}).addValue({
+    name:'academic_organization',
+    table: 'curso_ens_superior',
+    tableField: 'cod_organizacao_academica',
+    resultField: 'academic_organization_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_organizacao_academica'
+    }
+})
+
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'region',
+    table: 'localoferta_ens_superior',
+    tableField: ['nome_regiao', 'cod_regiao'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_regiao',
+	    table: 'localoferta_ens_superior'
+    },
+    join: {
+        primary: ['ano_censo', 'cod_curso'],
+        foreign: ['ano_censo', 'cod_curso'],
+        foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'state',
+    table: 'localoferta_ens_superior',
+    tableField: ['sigla_uf_t', 'cod_uf_t'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_uf',
+	    table: 'localoferta_ens_superior'
+    },
+    join: {
+        primary: ['ano_censo', 'cod_curso'],
+        foreign: ['ano_censo', 'cod_curso'],
+        foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_curso',
+        foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_curso',
+        foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'city',
+    table: 'localoferta_ens_superior',
+    tableField: ['cod_municipio_t', 'nome_municipio_t'],
+    resultField: ['city_id', 'city_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_municipio',
+        table: 'localoferta_ens_superior'
+    },
+    join: {
+        primary: ['ano_censo', 'cod_curso'],
+        foreign: ['ano_censo', 'cod_curso'],
+        foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'localoffer',
+    table: 'localoferta_ens_superior',
+    tableField: ['cod_local_oferta', 'nome'],
+    resultField: ['localoffer_id', 'localoffer_name'],
+    where: {
+	    relation: '=',
+	    type: 'integer',
+	    field: 'cod_curso',
+	    table: 'curso_ens_superior'
+    },
+    join: {
+        primary: ['ano_censo', 'cod_curso'],
+	    foreign: ['ano_censo', 'cod_curso'],
+	    foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'campi',
+    table: 'localoferta_ens_superior',
+    tableField: ['cod_local_oferta', 'nome'],
+    resultField: ['campi_id', 'campi_name'],
+    where: {
+	    relation: '=',
+	    type: 'integer',
+	    field: 'cod_curso',
+	    table: 'curso_ens_superior'
+    },
+    join: {
+        primary: ['ano_censo', 'cod_curso'],
+	    foreign: ['ano_censo', 'cod_curso'],
+	    foreignTable: 'curso_ens_superior'
+    }
+}).addValue({
+    name: 'university',
+    table: 'curso_ens_superior',
+    tableField: ['cod_ies', 'nome_ies'],
+    resultField: ['university_id', 'university_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies'
+    }
+}).addValue({
+    name: 'universityLocalOffer',
+    table: 'curso_ens_superior',
+    tableField: ['cod_ies', 'nome_ies'],
+    resultField: ['university_id', 'university_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies'
+    }
+}).addValue({
+    name:'upper_adm_dependency',
+    table: 'curso_ens_superior',
+    tableField: 'par_categoria_administrativa',
+    resultField: 'upper_adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_categoria_administrativa'
+    }
+}).addValue({
+    name:'academic_organization',
+    table: 'curso_ens_superior',
+    tableField: 'cod_organizacao_academica',
+    resultField: 'academic_organization_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_organizacao_academica'
+    }
+}).addValue({
+    name:'course',
+    table: 'curso_ens_superior',
+    tableField: 'nome_curso',
+    resultField: 'course_name',
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome_curso'
+    }
+}).addValue({
+    name:'ocde_specific',
+    table: 'curso_ens_superior',
+    tableField: ['cod_ocde_area_especifica', 'nome_ocde_area_especifica'],
+    resultField: ['ocde_specific_id', 'ocde_specific_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ocde_area_especifica'
+    }
+}).addValue({
+    name:'ocde_geral',
+    table: 'curso_ens_superior',
+    tableField: ['cod_ocde_area_geral', 'nome_ocde_area_geral'],
+    resultField: ['ocde_geral_id', 'ocde_geral_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ocde_area_geral'
+    }
+}).addValue({
+    name:'ocde_detailed',
+    table: 'curso_ens_superior',
+    tableField: ['cod_ocde_area_detalhada', 'nome_ocde_area_detalhada'],
+    resultField: ['ocde_detailed_id', 'ocde_detailed_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ocde_area_detalhada'
+    }
+}).addValue({
+    name:'cine_specific',
+    table: 'curso_ens_superior',
+    tableField: ['cod_cine_area_especifica', 'nome_cine_area_especifica'],
+    resultField: ['cine_specific_id', 'cine_specific_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cine_area_especifica'
+    }
+}).addValue({
+    name:'cine_geral',
+    table: 'curso_ens_superior',
+    tableField: ['cod_cine_area_geral', 'nome_cine_area_geral'],
+    resultField: ['cine_geral_id', 'cine_geral_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cine_area_geral'
+    }
+}).addValue({
+    name:'cine_detailed',
+    table: 'curso_ens_superior',
+    tableField: ['cod_cine_area_detalhada', 'nome_cine_area_detalhada'],
+    resultField: ['cine_detailed_id', 'cine_detailed_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cine_area_detalhada'
+    }
+}).addValue({
+    name:'academic_level',
+    table: 'curso_ens_superior',
+    tableField: 'cod_grau_academico',
+    resultField: 'academic_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_grau_academico'
+    }
+}).addValue({
+    name:'upper_education_mod',
+    table: 'curso_ens_superior',
+    tableField: 'cod_modalidade_ensino',
+    resultField: 'upper_education_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_modalidade_ensino'
+    }
+}).addValue({
+    name:'is_free',
+    table: 'curso_ens_superior',
+    tableField: 'gratuito',
+    resultField: 'is_free_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'gratuito'
+    }
+}).addValue({
+    name:'night_time',
+    table: 'curso_ens_superior',
+    tableField: 'noturno_curso_t',
+    resultField: 'night_time_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'noturno_curso_t'
+    }
+}).addValue({
+    name:'situation',
+    table: 'curso_ens_superior',
+    tableField: 'cod_situacao_curso',
+    resultField: 'situacao_curso_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_situacao_curso'
+    }
+}).addValue({
+    name:'year',
+    table: 'curso_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'curso_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'curso_ens_superior',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'curso_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'curso_ens_superior',
+        field: 'ano_censo'
+    }
+})
+
+courseCountApp.get('/count_by_name', rqfMapfor.parse(), (req, res, next) => {
+    req.sql.from('localoferta_ens_superior')
+    .field('COUNT(DISTINCT localoferta_ens_superior.cod_curso)', 'total')
+    .field('localoferta_ens_superior.ano_censo', 'year')
+    .join ('curso_ens_superior ON (localoferta_ens_superior.cod_curso = curso_ens_superior.cod_curso) AND (localoferta_ens_superior.ano_censo = curso_ens_superior.ano_censo)')
+    .where('curso_ens_superior.cod_nivel_academico = 1')
+    .where('curso_ens_superior.tipo_atributo_ingresso <> 1 OR curso_ens_superior.tipo_atributo_ingresso is NULL')
+    .group('localoferta_ens_superior.ano_censo')
+    .order('localoferta_ens_superior.ano_censo')
+    
+    next();
+}, rqfMapfor.build(), query, (req, res, next) =>{
+    if ('course' in req.dims){
+        var total_course = req.result.reduce((total, cur) => {return total += cur.total}, 0)
+        for (var course of req.result){
+            course.percentage = Number((( course.total / total_course ) * 100).toFixed(2))
+        }
+    }
+    next();
+}, id2str.transform(), response('count_by_name'));
+
+courseCountApp.get('/', rqf.parse(), (req, res, next) => {
+    if ("localoffer" in req.dims || "campi" in req.dims) {
+        if ("university" in req.dims || "universityLocalOffer" in req.dims) {
+            req.sql.from('curso_ens_superior')
+                .field('curso_ens_superior.ano_censo', 'year')
+                .field('COUNT(localoferta_ens_superior.cod_local_oferta)', 'total')
+                .where('curso_ens_superior.tipo_atributo_ingresso <> 1 OR curso_ens_superior.tipo_atributo_ingresso is NULL')
+                .where('curso_ens_superior.cod_nivel_academico = 1')
+                .group('curso_ens_superior.ano_censo')
+                .order('curso_ens_superior.ano_censo')
+                .order('localoferta_ens_superior.cod_local_oferta');
+        } else {
+            req.sql.from('curso_ens_superior')
+                .field('curso_ens_superior.ano_censo', 'year')
+                .field('COUNT(localoferta_ens_superior.cod_local_oferta)', 'total')
+                .field('localoferta_ens_superior.cod_ies', 'university_id')
+                .field('curso_ens_superior.nome_ies', 'university_name')
+                .where('curso_ens_superior.tipo_atributo_ingresso <> 1 OR curso_ens_superior.tipo_atributo_ingresso is NULL')
+                .where('curso_ens_superior.cod_nivel_academico = 1')
+                .group('curso_ens_superior.ano_censo')
+                .group('localoferta_ens_superior.cod_ies')
+                .group('curso_ens_superior.nome_ies')
+                .order('curso_ens_superior.ano_censo')
+                .order('localoferta_ens_superior.cod_local_oferta');
+        }
+    } else if (("state" in req.dims) || ("city" in req.dims) || ("microregion" in req.dims) || ("mesoregion" in req.dims) || ("region" in req.dims) || 
+        ("state" in req.filter) || ("city" in req.filter) || ("microregion" in req.filter) || ("mesoregion" in req.filter) || ("region" in req.filter) ) {
+        if ("course" in req.dims){
+            req.sql.from('curso_ens_superior')
+                .field('COUNT(*)', 'total')
+                .field("'Brasil'", 'name')
+                .field('curso_ens_superior.ano_censo', 'year')
+                .group('curso_ens_superior.ano_censo')
+                .order('curso_ens_superior.ano_censo')
+                .where('curso_ens_superior.cod_nivel_academico = 1')
+        }
+        else{
+            req.sql.from('curso_ens_superior')
+                .field('COUNT(DISTINCT curso_ens_superior.cod_curso)', 'total')
+                .field("'Brasil'", 'name')
+                .field('curso_ens_superior.ano_censo', 'year')
+                .group('curso_ens_superior.ano_censo')
+                .order('curso_ens_superior.ano_censo')
+                .where('curso_ens_superior.tipo_atributo_ingresso <> 1 OR curso_ens_superior.tipo_atributo_ingresso is NULL')
+                .where('curso_ens_superior.cod_nivel_academico = 1');
+        }
+        
+    } else if ("university" in req.dims || "universityLocalOffer" in req.dims) {
+        req.sql.from('curso_ens_superior')
+            .field('COUNT(curso_ens_superior.cod_curso)', 'total')
+            .field("'Brasil'", 'name')
+            .field('curso_ens_superior.ano_censo', 'year')
+            .group('curso_ens_superior.cod_ies')
+            .group('curso_ens_superior.ano_censo')
+            .order('curso_ens_superior.cod_ies')
+            .order('curso_ens_superior.ano_censo')
+            .where('curso_ens_superior.tipo_atributo_ingresso <> 1 OR curso_ens_superior.tipo_atributo_ingresso is NULL')
+            .where('curso_ens_superior.cod_nivel_academico = 1');
+    } else {
+        req.sql.from('curso_ens_superior')
+            .field('COUNT(curso_ens_superior.cod_curso)', 'total')
+            .field("'Brasil'", 'name')
+            .field('curso_ens_superior.ano_censo', 'year')
+            .group('curso_ens_superior.ano_censo')
+            .order('curso_ens_superior.ano_censo')
+            .where('curso_ens_superior.tipo_atributo_ingresso <> 1 OR curso_ens_superior.tipo_atributo_ingresso is NULL')
+            .where('curso_ens_superior.cod_nivel_academico = 1');
+    }
+    next();
+}, rqf.build(), query, (req, res, next) =>{
+    if ('course' in req.dims){
+        var total_course = req.result.reduce((total, cur) => {return total += cur.total}, 0)
+        for (var course of req.result){
+            course.percentage = Number((( course.total / total_course ) * 100).toFixed(2))
+        }
+    }
+    next();
+}, id2str.transform(), addMissing(rqf), response('course_count'));
+
+module.exports = courseCountApp;
diff --git a/src/libs/routes_v2/courseStudents.js b/src/libs/routes_v2/courseStudents.js
new file mode 100644
index 00000000..77180cb2
--- /dev/null
+++ b/src/libs/routes_v2/courseStudents.js
@@ -0,0 +1,201 @@
+const express = require('express');
+
+const courseStudentsApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+let rqf = new ReqQueryFields();
+
+courseStudentsApp.get('/enrolled_vacancies_freshmen', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 3; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.enrolledVacanciesFreshmen(i)
+        });
+    };
+    next();
+}, response('enrolled_vacancies_freshmen'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+})
+.addValueToField({
+    name: 'state',
+    table: 'ies_ens_superior',
+    tableField: 'cod_uf_ies',
+    resultField: 'state_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_uf_ies',
+	    table: 'ies_ens_superior'
+    }
+}, 'filter')
+.addValueToField({
+    name: 'min_year',
+    table: 'curso_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo',
+        table: 'curso_ens_superior'
+    }
+}, 'filter')
+.addValueToField({
+    name: 'max_year',
+    table: 'curso_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo',
+        table: 'ies_ens_superior'
+    }
+}, 'filter')
+.addValue({
+    name: 'upper_adm_dependency',
+    table: 'curso_ens_superior',
+    tableField: 'par_categoria_administrativa',
+    resultField: 'upper_adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_categoria_administrativa', //cod_categoria_administrativa
+        table: 'curso_ens_superior'
+    }
+})
+.addValue({
+    name: 'course',
+    table: 'curso_ens_superior',
+    tableField: 'nome_curso',
+    resultField: 'course_name',
+    where: {
+        relation: '=',
+        type: 'string',
+        field: 'nome_curso',
+        table: 'curso_ens_superior'
+    }
+})
+.addValue({
+    name: 'upper_education_mod',
+    table: 'curso_ens_superior',
+    tableField: 'cod_modalidade_ensino',
+    resultField: 'upper_education_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_modalidade_ensino',
+        table: 'curso_ens_superior'
+    }
+})
+.addValue({
+    name: 'academic_organization',
+    table: 'curso_ens_superior',
+    tableField: 'cod_organizacao_academica',
+    resultField: 'academic_organization_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_organizacao_academica',
+        table: 'curso_ens_superior'
+    }
+})
+.addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+})
+.addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+})
+.addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['id', 'nome'],
+    resultField: ['city_id', 'city_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+})
+
+
+
+courseStudentsApp.get('/', rqf.parse(), (req, res, next) => {
+    req.sql.field("curso_ens_superior.ano_censo", "year")
+    .field("SUM(curso_ens_superior.quantidade_inscritos_total)", "inscritos_total")
+    .field("SUM(curso_ens_superior.quantidade_vagas_totais)", "vagas_totais")
+    .field("SUM(curso_ens_superior.quantidade_ingresso_curso)", "ingresso_curso")
+    .from("curso_ens_superior")
+    .join("ies_ens_superior ON curso_ens_superior.ano_censo = ies_ens_superior.ano_censo AND curso_ens_superior.cod_ies = ies_ens_superior.cod_ies")
+    .where("curso_ens_superior.cod_nivel_academico = 1")
+    .where("curso_ens_superior.cod_grau_academico = 2 OR curso_ens_superior.cod_grau_academico = 4")
+    .group("curso_ens_superior.ano_censo")
+    .order("curso_ens_superior.ano_censo")
+    next();
+
+}, rqf.build(), query, (req, res, next) => {
+    for (var res of req.result){
+        res.inscritos_total = Number(res.inscritos_total);
+        res.vagas_totais = Number(res.vagas_totais);
+        res.ingresso_curso = Number(res.ingresso_curso);
+        res.total = null;
+    }
+
+    next();
+}, id2str.transform(), response('course_students'))
+
+module.exports = courseStudentsApp;
diff --git a/src/libs/routes_v2/cub.js b/src/libs/routes_v2/cub.js
new file mode 100644
index 00000000..cfc2848a
--- /dev/null
+++ b/src/libs/routes_v2/cub.js
@@ -0,0 +1,227 @@
+const express = require('express');
+
+const cubApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const passport = require('passport');
+
+const download = require(`${libs}/middlewares/downloadDatabase`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+let rqfCount = new ReqQueryFields();
+
+cubApp.get('/year_range', (req, res, next) => {
+    req.sql.from('cub')
+    .field('MIN(cub.ano_censo)', 'start_year')
+    .field('MAX(cub.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+cubApp.get('/years', (req, res, next) => {
+    req.sql.from('cub')
+    .field('DISTINCT cub.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+cubApp.get('/months', (req, res, next) => {
+    req.sql.from('cub')
+    .field('DISTINCT cub.mes_censo', 'month');
+    next();
+}, query, response('months'));
+
+cubApp.get('/years_months', (req, res, next) => {
+    req.sql.from('cub')
+    .field('DISTINCT cub.ano_censo AS "year", cub.mes_censo AS "month"');
+    next();
+}, query, response('years_months'));
+
+cubApp.get('/price_type', (req, res, next) => {
+    req.sql.from('cub')
+    .field('DISTINCT cub.tipo_preco', 'price_type');
+    next();
+}, query, response('price_type'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['sigla', 'id'],
+    resultField: ['sigla_uf', 'cod_uf'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'cub'
+    },
+    // join: {
+    //     primary: 'id',
+    //     foreign: 'estado_id',
+    //     foreignTable: 'cub'
+    // }
+}).addValue({
+    name: 'min_year',
+    table: 'cub',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'cub',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'cub',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'cub',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'min_month',
+    table: 'cub',
+    tableField: 'mes_censo',
+    resultField: 'month',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'cub',
+        field: 'mes_censo'
+    }
+}).addValue({
+    name: 'max_month',
+    table: 'cub',
+    tableField: 'mes_censo',
+    resultField: 'month',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'cub',
+        field: 'mes_censo'
+    }
+});
+
+cubApp.get('/last_state_values', rqf.parse(), rqf.build(), (req, res, next) => {
+
+    var price_by_id = squel.select().from('cub')
+    .field('estado_id')
+    .field('MAX(ano_censo*100 + mes_censo)', 'ano_censo')
+    .group('estado_id')
+
+    if (req.filter.size || req.dims.size){
+        if ('state' in req.filter || 'state' in req.dims){
+            //req.sql = states
+            req.sql.from('cub')
+            .field('cub.ano_censo', 'ano')
+            .field('cub.mes_censo', 'mes')
+            .field('cub.tipo_preco', 'tipo_preco')
+            .field('cub.preco', 'preco')
+            .join(
+                price_by_id, 
+                'sub', 
+                'cub.estado_id = sub.estado_id AND cub.ano_censo = (sub.ano_censo/100)'
+            )
+            .join('estado', null, 'cub.estado_id = estado.id')
+            .group('cub.ano_censo')
+            .group('cub.mes_censo')
+            .group('cub.tipo_preco')
+            .group('cub.preco')
+        }
+        else{
+            req.sql.from("cub")
+        }
+    }
+    else{
+        //req.sql = average
+        req.sql.from(
+            squel.select().from('cub')
+            .field('cub.tipo_preco', 'tipo_preco')
+            .field('cub.preco', 'preco')
+            .join(price_by_id, 'sub', 
+                'cub.estado_id = sub.estado_id AND cub.ano_censo = (sub.ano_censo/100)'
+            )
+            .join('estado', null, 'cub.estado_id = estado.id')
+            , "states")
+        .field('AVG(states.preco)', 'preco')
+        .field("'BR'", 'sigla_uf')
+        .field('states.tipo_preco')
+        .group('states.tipo_preco')
+    }
+    next();
+}, query, id2str.transform(), response('last_state_values'))
+
+cubApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    if (req.filter.size || req.filter.dims) {
+        if ('state' in req.filter || 'state' in req.dims) {
+            req.sql.from('cub')
+            .field('cub.estado_id', 'cod_uf')
+            .field('estado.sigla', 'sigla_uf')
+            .field('cub.tipo_preco', 'tipo_preco')
+            .field('cub.preco', 'preco')
+            .join('estado', null, 'cub.estado_id=estado.id')
+            .group('cub.ano_censo')
+            .group('cub.mes_censo')
+            .group('cub.estado_id')
+            .group('estado.sigla')
+            .group('cub.tipo_preco')
+            .group('cub.preco')
+        } else {
+            req.sql.from('cub')
+            .field("'BR'", 'sigla_uf')
+            .field("cub.tipo_preco", 'tipo_preco')
+            .field('AVG(cub.preco)', 'preco')
+            .join('estado', null, 'cub.estado_id=estado.id')
+            .group('cub.ano_censo')
+            .group('cub.mes_censo')
+            .group('cub.tipo_preco')
+        }
+    } else {
+        req.sql.from('cub')
+        .field('cub.estado_id', 'cod_uf')
+        .field('estado.sigla', 'sigla_uf')
+        .field('cub.tipo_preco', 'tipo_preco')
+        .field('cub.preco', 'preco')
+        .join('estado', null, 'cub.estado_id=estado.id')
+        .group('cub.ano_censo')
+        .group('cub.mes_censo')
+        .group('cub.estado_id')
+        .group('estado.sigla')
+        .group('cub.tipo_preco')
+        .group('cub.preco')
+    }
+
+    next();
+}, query, id2str.transform(), response('cub'));
+
+module.exports = cubApp;
diff --git a/src/libs/routes_v2/dailyChargeAmount.js b/src/libs/routes_v2/dailyChargeAmount.js
new file mode 100644
index 00000000..4af6cb7d
--- /dev/null
+++ b/src/libs/routes_v2/dailyChargeAmount.js
@@ -0,0 +1,467 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const dailyChargeAmountApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: { include: [200] } }).middleware;
+
+let rqf = new ReqQueryFields();
+
+let rqfCount = new ReqQueryFields();
+
+//dailyChargeAmountApp.use(cache('15 day'));
+
+dailyChargeAmountApp.get('/year_range', (req, res, next) => {
+    req.sql.from('turma')
+        .field('MIN(turma.ano_censo)', 'start_year')
+        .field('MAX(turma.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+dailyChargeAmountApp.get('/years', (req, res, next) => {
+    req.sql.from('turma')
+        .field('DISTINCT turma.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+dailyChargeAmountApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+        .field('fonte', 'source')
+        .where('tabela = \'turma\'');
+    next();
+}, query, response('source'));
+
+dailyChargeAmountApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+dailyChargeAmountApp.get('/adm_dependency_detailed', cache('15 day'), (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+dailyChargeAmountApp.get('/location', cache('15 day'), (req, res, next) => {
+    req.result = [
+        { id: 1, name: 'Urbana' },
+        { id: 2, name: 'Rural' }
+    ];
+    next();
+}, response('location'));
+
+dailyChargeAmountApp.get('/rural_location', (req, res, next) => {
+    req.result = [
+        { id: 1, name: "Urbana" },
+        { id: 2, name: "Rural" },
+        { id: 3, name: "Rural - Área de assentamento" },
+        { id: 4, name: "Rural - Terra indígena" },
+        { id: 5, name: "Rural - Área remanescente de quilombos" },
+        { id: 6, name: "Rural - Unidade de uso sustentável" }
+    ];
+    next();
+}, response('rural_location'));
+
+dailyChargeAmountApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        { id: null, name: 'Não classificada' },
+        { id: 1, name: 'Creche' },
+        { id: 2, name: 'Pré-Escola' },
+        { id: 3, name: 'Ensino Fundamental - anos iniciais' },
+        { id: 4, name: 'Ensino Fundamental - anos finais' },
+        { id: 5, name: 'Ensino Médio' },
+        { id: 6, name: 'EJA' },
+        { id: 7, name: 'EE exclusiva' }
+    ];
+    next();
+}, response('education_level_short'));
+
+dailyChargeAmountApp.get('/average/education_level_mod', (req, res, next) => {
+    req.result = [
+        { id: null, name: 'Não classificada' },
+        { id: 1, name: 'Creche' },
+        { id: 2, name: 'Pré-Escola' },
+        { id: 3, name: 'Educação Infantil Unificada' },
+        { id: 4, name: 'Ensino Fundamental - anos iniciais' },
+        { id: 5, name: 'Ensino Fundamental - anos finais' },
+        { id: 6, name: 'Ensino Médio' },
+        { id: 7, name: 'Turma Multisseriadas e Multietapas' },
+        { id: 8, name: 'EJA - Ensino Fundamental' },
+        { id: 9, name: 'EJA - Ensino Médio' },
+        { id: 10, name: 'Educação Profissional' }
+    ];
+    next();
+}, response('education_level_mod'));
+
+dailyChargeAmountApp.get('/period', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 3; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.period(i)
+        });
+    };
+    req.result.push({
+        id: 99,
+        name: id2str.period(99)
+    });
+    next();
+}, response('period'));
+
+dailyChargeAmountApp.get('/integral_time', (req, res, next) => {
+    req.result = [
+        { id: null, name: 'Não Disponível' },
+        { id: 0, name: 'Não' },
+        { id: 1, name: 'Sim' }
+    ];
+    next();
+}, response('integral_time'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'turma'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'turma'
+    }
+}).addValue({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_id',
+        foreignTable: 'turma'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'turma'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'turma'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'turma',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'turma',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'turma',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'turma',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'location',
+    table: 'turma',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'adm_dependency',
+    table: 'turma',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'turma',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'turma',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'education_level_mod',
+    table: 'turma',
+    tableField: 'etapas_mod_ensino_segmento_id',
+    resultField: 'education_level_mod_id',
+    dontGroup: true,
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapas_mod_ensino_segmento_id'
+    }
+}).addValue({
+    name: 'period',
+    table: 'turma',
+    tableField: 'turma_turno_id',
+    resultField: 'period_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'turma_turno_id'
+    }
+}).addValue({
+    name: 'integral_time',
+    table: 'turma',
+    tableField: 'tempo_integral',
+    resultField: 'integral_time_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'tempo_integral'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'turma',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+});
+
+dailyChargeAmountApp.get('/', rqf.parse(), (req, res, next) => {
+    var status = 0;
+    if ('period' in req.filter) {
+        if (req.filter['period'].length == 1 && (req.filter['period'][0] == '3' || req.filter['period'][0] === '4')) {
+            status = 1;
+        } else if (req.filter['period'].length <= 2 && (req.filter['period'].includes('1') || req.filter['period'].includes('2')) && (!req.filter['period'].includes('3'))) {
+            status = 1;
+        }
+    } else if (req.filter['integral_time'] == '1') {
+        status = 1;
+    }
+
+    if (status) {
+        req.sql.from('turma')
+            .field('turma.ano_censo', 'year')
+            .field('turma.etapa_resumida', 'education_level_short_id')
+            .field('AVG(turma.duracao_turma)/60.0', 'average_class_duration')
+            .field('MEDIAN(turma.duracao_turma)/60.0', 'median_class_duration')
+            .field('STDDEV_SAMP(turma.duracao_turma)/60.0', 'std_class_duration')
+            .field('QUANTILE(turma.duracao_turma, 0.25)/60.0', 'fstqt_class_duration')
+            .field('QUANTILE(turma.duracao_turma, 0.75)/60.0', 'thdqt_class_duration')
+            .group('turma.ano_censo')
+            .group('turma.etapa_resumida')
+            .order('turma.ano_censo')
+            .order('turma.etapa_resumida')
+            .where('((turma.tipo_turma_id <= 3 AND turma.tipo_atendimento_id is NULL) OR (turma.tipo_atendimento_id <= 2 AND turma.tipo_turma_id is NULL)) and turma.dependencia_adm_id <= 3')
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+
+    next();
+
+}, rqf.build(), query, addMissing(rqf), (req, res, next) => {
+
+    function sliced(object) {
+        return object['education_level_short_id'] > 3;
+    }
+
+    if ('period' in req.filter || 'period' in req.dims) {
+        req.filter['period'].forEach((element) => {
+            if (element == '3')
+                req.result = req.result.filter(sliced);
+        });
+    }
+
+    next();
+}, id2str.transform(), response('turma'));
+
+dailyChargeAmountApp.get('/average', rqf.parse(), rqf.build(), (req, res, next) => {
+    var status = 0;
+    if (('education_level_mod' in req.filter || 'education_level_mod' in req.dims)
+        && ('integral_time' in req.filter)) {
+        if (req.filter['integral_time'] == '0'
+            && ('period' in req.filter)) {
+            if (req.filter['period'].length == 1
+                && req.filter['period'][0] == '3') {
+                status = 1;
+            } else if (req.filter['period'].length <= 2
+                && (req.filter['period'].includes('1')
+                    || req.filter['period'].includes('2'))
+                && (!req.filter['period'].includes('3'))) {
+                status = 1;
+            }
+        } else if (req.filter['integral_time'] == '1') {
+            status = 1;
+        }
+    }
+
+    if (status) {
+        let baseQ = req.sql.clone();
+
+        let tableR = baseQ.clone();
+        tableR.from('turma')
+            .field('duracao_turma')
+            .field('etapas_mod_ensino_segmento_id')
+            .field('ROW_NUMBER() OVER(PARTITION BY etapas_mod_ensino_segmento_id ORDER BY duracao_turma)', 'rowno')
+            .where('((turma.tipo_turma_id <= 3 AND turma.tipo_atendimento_id is NULL) OR (turma.tipo_atendimento_id <= 2 AND turma.tipo_turma_id is NULL))')
+
+        let tableG = baseQ.clone();
+        tableG.from('turma')
+            .field('1+COUNT(*)', 'counter')
+            .field('etapas_mod_ensino_segmento_id')
+            .where('((turma.tipo_turma_id <= 3 AND turma.tipo_atendimento_id is NULL) OR (turma.tipo_atendimento_id <= 2 AND turma.tipo_turma_id is NULL))')
+            .group('etapas_mod_ensino_segmento_id')
+
+        let joinRG = squel.select();
+        joinRG.from(tableR, 'R')
+            .field('R.etapas_mod_ensino_segmento_id')
+            .field('AVG(1.0*R.duracao_turma)/60', 'median_value')
+            .join(tableG, 'G', 'R.etapas_mod_ensino_segmento_id = G.etapas_mod_ensino_segmento_id AND R.rowNo BETWEEN G.counter/2 AND G.counter/2+G.counter%2')
+            .group('R.etapas_mod_ensino_segmento_id')
+
+        req.sql
+            .from('turma')
+            .from(joinRG, 'm')
+            .field('turma.ano_censo', 'year')
+            .field('turma.etapas_mod_ensino_segmento_id', 'education_level_mod_id')
+            .field('AVG(turma.duracao_turma)/60.0', 'average_class_duration')
+            .field('AVG(m.median_value)', 'median_class_duration')
+            .field('STDDEV_SAMP(turma.duracao_turma)/60.0', 'std_class_duration')
+            .field('QUANTILE(turma.duracao_turma, 0.25)/60.0', 'fstqt_class_duration')
+            .field('QUANTILE(turma.duracao_turma, 0.75)/60.0', 'thdqt_class_duration')
+            .group('turma.ano_censo')
+            .group('turma.etapas_mod_ensino_segmento_id')
+            .order('turma.ano_censo')
+            .order('turma.etapas_mod_ensino_segmento_id')
+            .where('((turma.tipo_turma_id <= 3 AND turma.tipo_atendimento_id is NULL) OR (turma.tipo_atendimento_id <= 2 AND turma.tipo_turma_id is NULL)) \
+                and m.etapas_mod_ensino_segmento_id = turma.etapas_mod_ensino_segmento_id')
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+
+    next();
+}, query, addMissing(rqf), id2str.transform(), response('turma'));
+
+module.exports = dailyChargeAmountApp;
diff --git a/src/libs/routes_v2/disciplines.js b/src/libs/routes_v2/disciplines.js
new file mode 100644
index 00000000..63b05f32
--- /dev/null
+++ b/src/libs/routes_v2/disciplines.js
@@ -0,0 +1,682 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const disciplinesApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: { include: [200] } }).middleware;
+
+let rqf = new ReqQueryFields();
+
+disciplinesApp.use(cache('15 day'));
+
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+disciplinesApp.get('/year_range', (req, res, next) => {
+    req.sql.from('docente')
+        .field('MIN(docente.ano_censo)', 'start_year')
+        .field('MAX(docente.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+disciplinesApp.get('/years', (req, res, next) => {
+    req.sql.from('docente').
+        field('DISTINCT docente.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+disciplinesApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+        .field('fonte', 'source')
+        .where('tabela = \'docente\'');
+    next();
+}, query, response('source'));
+
+disciplinesApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 8; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+disciplinesApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+disciplinesApp.get('/education_level_mod', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 12; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.educationLevelMod(i)
+        });
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.educationLevelMod(99)
+    });
+    next();
+}, response('education_level_mod'));
+
+disciplinesApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        { id: null, name: 'Não classificada' },
+        { id: 1, name: 'Creche' },
+        { id: 2, name: 'Pré-Escola' },
+        { id: 3, name: 'Ensino Fundamental - anos iniciais' },
+        { id: 4, name: 'Ensino Fundamental - anos finais' },
+        { id: 5, name: 'Ensino Médio' },
+        { id: 6, name: 'EJA' },
+        { id: 7, name: 'EE exclusiva' }
+    ];
+    next();
+}, response('education_level_short'));
+
+disciplinesApp.get('/location', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.location(i)
+        });
+    };
+    next();
+}, response('location'));
+
+disciplinesApp.get('/rural_location', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 8; i++) {
+        req.result.push({
+            id: i,
+            name: id2str.ruralLocation(i)
+        });
+    };
+    next();
+}, response('rural_location'));
+
+disciplinesApp.get('/education_type', (req, res, next) => {
+    req.sql.from('docente')
+        .field('DISTINCT nivel_tipo_formacao', 'id')
+        .order('id');
+    next();
+}, query, (req, res, next) => {
+    req.result.forEach((result) => {
+        result.name = id2str.educationType(result.id);
+    });
+    next();
+}, response('education_type'));
+
+disciplinesApp.get('/gender', (req, res, next) => {
+    req.result = [
+        { id: 1, name: 'Masculino' },
+        { id: 2, name: 'Feminino' }
+    ];
+    next();
+}, response('gender'));
+
+
+disciplinesApp.get('/contract_type', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.contractType("null")
+    }];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.contractType(i)
+        });
+    }
+    next();
+}, response('contract_type'));
+
+disciplinesApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [];
+    for (let i = 0; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ethnicGroup(i)
+        });
+    }
+    next();
+}, response('ethnic_group'));
+
+disciplinesApp.get('/discipline', (req, res, next) => {
+    req.result = [];
+    for (let i = 1; i <= 19; i++) {
+        req.result.push({
+            id: i,
+            name: id2str.discipline(i)
+        })
+    }
+    next();
+}, response('discipline'))
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'adm_dependency',
+    table: 'docente',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'docente',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'contract_type',
+    table: 'docente',
+    tableField: 'tipo_contratacao',
+    resultField: 'contract_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'tipo_contratacao'
+    }
+}).addValue({
+    name: 'education_level_mod',
+    table: 'docente',
+    tableField: 'etapas_mod_ensino_segmento_id',
+    resultField: 'education_level_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapas_mod_ensino_segmento_id'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'docente',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+}).addValue({
+    name: 'education_type',
+    table: 'docente',
+    tableField: 'nivel_tipo_formacao',
+    resultField: 'education_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'nivel_tipo_formacao'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_regiao_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_estado_id',
+        foreignTable: 'docente'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}, 'filter').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'docente'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'docente'
+    }
+}, 'filter').addValue({
+    name: 'location',
+    table: 'docente',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'docente',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'gender',
+    table: 'docente',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: 'docente',
+    tableField: 'cor_raca',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca'
+    }
+}).addValue({
+    name: 'discipline',
+    table: 'docente',
+    tableField: '',
+    resultField: '',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: ''
+    }
+});
+
+disciplinesApp.get('/', rqf.parse(), (req, res, next) => {
+    if ('discipline' in req.dims) {
+        // delete req.filter.discipline;
+        delete req.dims.discipline;
+        req.tmp_discipline = true;
+
+        req.sql.field('SUM(n_disc)', 'total')
+            .field('SUM(n_disc_adequada)', 'total_suitable')
+
+            .field('SUM(quimica_not_null)', 'total_quimica')
+            .field('SUM(adequacao_quimica)', 'total_suitable_quimica')
+
+            .field('SUM(fisica_not_null)', 'total_fisica')
+            .field('SUM(adequacao_fisica)', 'total_suitable_fisica')
+
+            .field('SUM(matematica_not_null)', 'total_matematica')
+            .field('SUM(adequacao_matematica)', 'total_suitable_matematica')
+
+            .field('SUM(biologia_not_null)', 'total_biologia')
+            .field('SUM(adequacao_biologia)', 'total_suitable_biologia')
+
+            .field('SUM(ciencias_not_null)', 'total_ciencias')
+            .field('SUM(adequacao_ciencias)', 'total_suitable_ciencias')
+
+            .field('SUM(lingua_portuguesa_not_null)', 'total_lingua_portuguesa')
+            .field('SUM(adequacao_lingua_portuguesa)', 'total_suitable_lingua_portuguesa')
+
+            .field('SUM(lingua_inglesa_not_null)', 'total_lingua_inglesa')
+            .field('SUM(adequacao_lingua_inglesa)', 'total_suitable_lingua_inglesa')
+
+            .field('SUM(lingua_espanhola_not_null)', 'total_lingua_espanhola')
+            .field('SUM(adequacao_lingua_espanhola)', 'total_suitable_lingua_espanhola')
+
+            .field('SUM(lingua_francesa_not_null)', 'total_lingua_francesa')
+            .field('SUM(adequacao_lingua_francesa)', 'total_suitable_lingua_francesa')
+
+            .field('SUM(lingua_outra_not_null)', 'total_lingua_outra')
+            .field('SUM(adequacao_lingua_outra)', 'total_suitable_lingua_outra')
+
+            .field('SUM(lingua_indigena_not_null)', 'total_lingua_indigena')
+            .field('SUM(adequacao_lingua_indigena)', 'total_suitable_lingua_indigena')
+
+            .field('SUM(artes_not_null)', 'total_artes')
+            .field('SUM(adequacao_artes)', 'total_suitable_artes')
+
+            .field('SUM(educacao_fisica_not_null)', 'total_educacao_fisica')
+            .field('SUM(adequacao_educacao_fisica)', 'total_suitable_educacao_fisica')
+
+            .field('SUM(historia_not_null)', 'total_historia')
+            .field('SUM(adequacao_historia)', 'total_suitable_historia')
+
+            .field('SUM(geografia_not_null)', 'total_geografia')
+            .field('SUM(adequacao_geografia)', 'total_suitable_geografia')
+
+            .field('SUM(filosofia_not_null)', 'total_filosofia')
+            .field('SUM(adequacao_filosofia)', 'total_suitable_filosofia')
+
+            .field('SUM(ensino_religioso_not_null)', 'total_ensino_religioso')
+            .field('SUM(adequacao_ensino_religioso)', 'total_suitable_ensino_religioso')
+
+            .field('SUM(estudos_sociais_not_null)', 'total_estudos_sociais')
+            .field('SUM(adequacao_estudos_sociais)', 'total_suitable_estudos_sociais')
+
+            .field('SUM(sociologia_not_null)', 'total_sociologia')
+            .field('SUM(adequacao_sociologia)', 'total_suitable_sociologia')
+
+            .field("'Brasil'", 'name')
+            .field('docente.ano_censo', 'year')
+            .from('docente')
+            .group('docente.ano_censo')
+            .order('docente.ano_censo')
+            .where('(docente.tipo_docente = 1 OR docente.tipo_docente = 5) AND \
+            ((docente.tipo_turma_id >= 0 AND docente.tipo_turma_id <= 3 AND docente.tipo_turma_atendimento_id is NULL) \
+            OR ((docente.tipo_turma_atendimento_id = 1 OR docente.tipo_turma_atendimento_id = 2) AND docente.tipo_turma_id is NULL)) AND \
+            docente.etapas_mod_ensino_segmento_id <> 6 AND docente.etapas_mod_ensino_segmento_id <> 12');
+    }
+    else if ('discipline' in req.filter) {
+        const disciplines = ['quimica', 'fisica', 'matematica', 'biologia', 'ciencias', 'lingua_portuguesa', 'lingua_inglesa', 'lingua_espanhola', 'lingua_francesa', 'lingua_outra', 'lingua_indigena', 'artes', 'educacao_fisica', 'historia', 'geografia', 'filosofia', 'ensino_religioso', 'estudos_sociais', 'sociologia']
+
+        let totalQuery = 'SUM('
+        let totalSuitableQuery = 'SUM('
+
+        if (!(req.filter.discipline instanceof Array)) req.filter.discipline = [req.filter.discipline]
+
+        req.filter.discipline.forEach(d => {
+            totalQuery += disciplines[d - 1] + '_not_null + '
+            totalSuitableQuery += 'adequacao_' + disciplines[d - 1] + ' + '
+        })
+        totalQuery = totalQuery.slice(0, -2) + ')'
+        totalSuitableQuery = totalSuitableQuery.slice(0, -2) + ')'
+
+        delete req.filter.discipline;
+
+        req.sql.field(totalQuery, 'total')
+            .field(totalSuitableQuery, 'total_suitable')
+            .field("'Brasil'", 'name')
+            .field('docente.ano_censo', 'year')
+            .from('docente')
+            //    .join('turma', null, 'docente.turma_id=turma.id AND docente.ano_censo=turma.ano_censo')
+            .group('docente.ano_censo')
+            .order('docente.ano_censo')
+            .where('(docente.tipo_docente = 1 OR docente.tipo_docente = 5) AND \
+                ((docente.tipo_turma_id >= 0 AND docente.tipo_turma_id <= 3 AND docente.tipo_turma_atendimento_id is NULL) \
+                OR ((docente.tipo_turma_atendimento_id = 1 OR docente.tipo_turma_atendimento_id = 2) AND docente.tipo_turma_id is NULL)) AND \
+                docente.etapas_mod_ensino_segmento_id <> 6 AND docente.etapas_mod_ensino_segmento_id <> 12');
+    }
+    else {
+        req.sql.field('SUM(n_disc)', 'total')
+            .field('SUM(n_disc_adequada)', 'total_suitable')
+            .field("'Brasil'", 'name')
+            .field('docente.ano_censo', 'year')
+            .from('docente')
+            //    .join('turma', null, 'docente.turma_id=turma.id AND docente.ano_censo=turma.ano_censo')
+            .group('docente.ano_censo')
+            .order('docente.ano_censo')
+            .where('(docente.tipo_docente = 1 OR docente.tipo_docente = 5) AND \
+            ((docente.tipo_turma_id >= 0 AND docente.tipo_turma_id <= 3 AND docente.tipo_turma_atendimento_id is NULL) \
+            OR ((docente.tipo_turma_atendimento_id = 1 OR docente.tipo_turma_atendimento_id = 2) AND docente.tipo_turma_id is NULL)) AND \
+            docente.etapas_mod_ensino_segmento_id <> 6 AND docente.etapas_mod_ensino_segmento_id <> 12');
+    }
+
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    let filters = Object.keys(req.filter)
+	
+	// if(filters.includes("state")) {
+	// 	const disciplinesDB = ['quimica', 'fisica', 'matematica', 'biologia', 'ciencias', 'lingua_portuguesa', 'lingua_inglesa', 'lingua_espanhola',
+	// 		'lingua_francesa', 'lingua_outra', 'lingua_indigena', 'artes', 'educacao_fisica', 'historia', 'geografia', 'filosofia', 'ensino_religioso',
+	// 		'estudos_sociais', 'sociologia']
+	// 	const disciplinesAPI = ["Química", "Física", "Matemática", "Biologia", "Ciências", "Língua Portuguesa", "Língua Estrangeira – Inglês",
+	// 		"Língua Estrangeira - Espanhol","Língua Estrangeira - Francês", "Língua Estrangeira - Outras", "Língua Indígena", "Arte", "Educação Física", "História",
+	// 		"Geografia", "Filosofia", "Ensino religioso", "Estudos Sociais", "Sociologia"]
+	
+	// 	let jsonKeys = []
+	// 	let results = []
+	// 	req.result.forEach((r) => {
+	// 		jsonKeys = Object.keys(r)
+		
+	// 		let i
+	// 		let size = jsonKeys.length - 2  // Last two infos are "name" and "year"
+	// 		for(i = 0; i < size; i++) {
+	// 			let total_name = jsonKeys[i]
+	// 			let suitable_name = jsonKeys[i + 1]
+			
+	// 			// Calculates percentage
+	// 			let percentage = r[suitable_name] / r[total_name]
+	// 			percentage = percentage * 100
+	// 			percentage = percentage.toFixed(1)	// Rounds to 1 digit after comma, returns string
+	// 			percentage = percentage.replace(".", ",") + "%"
+
+	// 			// Parses name
+	// 	        total_name = total_name.replace("total_", "")
+	// 			let discipline_index = disciplinesDB.indexOf(total_name)
+	// 			let discipline_name = disciplinesAPI[discipline_index]
+
+	// 			let obj = {
+	// 				total: percentage,
+	// 				name: r["name"],
+	// 				year: r["year"],
+	// 				discipline_id: discipline_index + 1,	// Convert function starts at 1, not at 0
+	// 				discipline_name: discipline_name
+	// 			}
+	// 			results.push(obj)
+			
+	// 			i++;    // Ignore next, it's a suitable already used
+	// 		}
+	// 	})
+
+	// 	req.result = results;
+	// }
+	// else {
+		let disciplinesNotSuitable = [];
+		let disciplinesSuitable = [];
+
+		req.result.forEach((r) => {
+
+            let obj = {
+                sum_total: 0,
+                sum_suitable: 0
+            }
+
+	        Object.keys(r).forEach(k => {
+                if (k !== 'total' && k !== 'total_suitable') 
+                    obj[k] = r[k];
+			})
+
+            if (req.tmp_discipline){
+                Object.keys(r).forEach(k => {
+                    if (/^total_suitable_/.test(k)) // if k starts with total_suitable
+                        obj.sum_suitable += parseInt(r[k]);
+                    else if (/^total_(?!suitable)/.test(k))
+                        obj.sum_total += parseInt(r[k]);
+                })
+            } else {
+                delete obj.sum_total;
+                delete obj.sum_suitable;
+            }
+
+            let objNotSuitable = Object.assign({}, {
+				total: parseInt(r.total) - parseInt(r.total_suitable),
+				suitable: 0,
+                discipline_name: 'Formação não adequada',
+			}, obj)
+
+	        let objSuitable = Object.assign({}, {
+		        total: parseInt(r.total_suitable),
+			    suitable: 1,
+                discipline_name: 'Formação adequada',
+            }, obj)
+
+		    disciplinesNotSuitable.push(objNotSuitable)
+			disciplinesSuitable.push(objSuitable)
+		 })
+
+		 req.result = disciplinesNotSuitable.concat(disciplinesSuitable);
+    next();
+}, response('disciplines'));
+
+module.exports = disciplinesApp;
+
+
diff --git a/src/libs/routes_v2/distributionFactor.js b/src/libs/routes_v2/distributionFactor.js
new file mode 100644
index 00000000..8a7b0e28
--- /dev/null
+++ b/src/libs/routes_v2/distributionFactor.js
@@ -0,0 +1,210 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const distributionApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+distributionApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'fatores_matricula'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'fatores_matricula'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'fatores_matricula'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'fatores_matricula'
+    }
+}, 'filter').addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'fatores_matricula'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'fatores_matricula'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'fatores_matricula'
+    }
+})
+
+// Return all cities
+distributionApp.get('/', rqf.parse(), (req, res, next) => {
+    req.querySet = [];
+    req.queryIndex = {};
+
+    let relation = req.sql.clone();
+    relation.from('relacao_fatores_matricula').field('*');
+    req.queryIndex.relation = req.querySet.push(relation) - 1;
+
+    req.sql.from('fatores_matricula')
+    .field('fatores_matricula.municipio_id', 'municipio_id')
+    .field('fatores_matricula."mais_CRE_0"')
+	.field('fatores_matricula."mais_CRE_1"')
+	.field('fatores_matricula."mais_CRE_2"')
+	.field('fatores_matricula."mais_CRE_3"')
+	.field('fatores_matricula."mais_PRE"')
+	.field('fatores_matricula."mais_EFAI"')
+	.field('fatores_matricula."mais_EFAF"')
+	.field('fatores_matricula."mais_EM"')
+	.field('fatores_matricula."mais_EJA"')
+	.field('fatores_matricula."menos_CRE_0"')
+	.field('fatores_matricula."menos_CRE_1"')
+	.field('fatores_matricula."menos_CRE_2"')
+	.field('fatores_matricula."menos_CRE_3"')
+	.field('fatores_matricula."menos_PRE"')
+	.field('fatores_matricula."menos_EFAI"')
+	.field('fatores_matricula."menos_EFAF"')
+	.field('fatores_matricula."menos_EM"')
+	.field('fatores_matricula."menos_EJA"')
+    .group('fatores_matricula.municipio_id')
+    .group('fatores_matricula."mais_CRE_0"')
+	.group('fatores_matricula."mais_CRE_1"')
+	.group('fatores_matricula."mais_CRE_2"')
+	.group('fatores_matricula."mais_CRE_3"')
+	.group('fatores_matricula."mais_PRE"')
+	.group('fatores_matricula."mais_EFAI"')
+	.group('fatores_matricula."mais_EFAF"')
+	.group('fatores_matricula."mais_EM"')
+	.group('fatores_matricula."mais_EJA"')
+	.group('fatores_matricula."menos_CRE_0"')
+	.group('fatores_matricula."menos_CRE_1"')
+	.group('fatores_matricula."menos_CRE_2"')
+	.group('fatores_matricula."menos_CRE_3"')
+	.group('fatores_matricula."menos_PRE"')
+	.group('fatores_matricula."menos_EFAI"')
+	.group('fatores_matricula."menos_EFAF"')
+	.group('fatores_matricula."menos_EM"')
+	.group('fatores_matricula."menos_EJA"');
+
+    if(typeof req.dims.state !== 'undefined' || typeof req.filter.state !== 'undefined') {
+        req.sql.where('fatores_matricula.nivel = \'UF\'');
+    } else {
+        req.sql.where('fatores_matricula.nivel = \'BR\'');
+    }
+
+    next();
+}, rqf.build(), query, (req, res, next) => {
+    req.enrollmentFactor = req.result;
+    next();
+}, multiQuery, (req, res, next) => {
+    let relation = req.result[req.queryIndex.relation];
+    let result = [];
+    let first = true;
+    req.enrollmentFactor.forEach((city) => {
+        // if(first) console.log(city);
+        let obj = {
+            level: city.nivel,
+            region_id: city.regiao_id,
+            region_name: city.region_name,
+            state_id: city.state_id,
+            state_name: city.state_name,
+            city_id: city.municipio_id,
+            city_name: city.city_name,
+            series: []
+        };
+        // if(first) console.log(obj);
+        first = false;
+        relation.forEach((serie) => {
+            obj.series.push({
+                serie_id: serie.id,
+                distribution_factor_addition: city[serie.fator_adicao],
+                distribution_factor_reduction: city[serie.fator_reducao]
+            });
+        });
+        result.push(obj);
+    });
+    req.result = result;
+    next();
+}, response('ditributionFactor'));
+
+module.exports = distributionApp;
diff --git a/src/libs/routes_v2/downloads.js b/src/libs/routes_v2/downloads.js
new file mode 100644
index 00000000..2ec83b9e
--- /dev/null
+++ b/src/libs/routes_v2/downloads.js
@@ -0,0 +1,54 @@
+const express = require('express');
+
+const downloadApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const Download = require(`${libs}/models/download`);
+
+const User = require(`${libs}/models/user`);
+
+const passport = require('passport');
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+downloadApp.get('/', passport.authenticate('bearer', {session: false}), (req, res, next) => {
+    request.get(config.cdn.url + '/api/v1/file', (err, response, body) => {
+        let cdn = JSON.parse(body);
+        Download.find({userId: req.user._id}, (err, downloads) => {
+            if (err) {
+                log.error(err);
+                return next(err);
+            }
+            
+            if(!downloads) {
+                res.statusCode = 404;
+                return res.json({msg: 'Nenhum download encontrado'});
+            } else {
+                downloads.forEach((dl) => {
+                    for(let i = 0; i < cdn.length; ++i) {
+                        if(cdn[i].query == dl.query) {
+                            dl.status = cdn[i].expired ? 'Expirado' : 'Enviado';
+                            dl.size = cdn[i].size;
+                            dl.expired = cdn[i].expired;
+                            dl.updatedAt = cdn[i].lastAccess;
+                            dl.link = config.cdn.download + '/' + cdn[i]._id;
+
+                            dl.save((err) => {
+                                if(err) log.error(err);
+                            });
+                            return;
+                        }
+                    }
+                });
+            }
+            res.json(downloads);
+        });
+    });
+});
+
+module.exports = downloadApp;
diff --git a/src/libs/routes_v2/educationYears.js b/src/libs/routes_v2/educationYears.js
new file mode 100644
index 00000000..0d103883
--- /dev/null
+++ b/src/libs/routes_v2/educationYears.js
@@ -0,0 +1,46 @@
+const express = require('express');
+
+const educationYearsApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const response = require(`${libs}/middlewares/response`);
+
+const config = require(`${libs}/config`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+educationYearsApp.use(cache('15 day'));
+
+educationYearsApp.get('/', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 7; ++i) {
+        let edLvlShort = {
+            id: i,
+            name: id2str.educationLevelShort(i),
+            schoolYears: []
+        };
+
+        for(let j = i*10; j <= (i*10 + 9); ++j) {
+
+            let schoolYear = {
+                id: j,
+                name: id2str.schoolYear(j)
+            };
+
+            if(schoolYear.name !== id2str.schoolYear(99)) {
+                edLvlShort.schoolYears.push(schoolYear);
+            }
+        }
+        if(edLvlShort.name !== id2str.schoolYear(99)) {
+            req.result.push(edLvlShort);
+        }
+    }
+    next();
+}, response('educationYears'));
+
+module.exports = educationYearsApp;
diff --git a/src/libs/routes_v2/educationalBudget.js b/src/libs/routes_v2/educationalBudget.js
new file mode 100644
index 00000000..30f6afc1
--- /dev/null
+++ b/src/libs/routes_v2/educationalBudget.js
@@ -0,0 +1,325 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+
+const libs = `${process.cwd()}/libs`;
+
+const conn = require(`${libs}/db/monet`);
+
+const express = require('express');
+
+const educationalBudget = express.Router();
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+let rqfId = new ReqQueryFields();
+
+const db = require(`${libs}/db/query_exec`);
+
+const log = require(`${libs}/log`)(module);
+
+rqfId.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'federative_entity',
+    table: 'orcamento_educacional',
+    tableField: 'entidade_federativa',
+    resultField: 'entidade_federativa_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'entidade_federativa'
+    }
+});
+
+//Insert route in orcamento_educacional table
+educationalBudget.post('/insert', (req, res, next) => {
+    let id = req.body.id || null;
+    let entidade_federativa = req.body.entidade_federativa || null;
+    let csv = req.body.csv || null;
+
+    //csv example
+    //"RO|7777777|jiarana|8043|41768845|43141895|66081767|8216|24312922|158,2%"
+
+     req.id = id;
+     req.entidade_federativa = entidade_federativa;
+     req.csv = csv.toString();
+     req.date = new Date();
+
+     //split o csv para adicionar separadamente nas colunas
+    csv = csv.split("|");
+    req.uf = csv[0];
+    req.municipio_id = csv[1];
+    req.nome = csv[2];
+    req.matriculas = csv[3];
+    req.receitas_vinculadas = csv[4];
+    req.despesas_realizadas = csv[5];
+    req.despesas_correntes = csv[6];
+    req.valor_aluno = csv[7];
+    req.completacao = csv[8];
+    req.completacao_porcentagem = csv[9];
+
+    let sqlQueryParams = [];
+
+    //remove id duplicate
+    let deleteQuery = squel.delete()
+        .from("orcamento_educacional")
+        .where('id = ' + req.id)
+        .toString()
+
+    //Exec delete sql
+    log.debug(`Executing SQL query '${deleteQuery}' with params '${[sqlQueryParams]}'`);
+    return new Promise((resolve, reject) => {
+        // Prepare statement
+        conn.prepare(deleteQuery, true).then((dbQuery) => {
+            // Execute query
+            dbQuery.exec(sqlQueryParams).then((dbResult) => {
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                resolve(dbResult.data);
+                req.result = "Insertion Success"
+                next();
+            }).catch((queryError) => {
+                log.error(`SQL query execution error: ${queryError.message}`);
+                req.result = "SQL query execution error:" + queryError.message;
+                log.error(`SQL query: ${deleteQuery} with params: ${sqlQueryParams}`);
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                next();
+            });
+        }).catch((prepError) => {
+            log.error(`SQL prepared statement error: ${prepError.message}`);
+            req.result = "SQL query execution error:" + queryError.message;
+            log.error(`SQL query: ${deleteQuery} with params: ${sqlQueryParams}`);
+            next();
+        });
+    });
+
+}, (req, res, next) => {
+    //build query
+    let insertQuery = squel.insert()
+        .into("orcamento_educacional")
+        .set("id", req.id)
+        .set("entidade_federativa", req.entidade_federativa)
+        .set("csv", req.csv)
+        .set("data", req.date.toString())
+        .set("uf", req.uf)
+        .set("municipio_id", req.municipio_id)
+        .set("nome", req.nome)
+        .set("receitas_vinculadas", req.receitas_vinculadas)
+        .set("despesas_realizadas", req.despesas_realizadas)
+        .set("despesas_correntes", req.despesas_correntes)
+        .set("valor_aluno", req.valor_aluno)
+        .set("completacao", req.completacao)
+        .set("completacao_porcentagem", req.completacao_porcentagem)
+        .toString()
+
+    let sqlQueryParams = [];
+
+    //Exec insert sql
+    log.debug(`Executing SQL query '${insertQuery}' with params '${[sqlQueryParams]}'`);
+    return new Promise((resolve, reject) => {
+        // Prepare statement
+        conn.prepare(insertQuery, true).then((dbQuery) => {
+            // Execute query
+            dbQuery.exec(sqlQueryParams).then((dbResult) => {
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                resolve(dbResult.data);
+                req.result = "Insertion Success"
+                next();
+            }).catch((queryError) => {
+                log.error(`SQL query execution error: ${queryError.message}`);
+                req.result = "SQL query execution error:" + queryError.message;
+                log.error(`SQL query: ${insertQuery} with params: ${sqlQueryParams}`);
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                next();
+            });
+        }).catch((prepError) => {
+            log.error(`SQL prepared statement error: ${prepError.message}`);
+            req.result = "SQL query execution error:" + queryError.message;
+            log.error(`SQL query: ${insertQuery} with params: ${sqlQueryParams}`);
+            next();
+        });
+    });
+
+    next();
+}, response('educationalBudget'));
+
+//Delete orcamento_educacional table
+educationalBudget.get('/delete', (req, res, next) => {
+
+    //build query
+    let insertQuery = squel.delete()
+        .from("orcamento_educacional")
+        .toString()
+
+    let sqlQueryParams = [];
+
+    //Exec sql in monet
+    log.debug(`Executing SQL query '${insertQuery}' with params '${[sqlQueryParams]}'`);
+    return new Promise((resolve, reject) => {
+        // Prepare statement
+        conn.prepare(insertQuery, true).then((dbQuery) => {
+            // Execute query
+            dbQuery.exec(sqlQueryParams).then((dbResult) => {
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                resolve(dbResult.data);
+                req.result = "Delete Table Success"
+                next();
+            }).catch((queryError) => {
+                log.error(`SQL query execution error: ${queryError.message}`);
+                req.result = "SQL query execution error:" + queryError.message;
+                log.error(`SQL query: ${insertQuery} with params: ${sqlQueryParams}`);
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                next();
+            });
+        }).catch((prepError) => {
+            log.error(`SQL prepared statement error: ${prepError.message}`);
+            req.result = "SQL query execution error:" + queryError.message;
+            log.error(`SQL query: ${insertQuery} with params: ${sqlQueryParams}`);
+            next();
+        });
+    });
+
+    next();
+}, response('educationalBudget'));
+
+//Return all id's in table
+educationalBudget.get('/id', rqfId.parse(), rqfId.build(), (req, res, next) => {
+        req.sql.from('orcamento_educacional')
+        .field('orcamento_educacional.municipio_id')
+        next();
+}, query, response('educationalBudget'));
+
+//Return count id grouop by entidade_federativa
+educationalBudget.get('/finish', rqf.parse(), (req, res, next) => {
+        req.sql.field('COUNT(*)', 'total')
+        .field('orcamento_educacional.entidade_federativa', 'entidade_federativa')
+        .from('orcamento_educacional')
+        .group('orcamento_educacional.entidade_federativa')
+        .order('orcamento_educacional.entidade_federativa')
+        next();
+}, query, response('educationalBudget'));
+
+//return all data
+educationalBudget.get('/', rqf.parse(), (req, res, next) => {
+        req.sql.from('orcamento_educacional')
+        .field('orcamento_educacional.id')
+        .field('orcamento_educacional.entidade_federativa', 'entidade_federativa')
+        .field('orcamento_educacional.data', 'data_insercao')
+        .field('orcamento_educacional.uf', 'uf')
+        .field('orcamento_educacional.municipio_id', 'municipio_id')
+        .field('orcamento_educacional.nome', 'nome')
+        .field('orcamento_educacional.receitas_vinculadas', 'receitas_vinculadas')
+        .field('orcamento_educacional.despesas_realizadas', 'despesas_realizadas')
+        .field('orcamento_educacional.despesas_correntes', 'despesas_correntes')
+        .field('orcamento_educacional.valor_aluno', 'valor_aluno')
+        .field('orcamento_educacional.completacao', 'completacao')
+        .field('orcamento_educacional.completacao_porcentagem', 'completacao_porcentagem')
+        .field('orcamento_educacional.csv', 'csv')
+        next();
+}, query, response('educationalBudget'));
+
+//Insert route in orcamento_educacional table
+educationalBudget.post('/insert_pqr', (req, res, next) => {
+    let id = JSON.parse(req.body.id) || null;
+    let pqr = JSON.parse(req.body.pqr) || null;
+
+     req.id = id;
+     req.pqr = pqr;
+     req.date = new Date();
+
+    //build query
+    let insertQuery = squel.insert()
+        .into("orcamento_educacional_pqr")
+        .set("id", req.id.toString())
+        .set("pqr", req.pqr.toString())
+        .set("data", req.date.toString())
+        .toString()
+
+    let sqlQueryParams = [];
+
+    //Exec sql
+    log.debug(`Executing SQL query '${insertQuery}' with params '${[sqlQueryParams]}'`);
+    return new Promise((resolve, reject) => {
+        // Prepare statement
+        conn.prepare(insertQuery, true).then((dbQuery) => {
+            // Execute query
+            dbQuery.exec(sqlQueryParams).then((dbResult) => {
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                resolve(dbResult.data);
+                req.result = "Insertion Success"
+                next();
+            }).catch((queryError) => {
+                log.error(`SQL query execution error: ${queryError.message}`);
+                req.result = "SQL query execution error:" + queryError.message;
+                log.error(`SQL query: ${insertQuery} with params: ${sqlQueryParams}`);
+                // release resources allocated for the prepared statement
+                dbQuery.release();
+                next();
+            });
+        }).catch((prepError) => {
+            log.error(`SQL prepared statement error: ${prepError.message}`);
+            req.result = "SQL query execution error:" + queryError.message;
+            log.error(`SQL query: ${insertQuery} with params: ${sqlQueryParams}`);
+            next();
+        });
+    });
+
+    next();
+}, response('educationalBudget'));
+
+educationalBudget.get('/get_pqr', (req, res, next) => {
+    req.sql.from('orcamento_educacional_pqr')
+    .field('orcamento_educacional_pqr.id', 'id')
+    .field('orcamento_educacional_pqr.pqr', 'pqr')
+    .field('orcamento_educacional_pqr.data', 'data')
+    next();
+}, query, response('educationalBudget'));
+
+module.exports = educationalBudget;
+
diff --git a/src/libs/routes_v2/employees.js b/src/libs/routes_v2/employees.js
new file mode 100644
index 00000000..1ef4e7fd
--- /dev/null
+++ b/src/libs/routes_v2/employees.js
@@ -0,0 +1,656 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const employeesApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]}  }).middleware;
+
+let rqfTeacher = new ReqQueryFields();
+
+let rqfSchool = new ReqQueryFields();
+
+
+employeesApp.use(cache('15 day'));
+
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+employeesApp.get('/year_range', (req, res, next) => {
+    req.sql.from('escola')
+    .field('MIN(escola.ano_censo)', 'start_year')
+    .field('MAX(escola.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+employeesApp.get('/years', (req, res, next) => {
+    req.sql.from('escola').
+    field('DISTINCT escola.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+employeesApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'docente\'');
+    next();
+}, query, response('source'));
+
+employeesApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+employeesApp.get('/diff_location', (req, res, next) => {
+    req.result = [
+        {id: 0, name: "A escola não está em localidade diferenciada"},
+        {id: 1, name: "Área de assentamento"},
+        {id: 2, name: "Terra indígena"},
+        {id: 3, name: "Terra remanescente de quilombos"},
+    ];
+    next();
+}, response('diff_location'));
+
+employeesApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+employeesApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+employeesApp.get('/function', (req, res, next) => {
+    req.result = [
+        {id: 0, name: "Administrativos"},
+        {id: 1, name: "Serviços Gerais"},
+        {id: 2, name: "Bibliotecário"},
+        {id: 3, name: "Saúde"},
+        {id: 4, name: "Coordenador"},
+        {id: 5, name: "Fonoaudiólogo"},
+        {id: 6, name: "Nutricionista"},
+        {id: 7, name: "Psicólogo"},
+        {id: 8, name: "Alimentação"},
+        {id: 9, name: "Pedagogia"},
+        {id: 10, name: "Secretário"},
+        {id: 11, name: "Segurança"},
+        {id: 12, name: "Monitores"},
+        {id: 99, name: "Não Classificado"}
+    ];
+    next();
+}, response('function'));
+
+rqfSchool.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'adm_dependency',
+    table: '@',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: '@',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'filter').addValueToField({
+    name: 'school',
+    table: '@',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+}, 'dims').addValueToField({
+    name: 'school',
+    table: '@',
+    tableField: 'escola_nome',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+}, 'filter').addValue({
+    name: 'location',
+    table: '@',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: '@',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: '@',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'function',
+    table: '@',
+    tableField: 'a',
+    resultField: 'function_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'a'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+});
+
+rqfTeacher.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'adm_dependency',
+    table: '@',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: '@',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'filter').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id','ano_censo'],
+        foreign: ['escola_id','ano_censo'],
+        foreignTable: '@'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id','ano_censo'],
+        foreign: ['escola_id','ano_censo'],
+        foreignTable: '@'
+    }
+}, 'filter').addValue({
+    name: 'location',
+    table: '@',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: '@',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: '@',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+});
+
+
+function formatFunction(queryOriginal,reqDims) {
+    delete reqDims.size;
+    delete reqDims.function;
+    let dims = Object.keys(reqDims); //se for = 0, apenas lidamos com a dimensao function. Se for = 1, lidamos com function mais a dimensao q esta nesse array.
+    let name = {
+        qtde_admin: "Administrativos",
+        qtde_servicos_gerais: "Serviços Gerais",
+        qtde_bibliotecario: "Bibliotecário",
+        qtde_saude: "Saúde",
+        qtde_coordenador: "Coordenador",
+        qtde_fono: "Fonoaudiólogo",
+        qtde_nutricionista: "Nutricionista",
+        qtde_psicologo: "Psicólogo",
+        qtde_alimentacao: "Alimentação",
+        qtde_pedagogia: "Pedagogia",
+        qtde_secretario: "Secretário",
+        qtde_seguranca: "Segurança",
+        qtde_monitores: "Monitores",
+        qtde_null: "Não Classificado"
+    }
+    let resultObj = []
+    //Nesse caso apenas precisamos acertar as dimensoes que o banco retorna, ou seja, criando um objeto para cada funcao de funcionario
+    if (dims.length == 0) {
+        queryOriginal.forEach((result) => {
+            Object.keys(result).forEach(function(key,index) {
+                if (key.includes("qtde")) {
+                    let newObj = {
+                        year: result["year"],
+                        function_id: index,
+                        function_name: name[key],
+                        total: result[key]
+                    }
+                    resultObj.push(newObj);
+                }
+            })
+        })
+    }
+    //Nesse caso precisamos copiar o id e name da variavel que está na dimensão junto com funcionarios por função
+    else {
+        queryOriginal.forEach((result) => {
+            Object.keys(result).forEach(function(key,index) {
+                if (key.includes("qtde")) {
+                    let newObj = {
+                        year: result["year"],
+                        function_id: index,
+                        function_name: name[key],
+                        total: result[key]
+                    }
+                    newObj[dims[0] + "_id"] = result[dims[0] + "_id"];
+                    newObj[dims[0] + "_name"] = result[dims[0] + "_name"];
+                    resultObj.push(newObj);
+                }
+            })
+        })
+    }
+
+    return resultObj;
+
+}
+
+function matchQueries(queryTotal, queryPartial) {
+    let match = [];
+    queryTotal.forEach((result) => {
+        let newObj = {};
+        let keys = Object.keys(result);
+        keys.forEach((key) => {
+            newObj[key] = result[key];
+        });
+        let index = keys.indexOf('total');
+        if(index > -1) keys.splice(index, 1);
+        let objMatch = null;
+
+        for(let i = 0; i < queryPartial.length; ++i) {
+            let partial = queryPartial[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(partial[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                objMatch = partial;
+                break;
+            }
+        }
+
+        if(objMatch) {
+            newObj.total = result.total - objMatch.total;
+            if (newObj.total > 0) {
+                newObj.total_employees = result.total;
+                newObj.total_teachers = objMatch.total
+                match.push(newObj);
+            }
+        }
+    });
+
+    return match;
+}
+
+employeesApp.get('/', rqfSchool.parse(), (req, res, next) => {
+    req.allTeacher = {}
+    req.schoolTable = {}
+
+    if ("function" in req.dims) {
+        delete req.dims.function;
+        req.sql.field('SUM(funcionarios_por_escola.total)', 'qtde_null')
+        .field('funcionarios_por_escola.ano_censo', 'year')
+        .from('funcionarios_por_escola')
+        .group('funcionarios_por_escola.ano_censo')
+        .order('funcionarios_por_escola.ano_censo')
+				.where('funcionarios_por_escola.ano_censo <> 2009 or funcionarios_por_escola.estado_id <> 42')
+    } else {
+        delete req.dims.function;
+        req.sql.field('SUM(funcionarios_por_escola.total)', 'total')
+        .field('funcionarios_por_escola.ano_censo', 'year')
+        .from('funcionarios_por_escola')
+        .group('funcionarios_por_escola.ano_censo')
+        .order('funcionarios_por_escola.ano_censo')
+				.where('funcionarios_por_escola.ano_censo <> 2009 or funcionarios_por_escola.estado_id <> 42')
+    }
+    next();
+
+}, rqfSchool.build(), query, rqfSchool.parse(), id2str.transform(), (req, res, next) => {
+
+    req.allTeacher = req.result;
+    req.resetSql();
+    if ("function" in req.dims) {
+        req.sql.field('SUM(CASE WHEN escola.qt_prof_admin = 88888 THEN 0 ELSE escola.qt_prof_admin END)', 'qtde_admin')
+        .field('SUM(CASE WHEN escola.qtde_prof_servicos_gerais = 88888 THEN 0 ELSE escola.qtde_prof_servicos_gerais END) AS qtde_servicos_gerais')
+        .field('SUM(CASE WHEN escola.qtde_prof_bibliotecario = 88888 THEN 0 ELSE escola.qtde_prof_bibliotecario END)', 'qtde_bibliotecario')
+        .field('SUM(CASE WHEN escola.qtde_prof_saude = 88888 THEN 0 ELSE escola.qtde_prof_saude END)','qtde_saude')
+        .field('SUM(CASE WHEN escola.qtde_prof_coordenador = 88888 THEN 0 ELSE escola.qtde_prof_coordenador END)','qtde_coordenador')
+        .field('SUM(CASE WHEN escola.qtde_prof_fono = 88888 THEN 0 ELSE escola.qtde_prof_fono END)','qtde_fono')
+        .field('SUM(CASE WHEN escola.qtde_prof_nutricionista = 88888 THEN 0 ELSE escola.qtde_prof_nutricionista END)', 'qtde_nutricionista')
+        .field('SUM(CASE WHEN escola.qtde_prof_psicologo = 88888 THEN 0 ELSE escola.qtde_prof_psicologo END)', 'qtde_psicologo')
+        .field('SUM(CASE WHEN escola.qtde_prof_alimentacao = 88888 THEN 0 ELSE escola.qtde_prof_alimentacao END)','qtde_alimentacao')
+        .field('SUM(CASE WHEN escola.qtde_prof_pedagogia = 88888 THEN 0 ELSE escola.qtde_prof_pedagogia END)', 'qtde_pedagogia')
+        .field('SUM(CASE WHEN escola.qtde_prof_secretario = 88888 THEN 0 ELSE escola.qtde_prof_secretario END)','qtde_secretario')
+        .field('SUM(CASE WHEN escola.qtde_prof_seguranca = 88888 THEN 0 ELSE escola.qtde_prof_seguranca END)','qtde_seguranca')
+        .field('SUM(CASE WHEN escola.qtde_prof_monitores = 88888 THEN 0 ELSE escola.qtde_prof_monitores END)', 'qtde_monitores')
+        .field("'Brasil'", 'name')
+        .field('escola.ano_censo', 'year')
+        .from('escola')
+        .group('escola.ano_censo')
+        .order('escola.ano_censo')
+        .where('(escola.situacao_funcionamento_pareada = 1) AND (escola.ensino_regular = 1 OR escola.ensino_eja = 1 OR escola.educacao_profissional = 1) AND (escola.dependencia_adm_id = 2 OR escola.dependencia_adm_id = 3 OR escola.dependencia_adm_id = 4) and ano_censo >= 2019');
+        delete req.dims.function;
+    } else {
+        req.sql.field('SUM(CASE WHEN escola.qt_prof_admin = 88888 THEN 0 ELSE escola.qt_prof_admin END) + SUM(CASE WHEN escola.qtde_prof_servicos_gerais = 88888 THEN 0 ELSE escola.qtde_prof_servicos_gerais END) + SUM(CASE WHEN escola.qtde_prof_bibliotecario = 88888 THEN 0 ELSE escola.qtde_prof_bibliotecario END) + SUM(CASE WHEN escola.qtde_prof_saude = 88888 THEN 0 ELSE escola.qtde_prof_saude END) + SUM(CASE WHEN escola.qtde_prof_coordenador = 88888 THEN 0 ELSE escola.qtde_prof_coordenador END) + SUM(CASE WHEN escola.qtde_prof_fono = 88888 THEN 0 ELSE escola.qtde_prof_fono END) + SUM(CASE WHEN escola.qtde_prof_nutricionista = 88888 THEN 0 ELSE escola.qtde_prof_nutricionista END) + SUM(CASE WHEN escola.qtde_prof_psicologo = 88888 THEN 0 ELSE escola.qtde_prof_psicologo END) + SUM(CASE WHEN escola.qtde_prof_alimentacao = 88888 THEN 0 ELSE escola.qtde_prof_alimentacao END) + SUM(CASE WHEN escola.qtde_prof_pedagogia = 88888 THEN 0 ELSE escola.qtde_prof_pedagogia END) + SUM(CASE WHEN escola.qtde_prof_secretario = 88888 THEN 0 ELSE escola.qtde_prof_secretario END) + SUM(CASE WHEN escola.qtde_prof_seguranca = 88888 THEN 0 ELSE escola.qtde_prof_seguranca END) + SUM(CASE WHEN escola.qtde_prof_monitores = 88888 THEN 0 ELSE escola.qtde_prof_monitores END)', 'total')
+        .field("'Brasil'", 'name')
+        .field('escola.ano_censo', 'year')
+        .from('escola')
+        .group('escola.ano_censo')
+        .order('escola.ano_censo')
+        .where('(escola.situacao_funcionamento_pareada = 1) AND (escola.ensino_regular = 1 OR escola.ensino_eja = 1 OR escola.educacao_profissional = 1) AND (escola.dependencia_adm_id = 2 OR escola.dependencia_adm_id = 3 OR escola.dependencia_adm_id = 4) and ano_censo >= 2019');
+    }
+    next();
+
+}, rqfSchool.build(), query, rqfSchool.parse(), id2str.transform(), addMissing(rqfSchool), (req, res, next) => {
+
+    if ("function" in req.dims) {
+        let aux_employes = formatFunction(req.result, req.dims);
+        req.allTeacher = formatFunction(req.allTeacher, req.dims);
+        req.schoolTable = aux_employes;
+    } else {
+        req.schoolTable = req.result
+    }
+
+    if (req.filter.min_year <= 2018 && req.filter.max_year <= 2018) {
+        let aux_employees = req.allTeacher;
+        req.result = aux_employees;
+    } else if (req.filter.min_year >= 2019 && req.filter.max_year >= 2019) {
+        req.result = req.schoolTable;
+    } else if (req.filter.min_year <= 2018 && req.filter.max_year >= 2019) {
+        let aux_employees = req.allTeacher;
+        req.result = aux_employees.concat(req.schoolTable);
+    }
+    next();
+}, response('employees'));
+
+module.exports = employeesApp;
diff --git a/src/libs/routes_v2/enrollment.js b/src/libs/routes_v2/enrollment.js
new file mode 100644
index 00000000..87259385
--- /dev/null
+++ b/src/libs/routes_v2/enrollment.js
@@ -0,0 +1,1204 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+const { result } = require('lodash');
+
+const enrollmentApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+enrollmentApp.use(cache('15 day'));
+
+let rqf = new ReqQueryFields();
+
+// Complete range of the enrollments dataset.
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+enrollmentApp.get('/year_range', (req, res, next) => {
+    req.sql.from('matricula')
+    .field('MIN(matricula.ano_censo)', 'start_year')
+    .field('MAX(matricula.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+enrollmentApp.get('/years', (req, res, next) => {
+    req.sql.from('matricula')
+    .field('DISTINCT matricula.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+enrollmentApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'matricula\'');
+    next();
+}, query, response('source'));
+
+enrollmentApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+enrollmentApp.get('/diff_location', (req, res, next) => {
+    req.result = [
+        {id: 0, name: "A escola não está em localidade diferenciada"},
+        {id: 1, name: "Área de assentamento"},
+        {id: 2, name: "Terra indígena"},
+        {id: 3, name: "Terra remanescente de quilombos"},
+    ];
+    next();
+}, response('diff_location'));
+
+// Returns all school years available
+enrollmentApp.get('/school_year', (req, res, next) => {
+    req.result = [];
+    for(let i = 11; i <= 71; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.schoolYear(i)
+        };
+
+        if(obj.name !== id2str.schoolYear(99)) {
+            req.result.push(obj);
+        }
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.schoolYear(99)
+    });
+    next();
+}, response('school_year'));
+
+// Returns all school years available
+enrollmentApp.get('/education_level', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 74; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.educationLevel(i)
+        };
+
+        if(obj.name !== id2str.educationLevel(99)) {
+            req.result.push(obj);
+        }
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.educationLevel(99)
+    });
+    next();
+}, response('education_level'));
+
+// Returns all school years available
+enrollmentApp.get('/education_level_mod', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 12; ++i) {
+        if (i == 3 || i == 6)
+            continue;
+        
+        req.result.push({
+            id: i,
+            name: id2str.educationLevelMod(i)
+        });
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.educationLevelMod(99)
+    });
+    next();
+}, response('education_level_mod'));
+
+enrollmentApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Creche'},
+        {id: 2, name: 'Pré-Escola'},
+        {id: 3, name: 'Ensino Fundamental - anos iniciais'},
+        {id: 4, name: 'Ensino Fundamental - anos finais'},
+        {id: 5, name: 'Ensino Médio'},
+        {id: 6, name: 'EJA'},
+        {id: 7, name: 'EE exclusiva'}
+    ];
+    next();
+}, response('education_level_short'));
+
+// Returns all adm dependencies
+enrollmentApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+enrollmentApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 8; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+// Return genders
+enrollmentApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+// Return ethnic group
+enrollmentApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <=5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ethnicGroup(i)
+        });
+    }
+    next();
+}, response('ethnic_group'));
+
+enrollmentApp.get('/period', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.period(i)
+        });
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.period(99)
+    });
+    next();
+}, response('period'));
+
+// Returns integral-time avaible
+enrollmentApp.get('/integral_time', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.integralTime(i)
+        });
+    }
+    next();
+}, response('integral_time'));
+
+enrollmentApp.get('/special_class', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('special_class'));
+
+enrollmentApp.get('/pee', (req, res, next) => {
+    req.result = [
+        {id: true, name: id2str.booleanVariable(true)},
+        {id: false, name: id2str.booleanVariable(false)}
+    ];
+    next();
+}, response('pee'))
+
+enrollmentApp.get('/pee_por_categoria', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 14; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.peePorCategoria(i)
+        });
+    }
+    next();
+}, response('pee_por_categoria'));
+
+enrollmentApp.get('/age_range_all', (req, res, next) => {
+    req.result = [
+        {id: 1, name: '0 a 3 anos'},
+        {id: 2, name: '4 a 5 anos'},
+        {id: 3, name: '6 a 10 anos'},
+        {id: 4, name: '11 a 14 anos'},
+        {id: 5, name: '15 a 17 anos'},
+        {id: 6, name: '18 a 24 anos'},
+        {id: 7, name: '25 a 29 anos'},
+        {id: 8, name: '30 a 40 anos'},
+        {id: 9, name: '41 a 50 anos'},
+        {id: 10, name: '51 a 64 anos'},
+        {id: 11, name: 'Mais que 64 anos'}
+    ];
+    next();
+}, response('age_range_all'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'adm_dependency',
+    table: 'matricula',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'matricula',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'school_year',
+    table: 'matricula',
+    tableField: 'serie_ano_id',
+    resultField: 'school_year_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'serie_ano_id'
+    }
+}).addValue({
+    name: 'education_level',
+    table: 'matricula',
+    tableField: 'etapa_ensino_id',
+    resultField: 'education_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_ensino_id'
+    }
+}).addValue({
+    name: 'education_level_mod',
+    table: 'matricula',
+    tableField: 'etapas_mod_ensino_segmento_id',
+    resultField: 'education_level_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapas_mod_ensino_segmento_id'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'matricula',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'matricula'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'matricula'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'matricula'
+    }
+}).addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'matricula'
+    }
+}, 'dims').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: 'nome',
+    resultField: 'state_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'matricula'
+    }
+}, 'filter').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'matricula'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'matricula'
+    }
+}, 'filter').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'matricula'
+    }
+}, 'dims').addValueToField({
+    name: 'locale_id',
+    table: 'matricula',
+    tableField: 'localizacao_id',
+    resultField: 'locale_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}, 'dims').addValueToField({
+    name: 'school_id',
+    table: 'escola',
+    tableField: 'id',
+    resultField: 'school_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'matricula'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'matricula'
+    }
+}, 'filter').addValue({
+    name: 'location',
+    table: 'matricula',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: 'matricula',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'matricula',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'matricula',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'matricula',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'gender',
+    table: 'matricula',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: 'matricula',
+    tableField: 'cor_raca_id',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca_id'
+    }
+}).addValue({
+    name: 'period',
+    table: 'matricula',
+    tableField: 'turma_turno_id',
+    resultField: 'period_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'turma_turno_id'
+    }
+}).addValue({
+  name:'integral_time',
+  table: 'matricula',
+  tableField: 'tempo_integral',
+  resultField: 'integral_time_id',
+  where: {
+      relation: '=',
+      type: 'integer',
+      field: 'tempo_integral'
+  }
+}).addValue({
+  name:'age_range_all',
+  table: 'matricula',
+  tableField: 'faixa_etaria_31_03',
+  resultField: 'age_range_all_id',
+  where: {
+      relation: '=',
+      type: 'integer',
+      field: 'faixa_etaria_31_03'
+  }
+}).addValue({
+  name:'special_class',
+  table: 'matricula',
+  tableField: 'exclusiva_especial',
+  resultField: 'special_class_id',
+  where: {
+      relation: '=',
+      type: 'boolean',
+      field: 'exclusiva_especial'
+  }
+}).addValueToField({
+    name: 'period_not',
+    table: 'matricula',
+    tableField: 'turma_turno_id',
+    resultField: 'period_id',
+    where: {
+        relation: '<>',
+        type: 'integer',
+        field: 'turma_turno_id'
+    }
+}, 'filter')
+.addValue({
+    name: 'low_vision',
+    table: 'matricula',
+    tableField: 'baixa_visao',
+    resultField: 'low_vision',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'baixa_visao'
+    } 
+}).addValue({
+    name: 'blindness',
+    table: 'matricula',
+    tableField: 'cegueira',
+    resultField: 'blindness',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'cegueira'
+    }
+}).addValue({
+    name: 'deafness',
+    table: 'matricula',
+    tableField: 'surdez',
+    resultField: 'deafness',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'surdez'
+    }
+}).addValue({
+    name: 'hearing_deficiency',
+    table: 'matricula',
+    tableField: 'deficiencia_auditiva',
+    resultField: 'hearing_deficiency',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'deficiencia_auditiva'
+    }
+}).addValue({
+    name: 'deafblindness',
+    table: 'matricula',
+    tableField: 'surdo_cegueira',
+    resultField: 'deafblindness',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'surdo_cegueira'
+    }
+}).addValue({
+    name: 'physical_disability',
+    table: 'matricula',
+    tableField: 'deficiencia_fisica',
+    resultField: 'physical_disability',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'deficiencia_fisica'
+    }
+}).addValue({
+    name: 'intellectual_disability',
+    table: 'matricula',
+    tableField: 'deficiencia_intelectual',
+    resultField: 'intellectual_disability',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'deficiencia_intelectual'
+    }
+}).addValue({
+    name: 'multiple_disabilities',
+    table: 'matricula',
+    tableField: 'deficiencia_multiplas',
+    resultField: 'multiple_disabilities',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'deficiencia_multiplas'
+    }
+}).addValue({
+    name: 'autism',
+    table: 'matricula',
+    tableField: 'autismo',
+    resultField: 'autism',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'autismo'
+    }
+}).addValue({
+    name: 'autism_spectrum_disorder',
+    table: 'matricula',
+    tableField: 'transtorno_espectro_autista',
+    resultField: 'autism_spectrum_disorder',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'transtorno_espectro_autista'
+    }
+}).addValue({
+    name: 'asperger_syndrom',
+    table: 'matricula',
+    tableField: 'sindrome_asperger',
+    resultField: 'asperger_syndrom',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'sindrome_asperger'
+    }
+}).addValue({
+    name: 'rett_syndrom',
+    table: 'matricula',
+    tableField: 'sindrome_rett',
+    resultField: 'rett_syndrom',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'sindrome_rett'
+    }
+}).addValue({
+    name: 'childhood_desintegrative_disorder',
+    table: 'matricula',
+    tableField: 'transtorno_desintegrativo_da_infancia',
+    resultField: 'childhood_desintegrative_disorder',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'transtorno_desintegrativo_da_infancia'
+    }
+}).addValue({
+    name: 'supergifted',
+    table: 'matricula',
+    tableField: 'superdotado',
+    resultField: 'supergifted',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'superdotado'
+    }
+}).addValue({
+    name: 'pee',
+    table: 'matricula',
+    tableField: 'possui_necessidade_especial',
+    resultField: 'pee_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'possui_necessidade_especial'
+    }
+}).addValue({
+    name: 'pee_por_categoria',
+    table: 'matricula',
+    tableField: 'possui_necessidade_especial',
+    resultField: 'pee_por_categoria',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'possui_necessidade_especial'
+    }
+});
+
+enrollmentApp.get('/', rqf.parse(), (req, res, next) => {
+    if('pee_por_categoria' in req.dims){
+        delete req.dims.pee_por_categoria
+        req.pee_por_categoria = true
+        req.sql.field('SUM(CASE WHEN cegueira = true THEN 1 ELSE 0 END)', 'Cegueira')
+        .field('SUM(CASE WHEN baixa_visao = true THEN 1 ELSE 0 END)', 'Baixa visão')
+        .field('SUM(CASE WHEN surdez = true THEN 1 ELSE 0 END)', 'Surdez')
+        .field('SUM(CASE WHEN deficiencia_auditiva = true THEN 1 ELSE 0 END)', 'Deficiência auditiva')
+        .field('SUM(CASE WHEN surdo_cegueira = true THEN 1 ELSE 0 END)', 'Surdocegueira')
+        .field('SUM(CASE WHEN deficiencia_fisica = true THEN 1 ELSE 0 END)', 'Deficiência física')
+        .field('SUM(CASE WHEN deficiencia_intelectual = true THEN 1 ELSE 0 END)', 'Deficiência intelectual')
+        .field('SUM(CASE WHEN deficiencia_multiplas = true THEN 1 ELSE 0 END)', 'Deficiências múltiplas')
+        .field('SUM(CASE WHEN autismo = true THEN 1 ELSE 0 END)', 'Autismo')
+        .field('SUM(CASE WHEN transtorno_espectro_autista = true THEN 1 ELSE 0 END)', 'Transtorno do Espectro Autista (TEA)')
+        .field('SUM(CASE WHEN sindrome_asperger = true THEN 1 ELSE 0 END)', 'Síndrome de Asperger')
+        .field('SUM(CASE WHEN sindrome_rett = true THEN 1 ELSE 0 END)', 'Síndrome de Rett')
+        .field('SUM(CASE WHEN transtorno_desintegrativo_da_infancia = true THEN 1 ELSE 0 END)', 'Transtorno desintegrativo da infância')
+        .field('SUM(CASE WHEN superdotado = true THEN 1 ELSE 0 END)', 'Altas habilidades / Superdotação')
+        .field('matricula.ano_censo', 'year')
+        .from('matricula')
+        .group('matricula.ano_censo')
+        .order('matricula.ano_censo')
+        .where('((matricula.tipo<=3 OR matricula.tipo IS NULL) AND (matricula.tipo_atendimento_turma IS NULL OR matricula.tipo_atendimento_turma <= 2))');
+    }
+    else{
+        req.sql.field('COUNT(*)', 'total')
+        .field('matricula.ano_censo', 'year')
+        .from('matricula')
+        .group('matricula.ano_censo')
+        .order('matricula.ano_censo')
+        .where('((matricula.tipo<=3 OR matricula.tipo IS NULL) AND (matricula.tipo_atendimento_turma IS NULL OR matricula.tipo_atendimento_turma <= 2))');
+    }
+    next();
+}, rqf.build(), query, id2str.transform(false), (req, res, next) => {
+    if(req.pee_por_categoria === true){
+        let result = req.result;
+        let result_total = [];
+        for (var j = 0;j < result.length;j++){
+            let result_parcial = result[j];
+            for (var i in result_parcial){
+                if(i !== 'year'){
+                    let obj = {};
+                    obj.total = result_parcial[i];
+                    i = i.replace(/"/g, '');
+                    obj.pee_por_categoria_name = i;
+                    obj.year = result_parcial.year;
+                    result_total.push(obj);
+                }
+            }
+        }
+        req.result= result_total;
+    }
+    next();
+}, response('enrollment'));
+
+enrollmentApp.get('/diagnosis', rqf.parse(), (req, res, next) => {
+    req.dims = {};
+    req.dims.state = true;
+    req.dims.city = true;
+    req.dims.school_year = true;
+    req.dims.location = true;
+    req.dims.adm_dependency_detailed = true;
+
+    req.sql.field('COUNT(*)', 'total')
+    .field("'Brasil'", 'name')
+    .field('matricula.ano_censo', 'year')
+    .from('matricula')
+    .group('matricula.ano_censo')
+    .order('matricula.ano_censo')
+    .where('matricula.tipo<=3');
+
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    let enrollments = req.result;
+
+    // Gera a relação etapa de ensino X ano escolar
+    let educationSchoolYear = {};
+    for(let i = 10; i < 80; ++i) {
+        if(id2str.schoolYear(i) !== id2str.schoolYear(99)) {
+            let educationLevelId = Math.floor(i/10);
+            educationSchoolYear[i] = {
+                id: educationLevelId,
+                name: id2str.educationLevelShort(educationLevelId),
+            };
+        }
+    }
+
+    let result = [];
+    let educationLevelSet = new Set();
+    let schoolYearSet = new Set();
+    let i = 0;
+    while(i < enrollments.length) {
+        let enrollment = enrollments[i];
+        if(!educationSchoolYear[enrollment.school_year_id]) {
+            ++i;
+            continue;
+        }
+        let educationLevelHash = '' + enrollment.year + educationSchoolYear[enrollment.school_year_id].id + enrollment.city_id;
+        let schoolYearHash = '' + enrollment.year + enrollment.school_year_id + enrollment.city_id;
+
+        let currentEducation = null;
+        // Busca ou cria a etapa de ensino adequada
+        if(educationLevelSet.has(educationLevelHash)) {
+            let j = 0;
+            let edu = result[j];
+            while(j < result.length && (edu.year != enrollment.year || edu.education_level_school_year_id != educationSchoolYear[enrollment.school_year_id].id)) {
+                ++j;
+                edu = result[j];
+            }
+            if(j >= result.length) --j;
+            edu = result[j];
+
+            currentEducation = edu;
+        } else {
+            educationLevelSet.add(educationLevelHash);
+            let obj = {
+                year: enrollment.year,
+                name: enrollment.name,
+                state_id: enrollment.state_id,
+                state_name: enrollment.state_name,
+                city_id: enrollment.city_id,
+                city_name: enrollment.city_name,
+                education_level_school_year_id: educationSchoolYear[enrollment.school_year_id].id,
+                education_level_school_year_name: educationSchoolYear[enrollment.school_year_id].name,
+                total: 0,
+                adm_dependencies: [
+                    {
+                        adm_dependency_detailed_id: enrollment.adm_dependency_detailed_id,
+                        adm_dependency_detailed_name: enrollment.adm_dependency_detailed_name,
+                        total: 0
+                    }
+                ],
+                locations: [
+                    {
+                        location_id: enrollment.location_id,
+                        location_name: enrollment.location_name,
+                        total: 0
+                    }
+                ]
+            };
+
+            result.push(obj);
+            currentEducation = obj;
+        }
+
+        let currentSchoolYear = null;
+        // Busca ou cria a série adequada
+        if(schoolYearSet.has(schoolYearHash)) {
+            let j = 0;
+            let edu = result[j];
+            while(j < result.length && (edu.year != enrollment.year || edu.education_level_school_year_id != enrollment.school_year_id)) {
+                ++j;
+                edu = result[j];
+            }
+            if(j >= result.length) --j;
+            edu = result[j];
+
+            currentSchoolYear = edu;
+        } else {
+            schoolYearSet.add(schoolYearHash);
+            let obj = {
+                year: enrollment.year,
+                name: enrollment.name,
+                state_id: enrollment.state_id,
+                state_name: enrollment.state_name,
+                city_id: enrollment.city_id,
+                city_name: enrollment.city_name,
+                education_level_school_year_id: enrollment.school_year_id,
+                education_level_school_year_name: enrollment.school_year_name,
+                total: 0,
+                adm_dependencies: [
+                    {
+                        adm_dependency_detailed_id: enrollment.adm_dependency_detailed_id,
+                        adm_dependency_detailed_name: enrollment.adm_dependency_detailed_name,
+                        total: 0
+                    }
+                ],
+                locations: [
+                    {
+                        location_id: enrollment.location_id,
+                        location_name: enrollment.location_name,
+                        total: 0
+                    }
+                ]
+            };
+
+            result.push(obj);
+            currentSchoolYear = obj;
+        }
+
+        // Adiciona ao total
+        currentEducation.total += enrollment.total;
+        currentSchoolYear.total += enrollment.total;
+
+        // Adiciona ao total da dependência administrativa
+        let admDependencyIndex = 0;
+        let admDependency = currentEducation.adm_dependencies[admDependencyIndex];
+        while (admDependencyIndex < currentEducation.adm_dependencies.length && enrollment.adm_dependency_detailed_id > admDependency.adm_dependency_detailed_id) {
+            ++admDependencyIndex;
+            admDependency = currentEducation.adm_dependencies[admDependencyIndex];
+        }
+        if(admDependencyIndex >= currentEducation.adm_dependencies.length || admDependency.adm_dependency_detailed_id != enrollment.adm_dependency_detailed_id) { // não encontrou
+            let obj = {
+                adm_dependency_detailed_id: enrollment.adm_dependency_detailed_id,
+                adm_dependency_detailed_name: enrollment.adm_dependency_detailed_name,
+                total: 0
+            }
+            currentEducation.adm_dependencies.splice(admDependencyIndex, 0, obj);
+            admDependency = obj;
+        }
+        admDependency.total += enrollment.total;
+
+        admDependencyIndex = 0;
+        admDependency = currentSchoolYear.adm_dependencies[admDependencyIndex];
+        while (admDependencyIndex < currentSchoolYear.adm_dependencies.length && enrollment.adm_dependency_detailed_id > admDependency.adm_dependency_detailed_id) {
+            ++admDependencyIndex;
+            admDependency = currentSchoolYear.adm_dependencies[admDependencyIndex];
+        }
+        if(admDependencyIndex >= currentSchoolYear.adm_dependencies.length || admDependency.adm_dependency_detailed_id != enrollment.adm_dependency_detailed_id) { // não encontrou
+            let obj = {
+                adm_dependency_detailed_id: enrollment.adm_dependency_detailed_id,
+                adm_dependency_detailed_name: enrollment.adm_dependency_detailed_name,
+                total: 0
+            }
+            currentSchoolYear.adm_dependencies.splice(admDependencyIndex, 0, obj);
+            admDependency = obj;
+        }
+        admDependency.total += enrollment.total;
+
+        // Adiciona ao total da localidade
+        let locationIndex = 0;
+        let location = currentEducation.locations[locationIndex];
+        while (locationIndex < currentEducation.locations.length && enrollment.location_id > location.location_id) {
+            ++locationIndex;
+            location = currentEducation.locations[locationIndex];
+        }
+        if(locationIndex >= currentEducation.locations.length || location.location_id != enrollment.location_id) {
+            let obj = {
+                location_id: enrollment.location_id,
+                location_name: enrollment.location_name,
+                total: 0
+            }
+            currentEducation.locations.splice(locationIndex, 0, obj);
+            location = obj;
+        }
+        location.total += enrollment.total;
+
+        locationIndex = 0;
+        location = currentSchoolYear.locations[locationIndex];
+        while (locationIndex < currentSchoolYear.locations.length && enrollment.location_id > location.location_id) {
+            ++locationIndex;
+            location = currentSchoolYear.locations[locationIndex];
+        }
+        if(locationIndex >= currentSchoolYear.locations.length || location.location_id != enrollment.location_id) {
+            let obj = {
+                location_id: enrollment.location_id,
+                location_name: enrollment.location_name,
+                total: 0
+            }
+            currentSchoolYear.locations.splice(locationIndex, 0, obj);
+            location = obj;
+        }
+        location.total += enrollment.total;
+
+        ++i;
+    }
+
+    req.result = result;
+
+    next();
+}, response('enrollment_diagnosis'));
+
+enrollmentApp.get('/projection', rqf.parse(), (req, res, next) => {
+    req.dims = {};
+    req.dims.state = true;
+    req.dims.city = true;
+    req.dims.location = true;
+    req.dims.school_year = true;
+    req.dims.adm_dependency = true;
+    req.dims.period = true;
+    req.filter.adm_dependency = [1,2,3];
+
+    req.sql.field('COUNT(*)', 'total')
+    .field("'Brasil'", 'name')
+    .field('matricula.ano_censo', 'year')
+    .from('matricula')
+    .group('matricula.ano_censo')
+    .order('matricula.ano_censo')
+    .where('matricula.tipo<=3');
+
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    let enrollments = req.result;
+
+    // Gera a relação etapa de ensino X ano escolar
+    let educationSchoolYear = {};
+    for(let i = 10; i < 80; ++i) {
+        if(id2str.schoolYear(i) !== id2str.schoolYear(99)) {
+            let educationLevelId = Math.floor(i/10);
+            educationSchoolYear[i] = {
+                id: educationLevelId,
+                name: id2str.educationLevelShort(educationLevelId),
+            };
+        }
+    }
+
+    let result = [];
+    let educationLevelSet = new Set();
+    let schoolYearSet = new Set();
+    let i = 0;
+    while(i < enrollments.length) {
+        let enrollment = enrollments[i];
+        if(!educationSchoolYear[enrollment.school_year_id]) {
+            ++i;
+            continue;
+        }
+        let educationLevelHash = '' + enrollment.year + educationSchoolYear[enrollment.school_year_id].id + enrollment.city_id;
+        let schoolYearHash = '' + enrollment.year + enrollment.school_year_id + enrollment.city_id;
+
+        let currentEducation = null;
+        // Busca ou cria a etapa de ensino adequada
+        if(educationLevelSet.has(educationLevelHash)) {
+            let j = 0;
+            let edu = result[j];
+            while(j < result.length && (edu.year != enrollment.year || edu.education_level_school_year_id != educationSchoolYear[enrollment.school_year_id].id)) {
+                ++j;
+                edu = result[j];
+            }
+            if((j >= result.length)) --j;
+            edu = result[j];
+
+            currentEducation = edu;
+        } else {
+            educationLevelSet.add(educationLevelHash);
+            let obj = {
+                year: enrollment.year,
+                name: enrollment.name,
+                state_id: enrollment.state_id,
+                state_name: enrollment.state_name,
+                city_id: enrollment.city_id,
+                city_name: enrollment.city_name,
+                education_level_school_year_id: educationSchoolYear[enrollment.school_year_id].id,
+                education_level_school_year_name: educationSchoolYear[enrollment.school_year_id].name,
+                urban_day_total: 0,
+                urban_night_total: 0,
+                rural_day_total: 0,
+                rural_night_total: 0
+            };
+            result.push(obj);
+            currentEducation = obj;
+        }
+
+        let currentSchoolYear = null;
+        // Busca ou cria a série adequada
+        if(schoolYearSet.has(schoolYearHash)) {
+            let j = 0;
+            let edu = result[j];
+            while(j < result.length && (edu.year != enrollment.year || edu.education_level_school_year_id != enrollment.school_year_id)){
+                ++j;
+                edu = result[j];
+            }
+            if(j >= result.length) --j;
+            edu = result[j];
+
+            currentSchoolYear = edu;
+        } else {
+            schoolYearSet.add(schoolYearHash);
+            let obj = {
+                year: enrollment.year,
+                name: enrollment.name,
+                state_id: enrollment.state_id,
+                state_name: enrollment.state_name,
+                city_id: enrollment.city_id,
+                city_name: enrollment.city_name,
+                education_level_school_year_id: enrollment.school_year_id,
+                education_level_school_year_name: enrollment.school_year_name,
+                urban_day_total: 0,
+                urban_night_total: 0,
+                rural_day_total: 0,
+                rural_night_total: 0
+            };
+
+            result.push(obj);
+            currentSchoolYear = obj;
+        }
+
+        if(enrollment.location_id == 1) {
+            if(enrollment.period_id < 3) {
+                currentEducation.urban_day_total += enrollment.total;
+                currentSchoolYear.urban_day_total += enrollment.total;
+            } else {
+                currentEducation.urban_night_total += enrollment.total;
+                currentSchoolYear.urban_night_total += enrollment.total;
+            }
+        } else {
+            if(enrollment.period_id < 3) {
+                currentEducation.rural_day_total += enrollment.total;
+                currentSchoolYear.rural_day_total += enrollment.total;
+            } else {
+                currentEducation.rural_night_total += enrollment.total;
+                currentSchoolYear.rural_night_total += enrollment.total;
+            }
+        }
+
+        ++i;
+    }
+
+    req.result = result;
+
+    next();
+}, response('enrollment_projection'));
+
+module.exports = enrollmentApp;
diff --git a/src/libs/routes_v2/enrollmentProjection.js b/src/libs/routes_v2/enrollmentProjection.js
new file mode 100644
index 00000000..4dd92e5d
--- /dev/null
+++ b/src/libs/routes_v2/enrollmentProjection.js
@@ -0,0 +1,239 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const enrollmentProjectionApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'projecao_matricula_por_dependencia'
+    }
+}, 'dims').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: 'nome',
+    resultField: 'state_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'projecao_matricula_por_dependencia'
+    }
+}, 'filter').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'projecao_matricula_por_dependencia'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'projecao_matricula_por_dependencia'
+    }
+}, 'filter').addValue({
+    name: 'min_year',
+    table: 'projecao_matricula_por_dependencia',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'projecao_matricula_por_dependencia',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'adm_dependency',
+    table: 'projecao_matricula_por_dependencia',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+});
+
+enrollmentProjectionApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.field("'Brasil'", 'name')
+        .field('SUM(projecao_matricula_por_dependencia.urbano_dia_total)', 'urban_day_total')
+        .field('SUM(projecao_matricula_por_dependencia.urbano_noite_total)', 'urban_night_total')
+        .field('SUM(projecao_matricula_por_dependencia.rural_dia_total)', 'rural_day_total')
+        .field('SUM(projecao_matricula_por_dependencia.rural_noite_total)', 'rural_night_total')
+        .field('projecao_matricula_por_dependencia.etapa_ensino_escola_ano_id', 'education_level_school_year_id')
+        .field('projecao_matricula_por_dependencia.ano_censo', 'year')
+        .from('projecao_matricula_por_dependencia')
+        .where('projecao_matricula_por_dependencia.etapa_ensino_escola_ano_id <> 7 AND projecao_matricula_por_dependencia.etapa_ensino_escola_ano_id < 71')
+        .group('projecao_matricula_por_dependencia.etapa_ensino_escola_ano_id')
+        .group('projecao_matricula_por_dependencia.ano_censo')
+        .order('projecao_matricula_por_dependencia.ano_censo')
+        .order('projecao_matricula_por_dependencia.etapa_ensino_escola_ano_id');
+
+    next();
+}, query, id2str.transform(), (req, res, next) => {
+
+    let result = [];
+    let i = 0;
+    let j = 1;
+    let base_year = req.result[0].year;
+
+    let hashSet = new Set();
+    let atual_city_name;
+    let atual_city_id;
+    let atual_state_name;
+    let atual_state_id;
+    while (i < req.result.length || j <= 63) {      // Adciona séries não existentes no banco com 0 nos totais.
+        let atual = req.result[i];
+        if (j > 63) {       // Caso j passou da última série existente, mudamos de dimensão.
+            j = 1;
+
+            if (base_year !== atual.year) {
+                base_year = atual.year;
+                hashSet = new Set();
+            }
+        }
+        if (id2str.educationLevelSchoolYear(j) === id2str.educationLevelSchoolYear(99) || j === 7) {
+            j++;
+            continue;
+        }
+
+        if (j == 1) {
+            let hash = ""
+            if ('state' in req.dims)
+                hash += atual.state_id;
+            if ('city' in req.dims)
+                hash += atual.city_id;
+
+            if (!hashSet.has(hash)) {
+                hashSet.add(hash);
+                atual_city_id = atual.city_id;
+                atual_city_name = atual.city_name;
+                atual_state_id = atual.state_id;
+                atual_state_name = atual.state_name;
+            }
+        }
+
+        if (atual !== undefined && atual.education_level_school_year_id === j) {   // Série existe.
+            atual.urban_day_total = parseInt(atual.urban_day_total, 10);
+            atual.urban_night_total = ((atual.education_level_school_year_id >= 3 && atual.education_level_school_year_id < 10) || (atual.education_level_school_year_id >= 30)) ? parseInt(atual.urban_night_total, 10) : 0;   //Não conta matrículas noturnas da pré-escola e da creche
+            atual.rural_day_total = parseInt(atual.rural_day_total, 10);
+            atual.rural_night_total = ((atual.education_level_school_year_id >= 3 && atual.education_level_school_year_id < 10) || (atual.education_level_school_year_id >= 30)) ? parseInt(atual.rural_night_total, 10) : 0;
+            result.push(atual);
+
+            i++;
+        }
+        else {      //  Série não existe, adcionamos 0 ao resultado.
+            let base_result = {
+                name: req.result[0].name,
+                urban_day_total: 0,
+                urban_night_total: 0,
+                rural_day_total: 0,
+                rural_night_total: 0,
+                education_level_school_year_id: j,
+                year: base_year,
+                education_level_school_year_name: id2str.educationLevelSchoolYear(j)
+            };
+
+            if ('city' in req.dims) {           // adciona os campos de cidade e/ou estado
+                base_result.city_id = atual_city_id;
+                base_result.city_name = atual_city_name;
+            }
+            if ('state' in req.dims) {
+                base_result.state_id = atual_state_id;
+                base_result.state_name = atual_state_name;
+            }
+
+            result.push(base_result)
+        }
+        j++;
+    }
+
+    req.result = result;
+    next();
+}, response('enrollment_projection'));
+
+module.exports = enrollmentProjectionApp;
diff --git a/src/libs/routes_v2/financial.js b/src/libs/routes_v2/financial.js
new file mode 100644
index 00000000..dab9dfe8
--- /dev/null
+++ b/src/libs/routes_v2/financial.js
@@ -0,0 +1,186 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const financialApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+financialApp.get('/year_range', (req, res, next) => {
+    req.sql.from('indicadores_financeiros')
+    .field('MIN(indicadores_financeiros.ano_censo)', 'start_year')
+    .field('MAX(indicadores_financeiros.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+financialApp.get('/years', (req, res, next) => {
+    req.sql.from('indicadores_financeiros')
+    .field('DISTINCT indicadores_financeiros.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+financialApp.get('/sphere_adm', (req, res, next) => {
+    req.result = [
+        {id: 1, name: "1"},
+        {id: 2, name: "2"}
+    ]
+    next();
+}, response('sphere_adm'));
+
+financialApp.get('/financial_data', (req, res, next) => {
+    req.sql.from('indicadores_financeiros')
+    .field('DISTINCT indicadores_financeiros.dados_financeiros', 'financial_data');
+    next();
+}, query, response('financial_data'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['sigla', 'id'],
+    resultField: ['sigla_uf', 'cod_uf'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'indicadores_financeiros'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'indicadores_financeiros'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'indicadores_financeiros',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'indicadores_financeiros',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'indicadores_financeiros',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'indicadores_financeiros',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'sphere_adm',
+    table: 'indicadores_financeiros',
+    tableField: 'esfera_adm',
+    resultField: 'sphere_adm_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'esfera_adm'
+    }
+}).addValue({
+    name: 'city',
+    table: 'indicadores_financeiros',
+    tableField: 'municipio_id',
+    resultField: 'city_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id'
+    }
+}).addValue({
+    name: 'financial_data',
+    table: 'indicadores_financeiros',
+    tableField: 'dados_financeiros',
+    resultField: 'financial_data_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dados_financeiros'
+    }
+});
+
+financialApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+		if ("state" in req.filter) {
+	    req.sql.from('indicadores_financeiros')
+	    .field('indicadores_financeiros.estado_id', 'state_id')
+	    .field('indicadores_financeiros.ano_censo', 'year')
+	    .field('estado.sigla', 'state_abbreviation')
+	    .field('indicadores_financeiros.valor', 'valor')
+	    .field('indicadores_financeiros.esfera_adm', 'sphere_adm_id')
+	    .field('indicadores_financeiros.dados_financeiros', 'financial_data_id')
+	    .group('indicadores_financeiros.ano_censo')
+	    .group('indicadores_financeiros.estado_id')
+	    .group('estado.sigla')
+	    .group('indicadores_financeiros.valor')
+	    .group('indicadores_financeiros.dados_financeiros')
+	    .group('indicadores_financeiros.esfera_adm')
+		} else {
+	    req.sql.from('indicadores_financeiros')
+	    .field('indicadores_financeiros.estado_id', 'state_id')
+	    .field('indicadores_financeiros.ano_censo', 'year')
+	    .field('estado.sigla', 'state_abbreviation')
+	    .field('indicadores_financeiros.valor', 'valor')
+	    .field('indicadores_financeiros.esfera_adm', 'sphere_adm_id')
+	    .field('indicadores_financeiros.dados_financeiros', 'financial_data_id')
+	    .join('estado', null, 'indicadores_financeiros.estado_id=estado.id')
+	    .group('indicadores_financeiros.ano_censo')
+	    .group('indicadores_financeiros.estado_id')
+	    .group('estado.sigla')
+	    .group('indicadores_financeiros.valor')
+	    .group('indicadores_financeiros.dados_financeiros')
+	    .group('indicadores_financeiros.esfera_adm')
+		}
+    next();
+}, query, id2str.transform(), response('financial'));
+
+module.exports = financialApp;
diff --git a/src/libs/routes_v2/glossEnrollmentRatio.js b/src/libs/routes_v2/glossEnrollmentRatio.js
new file mode 100644
index 00000000..ab57e325
--- /dev/null
+++ b/src/libs/routes_v2/glossEnrollmentRatio.js
@@ -0,0 +1,390 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const glossEnrollmentRatioApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+glossEnrollmentRatioApp.use(cache('15 day'));
+
+// Complete range of the enrollments dataset.
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+glossEnrollmentRatioApp.get('/year_range', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('DISTINCT pnad.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('matricula')
+    .field('DISTINCT matricula.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let distinct_years = [];
+    let new_result = [];
+    for (let i = 0; i < req.oldResult.length; i++) {
+        for (let j = 0; j < req.result.length; j++) {
+            if(req.oldResult[i].year == req.result[j].year) {
+                distinct_years.push(req.oldResult[i]);
+            }
+        }
+    }
+    new_result.push({start_year: distinct_years[distinct_years.length -1].year, end_year: distinct_years[0].year});
+    req.result = new_result;
+    next();
+}, response('range'));
+
+glossEnrollmentRatioApp.get('/years', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('DISTINCT pnad.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('matricula')
+    .field('DISTINCT matricula.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let new_result = []
+    for (let i = 0; i < req.oldResult.length; i++) {
+        for (let j = 0; j < req.result.length; j++) {
+            if(req.oldResult[i].year == req.result[j].year) {
+                new_result.push(req.oldResult[i]);
+            }
+        }
+    }
+    req.result = new_result;
+    next();
+}, response('years'));
+
+glossEnrollmentRatioApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'pnad\'');
+    next();
+}, query, response('source'));
+
+glossEnrollmentRatioApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Creche'},
+        {id: 2, name: 'Pré-Escola'},
+        {id: 3, name: 'Ensino Fundamental - anos iniciais'},
+        {id: 4, name: 'Ensino Fundamental - anos finais'},
+        {id: 5, name: 'Ensino Médio'}
+    ];
+    next();
+}, response('education_level_short'));
+
+glossEnrollmentRatioApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+glossEnrollmentRatioApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [
+        {id: 0, name: 'Sem declaração'},
+        {id: 1, name: 'Branca'},
+        {id: 2, name: 'Preta'},
+        {id: 3, name: 'Parda'},
+        {id: 4, name: 'Amarela'},
+        {id: 5, name: 'Indígena'}
+    ];
+    next();
+}, response('ethnic_group'));
+
+glossEnrollmentRatioApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: '@',
+    tableField: 'cor_raca_id',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca_id'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'gender',
+    table: '@',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'location',
+    table: '@',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'matricula',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+});
+
+function matchQueries(queryTotal, queryPartial) {
+    let match = [];
+    queryPartial.forEach((result) => {
+        let newObj = {};
+        let keys = Object.keys(result);
+        keys.forEach((key) => {
+            newObj[key] = result[key];
+        });
+        // console.log('NEW OBJ');
+        // console.log(newObj);
+        // remove total
+        let index = keys.indexOf('total');
+        if(index > -1) keys.splice(index, 1);
+        // remove education_level_short_id
+        index = keys.indexOf('education_level_short_id');
+        if(index > -1) keys.splice(index, 1);
+        // remove education_level_short_name
+        index = keys.indexOf('education_level_short_name');
+        if(index > -1) keys.splice(index, 1);
+        let objMatch = null;
+
+        for(let i = 0; i < queryTotal.length; ++i) {
+            let total = queryTotal[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(total[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                objMatch = total;
+                break;
+            }
+        }
+
+        if(objMatch) {
+            // console.log('MATCH!!!!');
+            // console.log(objMatch);
+            newObj.total = (result.total / objMatch.total) * 100;
+            newObj.partial = result.total;
+            newObj.denominator = objMatch.total
+            match.push(newObj);
+        }
+    });
+    // console.log('TAMANHOS');
+    // console.log(queryTotal.length);
+    // console.log(queryPartial.length);
+    // console.log(match.length);
+    return match;
+}
+
+glossEnrollmentRatioApp.get('/', rqf.parse(),(req, res, next) => {
+    req.numerator = {};
+    req.denominator = {};
+    let glossEnrollmentRatioApp = {};
+
+    req.sql.from('matricula')
+    .field('count(*)', 'total')
+    .field('matricula.ano_censo', 'year')
+    .group('matricula.ano_censo')
+    .order('matricula.ano_censo')
+    .where('matricula.tipo <= 3')
+
+    if ( "education_level_short" in req.dims ) {
+        req.sql.field('matricula.etapa_resumida', 'age_range')
+        req.sql.where('matricula.etapa_resumida = 1 OR matricula.etapa_resumida = 2 OR matricula.etapa_resumida = 3 OR matricula.etapa_resumida = 4 OR matricula.etapa_resumida = 5')
+        req.sql.group('matricula.etapa_resumida', 'age_range');
+    }
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.numerator = req.result;
+    req.resetSql();
+    req.sql.field('sum(peso)', 'total')
+    .field('pnad.ano_censo','year')
+    .from('pnad')
+    .group('pnad.ano_censo')
+    .order('pnad.ano_censo')
+
+    function convert(result) {
+        if (result == 1) {
+            return 'pnad.faixa_etaria_31_03 = 1'
+        } else if (result == 2) {
+            return 'pnad.faixa_etaria_31_03 = 2'
+        } else if (result == 4) {
+            return 'pnad.faixa_etaria_31_03 = 3'
+        } else if (result == 5) {
+            return 'pnad.faixa_etaria_31_03 = 4'
+        } else if (result == 6) {
+            return 'pnad.faixa_etaria_31_03 = 5'
+        }
+    }
+
+    //remove education_level_short how filter and add faixa_etaria_31_03 in filter
+    if ("education_level_short" in req.filter) {
+        if (Array.isArray(req.filter.education_level_short)) {
+            var string_query = '';
+            for(let i = 0; i < req.filter.education_level_short.length - 1; i++)  {
+                string_query = string_query + convert(req.filter.education_level_short[i]) + ' OR ';
+            }
+            string_query = string_query + convert(req.filter.education_level_short[req.filter.education_level_short.length - 1]);
+            req.sql.where(string_query);
+            req.sql.field('pnad.faixa_etaria_31_03','age_range')
+            req.sql.group('pnad.faixa_etaria_31_03', 'age_range')
+        }
+    } else if ( "education_level_short" in req.dims ) {
+        req.sql.field('pnad.faixa_etaria_31_03','age_range')
+        req.sql.where('pnad.faixa_etaria_31_03 = 1 OR pnad.faixa_etaria_31_03 = 2 OR pnad.faixa_etaria_31_03 = 3 OR pnad.faixa_etaria_31_03 = 4 OR pnad.faixa_etaria_31_03 = 5')
+        req.sql.group('pnad.faixa_etaria_31_03', 'age_range');
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+
+    next();
+}, rqf.parse(), (req, res, next) => {
+    if ("education_level_short" in req.filter) {
+        delete req.filter.education_level_short;
+    }
+    if ("education_level_short" in req.dims) {
+        delete req.dims.education_level_short;
+    }
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.denominator = req.result;
+    req.result = []
+    let glossEnrollment = matchQueries(req.denominator, req.numerator);
+    req.result = glossEnrollment;
+
+    next();
+}, response('glossEnrollmentRatio'));
+
+module.exports = glossEnrollmentRatioApp;
diff --git a/src/libs/routes_v2/idhm.js b/src/libs/routes_v2/idhm.js
new file mode 100644
index 00000000..5b941793
--- /dev/null
+++ b/src/libs/routes_v2/idhm.js
@@ -0,0 +1,212 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const idhmApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+idhmApp.use(cache('15 day'));
+
+idhmApp.get('/year_range', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('MIN(adh_idh.ano_censo)', 'start_year')
+    .field('MAX(adh_idh.ano_censo)', 'end_year');
+    next();
+}, query, (req, res, next) => {
+    req.sql.from('adh_idh_uf')
+    .field('MIN(adh_idh_uf.ano_censo)', 'start_year')
+    .field('MAX(adh_idh_uf.ano_censo)', 'end_year');
+    req.old_result = req.result;
+    next();
+}, query, (req, res, next) => {
+    if (req.old_result[0].start_year < req.result[0].start_year) {
+        req.result[0].start_year = req.old_result[0].start_year;
+    }
+    if (req.old_result[0].end_year > req.result[0].end_year) {
+        req.result[0].end_year = req.old_result[0].old_result;
+    }
+    next();
+}, query, response('range'));
+
+idhmApp.get('/years', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('DISTINCT adh_idh.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('adh_idh_uf')
+    .field('DISTINCT adh_idh_uf.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let result = Object.assign(req.oldResult, req.result);
+    req.result = result;
+    next();
+}, response('years'));
+
+idhmApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'adh_idh\'');
+    next();
+}, query, response('source'));
+
+idhmApp.get('/IDHM_level', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Muito Baixa'},
+        {id: 2, name: 'Baixo'},
+        {id: 3, name: 'Médio'},
+        {id: 4, name: 'Alto'},
+        {id: 5, name: 'Muito Alto'}
+    ];
+    next();
+}, response('IDHM_level'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'adh_idh'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'adh_idh'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'idhm_level',
+    table: '@',
+    tableField: 'idhm_nivel',
+    resultField: 'idhm_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: '@',
+        field: 'idhm_nivel'
+    }
+});
+
+
+idhmApp.get('/', rqf.parse(), (req, res, next) => {
+      if (("city" in req.dims) || ("city" in req.filter)) {
+          req.sql.from('adh_idh')
+          .field('adh_idh.idhm', 'total')
+          .field('adh_idh.ano_censo', 'year')
+          .field('adh_idh.municipio_id', 'city_id')
+          .field('adh_idh.estado_id', 'state_id')
+          .group('adh_idh.idhm')
+          .group('adh_idh.ano_censo')
+          .group('adh_idh.municipio_id')
+          .group('adh_idh.estado_id')
+      } else if (("state" in req.filter) || ("state" in req.dims)) {
+          req.sql.from('adh_idh_uf')
+          .field('adh_idh_uf.idhm', 'total')
+          .field('adh_idh_uf.ano_censo', 'year')
+          .field('adh_idh_uf.estado_id', 'state_id')
+          .group('adh_idh_uf.idhm')
+          .group('adh_idh_uf.ano_censo')
+          .group('adh_idh_uf.estado_id')
+        } else {
+            res.status(400);
+            next({
+                status: 400,
+                message: 'Wrong/No filter specified'
+            });
+        }
+        next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('idhm'));
+
+module.exports = idhmApp;
diff --git a/src/libs/routes_v2/idhme.js b/src/libs/routes_v2/idhme.js
new file mode 100644
index 00000000..1b492c61
--- /dev/null
+++ b/src/libs/routes_v2/idhme.js
@@ -0,0 +1,185 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const idhmeApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+idhmeApp.use(cache('15 day'));
+
+idhmeApp.get('/year_range', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('MIN(adh_idh.ano_censo)', 'start_year')
+    .field('MAX(adh_idh.ano_censo)', 'end_year');
+    next();
+}, query, (req, res, next) => {
+    req.sql.from('adh_idh_uf')
+    .field('MIN(adh_idh_uf.ano_censo)', 'start_year')
+    .field('MAX(adh_idh_uf.ano_censo)', 'end_year');
+    req.old_result = req.result;
+    next();
+}, query, (req, res, next) => {
+    if (req.old_result[0].start_year < req.result[0].start_year) {
+        req.result[0].start_year = req.old_result[0].start_year;
+    }
+    if (req.old_result[0].end_year > req.result[0].end_year) {
+        req.result[0].end_year = req.old_result[0].old_result;
+    }
+    next();
+}, query, response('range'));
+
+idhmeApp.get('/years', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('DISTINCT adh_idh.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('adh_idh_uf')
+    .field('DISTINCT adh_idh_uf.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let result = Object.assign(req.oldResult, req.result);
+    req.result = result;
+    next();
+}, response('years'));
+
+idhmeApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'adh_idh\'');
+    next();
+}, query, response('source'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'adh_idh'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'adh_idh'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+});
+
+idhmeApp.get('/', rqf.parse(), (req, res, next) => {
+
+    if (("city" in req.dims) || ("city" in req.filter)) {
+        req.sql.from('adh_idh')
+        .field('adh_idh.idhm_e', 'total')
+        .field('adh_idh.ano_censo', 'year')
+        .field('adh_idh.municipio_id', 'city_id')
+        .group('adh_idh.idhm_e')
+        .group('adh_idh.ano_censo')
+        .group('adh_idh.municipio_id');
+    } else if (("state" in req.filter) || ("state" in req.dims)) {
+        req.sql.from('adh_idh_uf')
+        .field('adh_idh_uf.idhm_e', 'total')
+        .field('adh_idh_uf.ano_censo', 'year')
+        .field('adh_idh_uf.estado_id', 'state_id')
+        .group('adh_idh_uf.idhm_e')
+        .group('adh_idh_uf.ano_censo')
+        .group('adh_idh_uf.estado_id');
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+    next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('idhme'));
+
+module.exports = idhmeApp;
diff --git a/src/libs/routes_v2/idhml.js b/src/libs/routes_v2/idhml.js
new file mode 100644
index 00000000..eee89ddf
--- /dev/null
+++ b/src/libs/routes_v2/idhml.js
@@ -0,0 +1,185 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const idhmlApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+idhmlApp.use(cache('15 day'));
+
+idhmlApp.get('/year_range', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('MIN(adh_idh.ano_censo)', 'start_year')
+    .field('MAX(adh_idh.ano_censo)', 'end_year');
+    next();
+}, query, (req, res, next) => {
+    req.sql.from('adh_idh_uf')
+    .field('MIN(adh_idh_uf.ano_censo)', 'start_year')
+    .field('MAX(adh_idh_uf.ano_censo)', 'end_year');
+    req.old_result = req.result;
+    next();
+}, query, (req, res, next) => {
+    if (req.old_result[0].start_year < req.result[0].start_year) {
+        req.result[0].start_year = req.old_result[0].start_year;
+    }
+    if (req.old_result[0].end_year > req.result[0].end_year) {
+        req.result[0].end_year = req.old_result[0].old_result;
+    }
+    next();
+}, query, response('range'));
+
+idhmlApp.get('/years', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('DISTINCT adh_idh.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('adh_idh_uf')
+    .field('DISTINCT adh_idh_uf.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let result = Object.assign(req.oldResult, req.result);
+    req.result = result;
+    next();
+}, response('years'));
+
+idhmlApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'adh_idh\'');
+    next();
+}, query, response('source'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'adh_idh'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'adh_idh'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+});
+
+idhmlApp.get('/', rqf.parse(), (req, res, next) => {
+
+    if (("city" in req.dims) || ("city" in req.filter)) {
+        req.sql.from('adh_idh')
+        .field('adh_idh.idhm_l', 'total')
+        .field('adh_idh.ano_censo', 'year')
+        .field('adh_idh.municipio_id', 'city_id')
+        .group('adh_idh.idhm_l')
+        .group('adh_idh.ano_censo')
+        .group('adh_idh.municipio_id');
+    } else if (("state" in req.filter) || ("state" in req.dims)) {
+        req.sql.from('adh_idh_uf')
+        .field('adh_idh_uf.idhm_l', 'total')
+        .field('adh_idh_uf.ano_censo', 'year')
+        .field('adh_idh_uf.estado_id', 'state_id')
+        .group('adh_idh_uf.idhm_l')
+        .group('adh_idh_uf.ano_censo')
+        .group('adh_idh_uf.estado_id');
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+    next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('idhme'));
+
+module.exports = idhmlApp;
diff --git a/src/libs/routes_v2/idhmr.js b/src/libs/routes_v2/idhmr.js
new file mode 100644
index 00000000..a9117cb4
--- /dev/null
+++ b/src/libs/routes_v2/idhmr.js
@@ -0,0 +1,188 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const idhmrApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+idhmrApp.use(cache('15 day'));
+
+idhmrApp.get('/year_range', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('MIN(adh_idh.ano_censo)', 'start_year')
+    .field('MAX(adh_idh.ano_censo)', 'end_year');
+    next();
+}, query, (req, res, next) => {
+    req.sql.from('adh_idh_uf')
+    .field('MIN(adh_idh_uf.ano_censo)', 'start_year')
+    .field('MAX(adh_idh_uf.ano_censo)', 'end_year');
+    req.old_result = req.result;
+    next();
+}, query, (req, res, next) => {
+    if (req.old_result[0].start_year < req.result[0].start_year) {
+        req.result[0].start_year = req.old_result[0].start_year;
+    }
+    if (req.old_result[0].end_year > req.result[0].end_year) {
+        req.result[0].end_year = req.old_result[0].old_result;
+    }
+    next();
+}, query, response('range'));
+
+idhmrApp.get('/years', (req, res, next) => {
+    req.sql.from('adh_idh')
+    .field('DISTINCT adh_idh.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('adh_idh_uf')
+    .field('DISTINCT adh_idh_uf.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let result = Object.assign(req.oldResult, req.result);
+    req.result = result;
+    next();
+}, response('years'));
+
+idhmrApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'adh_idh\'');
+    next();
+}, query, response('source'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'adh_idh'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'adh_idh'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+});
+
+idhmrApp.get('/', rqf.parse(), (req, res, next) => {
+    if (("city" in req.dims) || ("city" in req.filter)) {
+        req.sql.from('adh_idh')
+        .field('adh_idh.idhm_r', 'total')
+        .field('adh_idh.ano_censo', 'year')
+        .field('adh_idh.municipio_id', 'city_id')
+        .field('adh_idh.estado_id', 'state_id')
+        .group('adh_idh.idhm_r')
+        .group('adh_idh.ano_censo')
+        .group('adh_idh.municipio_id')
+        .group('adh_idh.estado_id')
+    } else if (("state" in req.filter) || ("state" in req.dims)) {
+        req.sql.from('adh_idh_uf')
+        .field('adh_idh_uf.idhm_r', 'total')
+        .field('adh_idh_uf.ano_censo', 'year')
+        .field('adh_idh_uf.estado_id', 'state_id')
+        .group('adh_idh_uf.idhm_r')
+        .group('adh_idh_uf.ano_censo')
+        .group('adh_idh_uf.estado_id')
+      } else {
+          res.status(400);
+          next({
+              status: 400,
+              message: 'Wrong/No filter specified'
+          });
+      }
+      next();
+}, rqf.build(), query, addMissing(rqf), id2str.transform(), response('idhmr'));
+
+module.exports = idhmrApp;
diff --git a/src/libs/routes_v2/infrastructure.js b/src/libs/routes_v2/infrastructure.js
new file mode 100644
index 00000000..1ececa91
--- /dev/null
+++ b/src/libs/routes_v2/infrastructure.js
@@ -0,0 +1,585 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const infrastructureApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+infrastructureApp.use(cache('15 day'));
+
+infrastructureApp.get('/year_range', (req, res, next) => {
+    req.sql.from('escola')
+    .field('MIN(escola.ano_censo)', 'start_year')
+    .field('MAX(escola.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+infrastructureApp.get('/years', (req, res, next) => {
+    req.sql.from('escola')
+    .field('DISTINCT escola.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+infrastructureApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'escola\'');
+    next();
+}, query, response('source'));
+
+infrastructureApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+infrastructureApp.get('/rural_location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: "Urbana"},
+        {id: 2, name: "Rural"},
+        {id: 3, name: "Rural - Área de assentamento"},
+        {id: 4, name: "Rural - Terra indígena"},
+        {id: 5, name: "Rural - Área remanescente de quilombos"},
+        {id: 6, name: "Rural - Unidade de uso sustentável"}
+    ];
+    next();
+}, response('rural_location'));
+
+infrastructureApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+infrastructureApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'id',
+    resultField: 'city_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'filter').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}, 'dims').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: 'id',
+    resultField: 'state_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}, 'filter').addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'location',
+    table: 'escola',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'escola',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'adm_dependency',
+    table: 'escola',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'escola',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+});
+
+function matchQueries(queryTotal, queryPartial) {
+    let match = [];
+    queryTotal.forEach((result) => {
+        let newObj = {};
+        let keys = Object.keys(result);
+        keys.forEach((key) => {
+            newObj[key] = result[key];
+        });
+        let index = keys.indexOf('total');
+        if(index > -1) keys.splice(index, 1);
+        let objMatch = null;
+
+        for(let i = 0; i < queryPartial.length; ++i) {
+            let partial = queryPartial[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(partial[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                objMatch = partial;
+                break;
+            }
+        }
+
+        if(objMatch) {
+            newObj.percentage = (objMatch.total / result.total) * 100;
+            newObj.partial = objMatch.total;
+            newObj.total = result.total
+            match.push(newObj);
+        }
+    });
+
+    return match;
+}
+
+infrastructureApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.querySet = [];
+    req.queryIndex = {};
+
+    // Local de funcionamento
+    let allSchools = req.sql.clone();
+    allSchools.from('escola').field('COUNT(escola.id)', 'total')
+    .field("'Brasil'", 'name')
+    .field('escola.ano_censo', 'year')
+    .group('escola.ano_censo')
+    .where('escola.situacao_de_funcionamento = 1')
+    .order('escola.ano_censo');
+    req.queryIndex.allSchools = req.querySet.push(allSchools) - 1;
+
+    let schoolPlace = allSchools.clone();
+    schoolPlace.where('escola.func_predio_escolar = 1 AND escola.func_salas_empresa = 0 AND escola.func_templo_igreja = 0 AND escola.func_casa_professor = 0 AND escola.func_galpao = 0 AND escola.biblioteca = 1');
+    req.queryIndex.schoolPlace = req.querySet.push(schoolPlace) - 1;
+
+    // Bibliotecas
+    let allLibraries = allSchools.clone();
+    allLibraries.where('escola.func_predio_escolar = 1 AND escola.localizacao_id = 1');
+    req.queryIndex.allLibraries = req.querySet.push(allLibraries) - 1;
+
+    let haveLibraries = allLibraries.clone();
+    haveLibraries.where('escola.biblioteca = 1');
+    req.queryIndex.haveLibraries = req.querySet.push(haveLibraries) - 1;
+
+    // Bibliotecas/Sala de leitura
+    let allLibrariesReadingRoom = allSchools.clone();
+    allLibrariesReadingRoom.where('escola.func_predio_escolar = 1 AND escola.localizacao_id = 2');
+    req.queryIndex.allLibrariesReadingRoom = req.querySet.push(allLibrariesReadingRoom) - 1;
+
+    let haveLibrariesReadingRoom = allLibrariesReadingRoom.clone();
+    haveLibrariesReadingRoom.where('escola.sala_leitura = 1');
+    req.queryIndex.haveLibrariesReadingRoom = req.querySet.push(haveLibrariesReadingRoom) - 1;
+
+    // Laboratório de informática
+    let allInfLab = allSchools.clone();
+    allInfLab.where('escola.func_predio_escolar = 1')
+    .where('escola.reg_fund_ai = 1 OR escola.reg_fund_af = 1 OR escola.reg_medio_medio = 1 OR escola.reg_medio_integrado = 1 OR escola.reg_medio_normal = 1 OR escola.ensino_eja_fund = 1 OR escola.ensino_eja_medio = 1 OR escola.ensino_eja_prof = 1');
+    req.queryIndex.allInfLab = req.querySet.push(allInfLab) - 1;
+
+    let haveInfLab = allInfLab.clone();
+    haveInfLab.where('escola.lab_informatica = 1');
+    req.queryIndex.haveInfLab = req.querySet.push(haveInfLab) - 1;
+
+    // Laboratório de ciências
+    let allScienceLab = allInfLab.clone();
+    req.queryIndex.allScienceLab = req.querySet.push(allScienceLab) - 1;
+
+    let haveScienceLab = allScienceLab.clone();
+    haveScienceLab.where('escola.lab_ciencias = 1');
+    req.queryIndex.haveScienceLab = req.querySet.push(haveScienceLab) - 1;
+
+    // Parque infantil
+    let allKidsPark = allSchools.clone();
+    allKidsPark.where('escola.func_predio_escolar = 1')
+    .where('escola.reg_infantil_creche = 1 OR escola.reg_infantil_preescola = 1 OR escola.reg_fund_ai = 1 OR escola.esp_infantil_creche = 1 OR escola.esp_exclusiva_creche = 1 OR escola.reg_esp_exclusiva_fund_ai = 1');
+    req.queryIndex.allKidsPark = req.querySet.push(allKidsPark) - 1;
+
+    let haveKidsPark = allKidsPark.clone();
+    haveKidsPark.where('escola.parque_infantil = 1');
+    req.queryIndex.haveKidsPark = req.querySet.push(haveKidsPark) - 1;
+
+    // Berçário
+    let allCribs = allSchools.clone();
+    allCribs.where('escola.func_predio_escolar = 1')
+    .where('escola.reg_infantil_creche = 1 OR escola.esp_infantil_creche = 1');
+    req.queryIndex.allCribs = req.querySet.push(allCribs) - 1;
+
+    let haveCribs = allCribs.clone();
+    haveCribs.where('escola.bercario = 1');
+    req.queryIndex.haveCribs = req.querySet.push(haveCribs) - 1;
+
+    // Quadra
+    let allSportsCourt = allScienceLab.clone();
+    allSportsCourt.where('escola.localizacao_id = 1');
+    req.queryIndex.allSportsCourt = req.querySet.push(allSportsCourt) - 1;
+
+    let haveSportsCourt = allSportsCourt.clone();
+    haveSportsCourt.where('escola.quadra_esportes = 1');
+    req.queryIndex.haveSportsCourt = req.querySet.push(haveSportsCourt) - 1;
+
+    // Quadra coberta
+    req.queryIndex.allCoveredSportsCourt = req.queryIndex.allSportsCourt;
+
+    let haveCoveredSportsCourt = allSportsCourt.clone();
+    haveCoveredSportsCourt.where('escola.quadra_esportes_coberta = 1');
+    req.queryIndex.haveCoveredSportsCourt = req.querySet.push(haveCoveredSportsCourt) - 1;
+
+    // Quadra Descoberta
+    let allUncoveredSportsCourt = allSportsCourt.clone();
+    allUncoveredSportsCourt.where('escola.quadra_esportes_coberta = 0');
+    req.queryIndex.allUncoveredSportsCourt = req.querySet.push(allUncoveredSportsCourt) - 1;
+
+    let haveUncoveredSportsCourt = allUncoveredSportsCourt.clone();
+    haveUncoveredSportsCourt.where('escola.quadra_esportes_descoberta = 1');
+    req.queryIndex.haveUncoveredSportsCourt = req.querySet.push(haveUncoveredSportsCourt) - 1;
+
+    // Sala de direção
+    let allDirectorRoom = allSchools.clone();
+    allDirectorRoom.where('escola.func_predio_escolar = 1 AND escola.localizacao_id = 1');
+    req.queryIndex.allDirectorRoom = req.querySet.push(allDirectorRoom) - 1;
+
+    let haveDirectorRoom = allDirectorRoom.clone();
+    haveDirectorRoom.where('escola.sala_diretoria = 1');
+    req.queryIndex.haveDirectorRoom = req.querySet.push(haveDirectorRoom) - 1;
+
+    // Secretaria
+    let allSecretary = allSchools.clone();
+    allSecretary.where('escola.func_predio_escolar = 1');
+    req.queryIndex.allSecretary = req.querySet.push(allSecretary) - 1;
+
+    let haveSecretary = allSecretary.clone();
+    haveSecretary.where('escola.secretaria = 1');
+    req.queryIndex.haveSecretary = req.querySet.push(haveSecretary) - 1;
+
+    // Sala de professores
+    req.queryIndex.allTeacherRoom = req.queryIndex.allSecretary;
+
+    let haveTeacherRoom = allSecretary.clone();
+    haveTeacherRoom.where('escola.sala_professor = 1');
+    req.queryIndex.haveTeacherRoom = req.querySet.push(haveTeacherRoom) - 1;
+
+    // Cozinha
+    req.queryIndex.allKitchen = req.queryIndex.allSecretary;
+
+    let haveKitchen = allSecretary.clone();
+    haveKitchen.where('escola.cozinha = 1');
+    req.queryIndex.haveKitchen = req.querySet.push(haveKitchen) - 1;
+
+    // Despensa
+    req.queryIndex.allStoreroom = req.queryIndex.allSecretary;
+
+    let haveStoreroom = allSecretary.clone();
+    haveStoreroom.where('escola.despensa = 1');
+    req.queryIndex.haveStoreroom = req.querySet.push(haveStoreroom) - 1;
+
+    // Almoxarifado
+    req.queryIndex.allWarehouse = req.queryIndex.allSecretary;
+
+    let haveWarehouse = allSecretary.clone();
+    haveWarehouse.where('escola.almoxarifado = 1');
+    req.queryIndex.haveWarehouse = req.querySet.push(haveWarehouse) - 1;
+
+    // Internet
+    req.queryIndex.allInternet = req.queryIndex.allLibrariesReadingRoom;
+
+    let haveInternet = allLibrariesReadingRoom.clone();
+    haveInternet.where('escola.internet = 1');
+    req.queryIndex.haveInternet = req.querySet.push(haveInternet) - 1;
+
+    // Internet banda larga
+    req.queryIndex.allBroadbandInternet = req.queryIndex.allLibraries;
+
+    let haveBroadbandInternet = allLibraries.clone();
+    haveBroadbandInternet.where('escola.internet_banda_larga = 1');
+    req.queryIndex.haveBroadbandInternet = req.querySet.push(haveBroadbandInternet) - 1;
+
+    // Banheiro dentro do prédio
+    req.queryIndex.allInsideBathroom = req.queryIndex.allSecretary;
+
+    let haveInsideBathroom = allSecretary.clone();
+    haveInsideBathroom.where('escola.sanitario_dentro_predio = 1');
+    req.queryIndex.haveInsideBathroom = req.querySet.push(haveInsideBathroom) - 1;
+
+    // Banheiro adequado para educação infantil dentro do prédio
+    req.queryIndex.allInsideKidsBathroom = req.queryIndex.allKidsPark;
+
+    let haveInsideKidsBathroom = allKidsPark.clone();
+    haveInsideKidsBathroom.where('escola.sanitario_ei = 1');
+    req.queryIndex.haveInsideKidsBathroom = req.querySet.push(haveInsideKidsBathroom) - 1;
+
+    // Fornecimento de energia
+    req.queryIndex.allEletricEnergy = req.queryIndex.allSecretary;
+
+    let haveEletricEnergy = allSecretary.clone();
+    haveEletricEnergy.where('escola.fornecimento_energia = 1');
+    req.queryIndex.haveEletricEnergy = req.querySet.push(haveEletricEnergy) - 1;
+
+    // Abastecimento de água
+    req.queryIndex.allWaterSupply = req.queryIndex.allSecretary;
+
+    let haveWaterSupply = allSecretary.clone();
+    haveWaterSupply.where('escola.fornecimento_agua = 1');
+    req.queryIndex.haveWaterSupply = req.querySet.push(haveWaterSupply) - 1;
+
+    // Água filtrada
+    req.queryIndex.allFilteredWater = req.queryIndex.allSecretary;
+
+    let haveFilteredWater = allSecretary.clone();
+    haveFilteredWater.where('escola.agua_filtrada = 1');
+    req.queryIndex.haveFilteredWater = req.querySet.push(haveFilteredWater) - 1;
+
+    // Coleta de esgoto
+    req.queryIndex.allSewage = req.queryIndex.allSecretary;
+
+    let haveSewage = allSecretary.clone();
+    haveSewage.where('escola.esgoto_sanitario = 1');
+    req.queryIndex.haveSewage = req.querySet.push(haveSewage) - 1;
+
+    // Sala de recursos multifuncionais para Atendimento Educacional Especializado
+    req.queryIndex.allMultifunctionRoom = req.queryIndex.allSecretary;
+
+    let haveMultifunctionRoom = allSecretary.clone();
+    haveMultifunctionRoom.where('escola.sala_atendimento_especial = 1');
+    req.queryIndex.haveMultifunctionRoom = req.querySet.push(haveMultifunctionRoom) - 1;
+
+    // Banheiros adaptados para pessoas com deficiências
+    req.queryIndex.allSpecialBathroom = req.queryIndex.allSecretary;
+
+    let haveSpecialBathroom = allSecretary.clone();
+    haveSpecialBathroom.where('escola.sanitario_pne = 1');
+    req.queryIndex.haveSpecialBathroom = req.querySet.push(haveSpecialBathroom) - 1;
+
+    // Dependências adaptada para pessoas com deficiências
+    req.queryIndex.allAdaptedBuilding = req.queryIndex.allSecretary;
+
+    let haveAdaptedBuilding = allSecretary.clone();
+    haveAdaptedBuilding.where('escola.dependencias_pne = 1');
+    req.queryIndex.haveAdaptedBuilding = req.querySet.push(haveAdaptedBuilding) - 1;
+
+    next();
+}, multiQuery, (req, res, next) => {
+    // Faz o matching entre os resultados
+    let school_place = matchQueries(req.result[req.queryIndex.allSchools], req.result[req.queryIndex.schoolPlace]);
+    let libraries = matchQueries(req.result[req.queryIndex.allLibraries], req.result[req.queryIndex.haveLibraries]);
+    let libraries_reading_room = matchQueries(req.result[req.queryIndex.allLibrariesReadingRoom], req.result[req.queryIndex.haveLibrariesReadingRoom]);
+    let computer_lab = matchQueries(req.result[req.queryIndex.allInfLab], req.result[req.queryIndex.haveInfLab]);
+    let science_lab = matchQueries(req.result[req.queryIndex.allScienceLab], req.result[req.queryIndex.haveScienceLab]);
+    let kids_park = matchQueries(req.result[req.queryIndex.allKidsPark], req.result[req.queryIndex.haveKidsPark]);
+    let nursery = matchQueries(req.result[req.queryIndex.allCribs], req.result[req.queryIndex.haveCribs]);
+    let sports_court = matchQueries(req.result[req.queryIndex.allSportsCourt], req.result[req.queryIndex.haveSportsCourt]);
+    let covered_sports_court = matchQueries(req.result[req.queryIndex.allCoveredSportsCourt], req.result[req.queryIndex.haveCoveredSportsCourt]);
+    let uncovered_sports_court = matchQueries(req.result[req.queryIndex.allUncoveredSportsCourt], req.result[req.queryIndex.haveUncoveredSportsCourt]);
+    let director_room = matchQueries(req.result[req.queryIndex.allDirectorRoom], req.result[req.queryIndex.haveDirectorRoom]);
+    let secretary = matchQueries(req.result[req.queryIndex.allSecretary], req.result[req.queryIndex.haveSecretary]);
+    let teacher_room = matchQueries(req.result[req.queryIndex.allTeacherRoom], req.result[req.queryIndex.haveTeacherRoom]);
+    let kitchen = matchQueries(req.result[req.queryIndex.allKitchen], req.result[req.queryIndex.haveKitchen]);
+    let storeroom = matchQueries(req.result[req.queryIndex.allStoreroom], req.result[req.queryIndex.haveStoreroom]);
+    let warehouse = matchQueries(req.result[req.queryIndex.allWarehouse], req.result[req.queryIndex.haveWarehouse]);
+    let internet = matchQueries(req.result[req.queryIndex.allInternet], req.result[req.queryIndex.haveInternet]);
+    let broadband_internet = matchQueries(req.result[req.queryIndex.allBroadbandInternet], req.result[req.queryIndex.haveBroadbandInternet]);
+    let inside_bathroom = matchQueries(req.result[req.queryIndex.allInsideBathroom], req.result[req.queryIndex.haveInsideBathroom]);
+    let inside_kids_bathroom = matchQueries(req.result[req.queryIndex.allInsideKidsBathroom], req.result[req.queryIndex.haveInsideKidsBathroom]);
+    let eletric_energy = matchQueries(req.result[req.queryIndex.allEletricEnergy], req.result[req.queryIndex.haveEletricEnergy]);
+    let water_supply = matchQueries(req.result[req.queryIndex.allWaterSupply], req.result[req.queryIndex.haveWaterSupply]);
+    let filtered_water = matchQueries(req.result[req.queryIndex.allFilteredWater], req.result[req.queryIndex.haveFilteredWater]);
+    let sewage_treatment = matchQueries(req.result[req.queryIndex.allSewage], req.result[req.queryIndex.haveSewage]);
+    let special_multifunction_room = matchQueries(req.result[req.queryIndex.allMultifunctionRoom], req.result[req.queryIndex.haveMultifunctionRoom]);
+    let special_bathroom = matchQueries(req.result[req.queryIndex.allSpecialBathroom], req.result[req.queryIndex.haveSpecialBathroom]);
+    let adapted_building = matchQueries(req.result[req.queryIndex.allAdaptedBuilding], req.result[req.queryIndex.haveAdaptedBuilding]);
+
+    req.result = [{
+        school_place,
+        libraries,
+        libraries_reading_room,
+        computer_lab,
+        science_lab,
+        kids_park,
+        nursery,
+        sports_court,
+        covered_sports_court,
+        uncovered_sports_court,
+        director_room,
+        secretary,
+        teacher_room,
+        kitchen,
+        storeroom,
+        warehouse,
+        internet,
+        broadband_internet,
+        inside_bathroom,
+        inside_kids_bathroom,
+        eletric_energy,
+        water_supply,
+        filtered_water,
+        sewage_treatment,
+        special_multifunction_room,
+        special_bathroom,
+        adapted_building
+    }];
+
+    next();
+}, id2str.multitransform(false), response('infrastructure'));
+
+module.exports = infrastructureApp;
diff --git a/src/libs/routes_v2/liquidEnrollmentRatio.js b/src/libs/routes_v2/liquidEnrollmentRatio.js
new file mode 100644
index 00000000..63bd9168
--- /dev/null
+++ b/src/libs/routes_v2/liquidEnrollmentRatio.js
@@ -0,0 +1,434 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const liquidEnrollmentRatioApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+liquidEnrollmentRatioApp.use(cache('15 day'));
+
+// Complete range of the enrollments dataset.
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+liquidEnrollmentRatioApp.get('/year_range', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('DISTINCT pnad.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('matricula')
+    .field('DISTINCT matricula.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let distinct_years = [];
+    let new_result = [];
+    for (let i = 0; i < req.oldResult.length; i++) {
+        for (let j = 0; j < req.result.length; j++) {
+            if(req.oldResult[i].year == req.result[j].year) {
+                distinct_years.push(req.oldResult[i]);
+            }
+        }
+    }
+    new_result.push({start_year: distinct_years[distinct_years.length -1].year, end_year: distinct_years[0].year});
+    req.result = new_result;
+    next();
+}, response('range'));
+
+liquidEnrollmentRatioApp.get('/years', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('DISTINCT pnad.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('matricula')
+    .field('DISTINCT matricula.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let new_result = []
+    for (let i = 0; i < req.oldResult.length; i++) {
+        for (let j = 0; j < req.result.length; j++) {
+            if(req.oldResult[i].year == req.result[j].year) {
+                new_result.push(req.oldResult[i]);
+            }
+        }
+    }
+    req.result = new_result;
+    next();
+}, response('years'));
+
+liquidEnrollmentRatioApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'pnad\'');
+    next();
+}, query, response('source'));
+
+liquidEnrollmentRatioApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Creche'},
+        {id: 2, name: 'Pré-Escola'},
+        {id: 3, name: 'Ensino Fundamental - anos iniciais'},
+        {id: 4, name: 'Ensino Fundamental - anos finais'},
+        {id: 5, name: 'Ensino Médio'}
+    ];
+    next();
+}, response('education_level_short'));
+
+liquidEnrollmentRatioApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+liquidEnrollmentRatioApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [
+        {id: 0, name: 'Sem declaração'},
+        {id: 1, name: 'Branca'},
+        {id: 2, name: 'Preta'},
+        {id: 3, name: 'Parda'},
+        {id: 4, name: 'Amarela'},
+        {id: 5, name: 'Indígena'}
+    ];
+    next();
+}, response('ethnic_group'));
+
+liquidEnrollmentRatioApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: '@',
+    tableField: 'cor_raca_id',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca_id'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'gender',
+    table: '@',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'location',
+    table: '@',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'matricula',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+});
+
+function matchQueries(queryTotal, queryPartial) {
+    let match = [];
+    queryPartial.forEach((result) => {
+        let newObj = {};
+        let keys = Object.keys(result);
+        keys.forEach((key) => {
+            newObj[key] = result[key];
+        });
+        // console.log('NEW OBJ');
+        // console.log(newObj);
+        // remove total
+        let index = keys.indexOf('total');
+        if(index > -1) keys.splice(index, 1);
+        // remove education_level_short_id
+        index = keys.indexOf('education_level_short_id');
+        if(index > -1) keys.splice(index, 1);
+        // remove education_level_short_name
+        index = keys.indexOf('education_level_short_name');
+        if(index > -1) keys.splice(index, 1);
+        let objMatch = null;
+
+        for(let i = 0; i < queryTotal.length; ++i) {
+            let total = queryTotal[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(total[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                objMatch = total;
+                break;
+            }
+        }
+
+        if(objMatch) {
+            // console.log('MATCH!!!!');
+            // console.log(objMatch);
+            newObj.total = (result.total / objMatch.total) * 100;
+            newObj.partial = result.total;
+            newObj.denominator = objMatch.total
+            match.push(newObj);
+        }
+    });
+    // console.log('TAMANHOS');
+    // console.log(queryTotal.length);
+    // console.log(queryPartial.length);
+    // console.log(match.length);
+    return match;
+}
+
+function ConvertEnrollment(result) {
+    if (result == 1) {
+        return '(matricula.faixa_etaria_31_03 = 1 AND matricula.etapa_resumida = 1)'
+    } else if (result == 2) {
+        return '(matricula.faixa_etaria_31_03 = 2 AND matricula.etapa_resumida = 2)'
+    } else if (result == 4) {
+        return '(matricula.faixa_etaria_31_03 = 3 AND matricula.etapa_resumida = 3)'
+    } else if (result == 5) {
+        return '(matricula.faixa_etaria_31_03 = 4 AND matricula.etapa_resumida = 4)'
+    } else if (result == 6) {
+        return '(matricula.faixa_etaria_31_03 = 5 AND matricula.etapa_resumida = 5)'
+    }
+}
+
+
+function convertPnad(result) {
+    if (result == 1) {
+        return 'pnad.faixa_etaria_31_03 = 1'
+    } else if (result == 2) {
+        return 'pnad.faixa_etaria_31_03 = 2'
+    } else if (result == 4) {
+        return 'pnad.faixa_etaria_31_03 = 3'
+    } else if (result == 5) {
+        return 'pnad.faixa_etaria_31_03 = 4'
+    } else if (result == 6) {
+        return 'pnad.faixa_etaria_31_03 = 5'
+    }
+}
+
+liquidEnrollmentRatioApp.get('/', rqf.parse(),(req, res, next) => {
+    req.numerator = {};
+    req.denominator = {};
+    let liquidEnrollmentRatioApp = {};
+
+    req.sql.from('matricula')
+    .field('count(*)', 'total')
+    .field('matricula.ano_censo', 'year')
+    .group('matricula.ano_censo')
+    .order('matricula.ano_censo')
+    .where('matricula.tipo <= 3')
+
+    if ("education_level_short" in req.filter) {
+
+        if (Array.isArray(req.filter.education_level_short)) {
+            var stringQuery = '';
+            for(let i = 0; i < req.filter.education_level_short.length - 1; i++)  {
+                stringQuery = stringQuery + ConvertEnrollment(req.filter.education_level_short[i]) + ' OR ';
+            }
+
+            stringQuery = stringQuery + ConvertEnrollment(req.filter.education_level_short[req.filter.education_level_short.length - 1]);
+            delete req.filter.education_level_short;
+            req.sql.where(stringQuery);
+            req.sql.field('matricula.faixa_etaria_31_03', 'age_range')
+            req.sql.group('matricula.faixa_etaria_31_03', 'age_range');
+        }
+
+    } else if ( "education_level_short" in req.dims ) {
+
+        req.sql.field('matricula.faixa_etaria_31_03', 'age_range')
+        req.sql.where('(matricula.etapa_resumida = 1  AND matricula.faixa_etaria_31_03 = 1) OR (matricula.etapa_resumida = 2  AND matricula.faixa_etaria_31_03 = 2) OR (matricula.etapa_resumida = 3  AND matricula.faixa_etaria_31_03 = 3) OR (matricula.etapa_resumida = 4  AND matricula.faixa_etaria_31_03 = 4) OR (matricula.etapa_resumida = 5  AND matricula.faixa_etaria_31_03 = 5)');
+        req.sql.group('matricula.faixa_etaria_31_03', 'age_range');
+
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+
+    next();
+}, rqf.build(), query, id2str.transform(),  (req, res, next) => {
+    req.numerator = req.result;
+    req.resetSql();
+    req.sql.field('sum(peso)', 'total')
+    .field('pnad.ano_censo','year')
+    .from('pnad')
+    .group('pnad.ano_censo')
+    .order('pnad.ano_censo')
+
+    //remove education_level_short how filter and add faixa_etaria_31_03 in filter
+
+    if ("education_level_short" in req.filter) {
+
+        if (Array.isArray(req.filter.education_level_short)) {
+            var stringQuery = '';
+            for(let i = 0; i < req.filter.education_level_short.length - 1; i++)  {
+                stringQuery = stringQuery + convertPnad(req.filter.education_level_short[i]) + ' OR ';
+            }
+            stringQuery = stringQuery + convertPnad(req.filter.education_level_short[req.filter.education_level_short.length - 1]);
+            req.sql.where(stringQuery);
+        }
+        req.sql.field('pnad.faixa_etaria_31_03', 'age_range')
+        req.sql.group('pnad.faixa_etaria_31_03', 'age_range');
+
+    } else if ( "education_level_short" in req.dims ) {
+
+        req.sql.field('pnad.faixa_etaria_31_03','age_range')
+        req.sql.where('pnad.faixa_etaria_31_03 = 1 OR pnad.faixa_etaria_31_03 = 2 OR pnad.faixa_etaria_31_03 = 3 OR pnad.faixa_etaria_31_03 = 4 OR pnad.faixa_etaria_31_03 = 5')
+        req.sql.group('pnad.faixa_etaria_31_03', 'age_range');
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+
+    next();
+}, rqf.parse(), (req, res, next) => {
+    if ("education_level_short" in req.filter) {
+        delete req.filter.education_level_short;
+    }
+    if ("education_level_short" in req.dims) {
+        delete req.dims.education_level_short;
+    }
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.denominator = req.result;
+
+    //division to generate req.result final
+    req.result = []
+    let liquidEnrollment = matchQueries(req.denominator, req.numerator);
+    req.result = liquidEnrollment;
+    next();
+}, response('liquidEnrollmentRatio'));
+
+module.exports = liquidEnrollmentRatioApp;
diff --git a/src/libs/routes_v2/location.js b/src/libs/routes_v2/location.js
new file mode 100644
index 00000000..d493c452
--- /dev/null
+++ b/src/libs/routes_v2/location.js
@@ -0,0 +1,135 @@
+const express = require('express');
+
+const locationApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const passport = require('passport');
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+locationApp.use(cache('15 day'))
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'id',
+    table: 'localizacao_escolas',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'dims');
+
+locationApp.get('/school', rqf.parse(), (req, res, next) => {
+    req.dims.city=true;
+    req.dims.mesoregion=true;
+    req.dims.microregion=true;
+
+    req.sql.from('localizacao_escolas')
+        .field('localizacao_escolas.nome', 'name')
+        .field('localizacao_escolas.id', 'id')
+        .field('localizacao_escolas.latitude', 'latitude')
+        .field('localizacao_escolas.longitude', 'longitude')
+        .group('localizacao_escolas.nome')
+        .group('localizacao_escolas.id')
+        .group('localizacao_escolas.latitude')
+        .group('localizacao_escolas.longitude');
+    next();
+}, rqf.build(), query, id2str.transform(), response('location'));
+
+locationApp.get('/campi', rqf.parse(), (req, res, next) => {
+    req.dims.city=true;
+    req.dims.mesoregion=true;
+    req.dims.microregion=true;
+
+    req.sql.from('localizacao_campi')
+        .field('localizacao_campi.nome', 'name')
+        .field('localizacao_campi.id', 'id')
+        .field('localizacao_campi.ies_id', 'ies_id')
+        .field('localizacao_campi.latitude', 'latitude')
+        .field('localizacao_campi.longitude', 'longitude')
+        .group('localizacao_campi.nome')
+        .group('localizacao_campi.id')
+        .group('localizacao_campi.ies_id')
+        .group('localizacao_campi.latitude')
+        .group('localizacao_campi.longitude');
+    next();
+}, rqf.build(), query, id2str.transform(), response('location'));
+
+module.exports = locationApp;
\ No newline at end of file
diff --git a/src/libs/routes_v2/mesoregion.js b/src/libs/routes_v2/mesoregion.js
new file mode 100644
index 00000000..001976f2
--- /dev/null
+++ b/src/libs/routes_v2/mesoregion.js
@@ -0,0 +1,86 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const mesoregionApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+mesoregionApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'municipio'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: 'mesorregiao_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id'
+    }
+});
+
+mesoregionApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.from('municipio')
+        .field('municipio.nome_mesorregiao', 'name')
+        .field('municipio.mesorregiao_id', 'id')
+        .field('municipio.estado_id', 'state_id')
+        .group('municipio.nome_mesorregiao')
+        .group('municipio.mesorregiao_id')
+        .group('municipio.estado_id')
+        .order('municipio.mesorregiao_id');
+    next();
+}, query, response('mesoregion'));
+
+module.exports = mesoregionApp;
diff --git a/src/libs/routes_v2/message.js b/src/libs/routes_v2/message.js
new file mode 100644
index 00000000..f9e0e330
--- /dev/null
+++ b/src/libs/routes_v2/message.js
@@ -0,0 +1,55 @@
+/*
+Copyright (C) 2021 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const messageApp = express.Router();
+
+const email = require(`../middlewares/email`);
+
+const log = require(`../log`)(module);
+
+messageApp.post('/', (req, res, next) => {
+    var reqName = req.body.name
+    var reqEmail = req.body.email
+    var reqContents = req.body.contents
+    var reqOrigin = req.body.origin ? req.body.origin : "";
+
+    var sub = "Contato " + reqOrigin
+    let mailOptions = {
+	to: ["dadoseducacionais@ufpr.br", reqEmail], 
+	from: `\"${reqName}\" <dadoseducacionais@ufpr.br>`,
+	text: reqContents,
+        subject: sub
+    }
+
+    email(mailOptions, (err, info) => {
+        if(err) {
+            log.error(err);
+            console.log(err);
+            res.status(500).json({msg: 'Undelivered Contact Mail'});
+        } else {
+            log.info(`Message ${info.messageId} sent: ${info.response}`);
+            res.json({msg: 'Contact Mail Successfully Delivered'});
+        }
+    });
+})
+
+module.exports = messageApp;
\ No newline at end of file
diff --git a/src/libs/routes_v2/microregion.js b/src/libs/routes_v2/microregion.js
new file mode 100644
index 00000000..c4aba0fe
--- /dev/null
+++ b/src/libs/routes_v2/microregion.js
@@ -0,0 +1,99 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const microregionApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+microregionApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'municipio'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: 'mesorregiao_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: 'microrregiao_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id'
+    }
+});
+
+microregionApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.from('municipio')
+        .field('municipio.nome_microrregiao', 'name')
+        .field('municipio.microrregiao_id', 'id')
+        .field('municipio.nome_mesorregiao', 'mesoregion_name')
+        .field('municipio.mesorregiao_id', 'mesoregion_id')
+        .field('municipio.estado_id', 'state_id')
+        .group('municipio.nome_microrregiao')
+        .group('municipio.microrregiao_id')
+        .group('municipio.nome_mesorregiao')
+        .group('municipio.mesorregiao_id')
+        .group('municipio.estado_id')
+        .order('municipio.microrregiao_id');
+    next();
+}, query, response('microregion'));
+
+module.exports = microregionApp;
diff --git a/src/libs/routes_v2/outOfSchool.js b/src/libs/routes_v2/outOfSchool.js
new file mode 100644
index 00000000..fcd57867
--- /dev/null
+++ b/src/libs/routes_v2/outOfSchool.js
@@ -0,0 +1,377 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const outOfSchoolApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+outOfSchoolApp.use(cache('15 day'));
+
+outOfSchoolApp.get('/year_range', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('MIN(pnad.ano_censo)', 'start_year')
+    .field('MAX(pnad.ano_censo)', 'end_year')
+    .where('pnad.ano_censo >= 2007 AND pnad.ano_censo <= 2015');
+    next();
+}, query, response('range'));
+
+outOfSchoolApp.get('/years', (req, res, next) => {
+    req.sql.from('pnad').
+    field('DISTINCT pnad.ano_censo', 'year')
+    .where('pnad.ano_censo >= 2007 AND pnad.ano_censo <= 2015');
+    next();
+}, query, response('years'));
+
+outOfSchoolApp.get('/full_age_range', (req, res, next) => {
+    req.result = [
+        {id: 1, name: '0 a 3 anos'},
+        {id: 2, name: '4 a 5 anos'},
+        {id: 3, name: '6 a 10 anos'},
+        {id: 4, name: '11 a 14 anos'},
+        {id: 5, name: '15 a 17 anos'},
+        {id: 6, name: '18 a 24 anos'},
+        {id: 7, name: '25 a 29 anos'},
+        {id: 8, name: '30 a 40 anos'},
+        {id: 9, name: '41 a 50 anos'},
+        {id: 10, name: '51 a 64 anos'},
+        {id: 11, name: 'Mais de 64 anos'}
+    ];
+    next();
+}, response('full_age_range'));
+
+outOfSchoolApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [
+        {id: 0, name: 'Sem declaração'},
+        {id: 1, name: 'Branca'},
+        {id: 2, name: 'Preta'},
+        {id: 3, name: 'Parda'},
+        {id: 4, name: 'Amarela'},
+        {id: 5, name: 'Indígena'}
+    ];
+    next();
+}, response('ethnic_group'));
+
+outOfSchoolApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+outOfSchoolApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+outOfSchoolApp.get('/fifth_household_income', (req, res, next) => {
+    req.result = [
+		{id: 1, name: '20% menores'},
+        {id: 2, name: '2o quinto'},
+        {id: 3, name: '3o quinto'},
+        {id: 4, name: '4o quinto'},
+		{id: 5, name: '20% maiores'},
+        {id: -1, name: 'Sem declaração'}
+	];
+    next();
+},response('fifth_household_income'));
+
+outOfSchoolApp.get('/extremes_household_income', (req, res, next) => {
+    req.result = [
+		{id: 1, name: '10% menores'},
+        {id: 2, name: '10% maiores'},
+        {id: -1, name: 'Sem declaração'}
+	];
+    next();
+}, response('extremes_household_income'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'pnad'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'pnad'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: 'pnad',
+    tableField: 'cor_raca_id',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca_id'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'pnad',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'pnad',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'pnad',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'pnad',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'full_age_range',
+    table: 'pnad',
+    tableField: 'faixa_etaria_31_03',
+    resultField: 'full_age_range_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'faixa_etaria_31_03'
+    }
+}).addValue({
+    name: 'gender',
+    table: 'pnad',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'location',
+    table: 'pnad',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'extremes_household_income',
+    table: 'pnad',
+    tableField: 'extremos_nivel_rendimento',
+    resultField: 'extremes_household_income_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'extremos_nivel_rendimento'
+    }
+}).addValue({
+    name: 'fifth_household_income',
+    table: 'pnad',
+    tableField: 'quintil_nivel_rendimento',
+    resultField: 'fifth_household_income_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'quintil_nivel_rendimento'
+    }
+});
+
+outOfSchoolApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.from('pnad')
+    .field('SUM(pnad.peso)', 'total')
+    .field('pnad.ano_censo', 'year')
+    .where('pnad.escolaridade_familiar >= 1 AND pnad.escolaridade_familiar <= 4 AND pnad.frequenta_escola_creche = 4')
+    .group('pnad.ano_censo')
+    .order('pnad.ano_censo');
+
+    next();
+}, query, addMissing(rqf), id2str.transform(), response('out_of_school'));
+
+// Versão para o SimCAQ
+let simcaqRqf = new ReqQueryFields();
+
+simcaqRqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'brazil_total',
+    table: 'populacao_fora_da_escola',
+    tableField: 'brasil',
+    resultField: 'brazil_total_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'brasil'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: 'nome',
+    resultField: 'state_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'populacao_fora_da_escola'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'populacao_fora_da_escola'
+    }
+}).addValue({
+    name: 'pfe',
+    table: 'populacao_fora_da_escola',
+    tableField: 'codigo_pfe',
+    resultField: 'pfe_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'codigo_pfe'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'populacao_fora_da_escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'populacao_fora_da_escola',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'populacao_fora_da_escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'populacao_fora_da_escola',
+        field: 'ano_censo'
+    }
+});
+
+outOfSchoolApp.get('/simcaq', simcaqRqf.parse(), (req, res, next) => {
+    if ('state' in req.filter || 'city' in req.filter || 'state' in req.dims || 'city' in req.dims) { // Query in state/city level
+
+        if ('city' in req.filter && 'state' in req.filter) delete req.filter.state   // use only the city filter because of the table particularities
+
+        req.sql.from('populacao_fora_da_escola')
+        .field('SUM(populacao_fora_da_escola.pop_fora_escola)', 'total')
+        .field("'Brasil'", 'name')
+        .field('populacao_fora_da_escola.ano_censo')
+        .group('populacao_fora_da_escola.ano_censo')
+        .order('populacao_fora_da_escola.ano_censo');
+    } else { // Query in 'Brasil' level
+        req.sql.from('populacao_fora_da_escola')
+        .field('SUM(populacao_fora_da_escola.pop_fora_escola)', 'total')
+        .field("'Brasil'", 'name')
+        .field('populacao_fora_da_escola.ano_censo')
+        .where('populacao_fora_da_escola.brasil = 1')
+        .group('populacao_fora_da_escola.ano_censo')
+        .order('populacao_fora_da_escola.ano_censo'); 
+    }
+
+    next();
+}, simcaqRqf.build(), query, (req, res, next) => {
+	req.result.forEach((i) => {
+		i.total = parseInt(i.total);
+	});
+	next();
+}, addMissing(simcaqRqf), id2str.transform(), response('out_of_school'));
+
+module.exports = outOfSchoolApp;
diff --git a/src/libs/routes_v2/pibpercapita.js b/src/libs/routes_v2/pibpercapita.js
new file mode 100644
index 00000000..c5c77a92
--- /dev/null
+++ b/src/libs/routes_v2/pibpercapita.js
@@ -0,0 +1,250 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const pibpercapitaApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+pibpercapitaApp.use(cache('15 day'));
+
+pibpercapitaApp.get('/year_range', (req, res, next) => {
+    req.sql.from('ibge_pib')
+    .field('MIN(ibge_pib.ano_censo)', 'start_year')
+    .field('MAX(ibge_pib.ano_censo)', 'end_year')
+    .where('ibge_pib.ano_censo > 2013');
+    next();
+}, query, response('range'));
+
+pibpercapitaApp.get('/years', (req, res, next) => {
+    req.sql.from('ibge_pib').
+    field('DISTINCT ibge_pib.ano_censo', 'year')
+    .where('ibge_pib.ano_censo > 2013');
+    next();
+}, query, response('years'));
+
+pibpercapitaApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'ibge_pib\'');
+    next();
+}, query, response('source'));
+
+pibpercapitaApp.get('/income_level', (req, res, next) => {
+    req.result = [
+        {id: 1, name: "1º quintil – 20% menores"},
+        {id: 2, name: "2º quintil"},
+        {id: 3, name: "3º quintil"},
+        {id: 4, name: "4º quintil"},
+        {id: 5, name: "5º quintil – 20% maiores"},
+    ];
+    next();
+}, response('income_level'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'ibge_pib'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'ibge_pib'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+      relation: '=',
+      type: 'integer',
+      field: 'regiao_id',
+      table: 'ibge_pib'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'ibge_pib'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'income_level',
+    table: 'ibge_pib',
+    tableField: 'nivel_renda_per_capita',
+    resultField: 'income_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'nivel_renda_per_capita'
+    }
+}).addValue({
+    name: 'income_level_brasil',
+    table: 'ibge_pib',
+    tableField: 'nivel_renda_brasil',
+    resultField: 'income_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'nivel_renda_brasil'
+    }
+}).addValue({
+    name: 'income_level_uf',
+    table: 'ibge_pib',
+    tableField: 'nivel_renda_uf',
+    resultField: 'income_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'nivel_renda_uf'
+    }
+});;
+
+pibpercapitaApp.get('/', rqf.parse(), (req, res, next) => {
+    if ('income_level' in req.dims) {   // Retorna os quintis
+        delete req.dims.income_level
+
+        req.sql.from('pib_quintis')
+        .field('pib_quintis.valor', 'total')
+        .field('pib_quintis.tipo', 'income_level_id')
+        .field('pib_quintis.ano_censo', 'year')
+        .group('pib_quintis.ano_censo')
+        .group('pib_quintis.valor')
+        .group('pib_quintis.tipo')
+        .order('pib_quintis.tipo')
+
+        if (!('state' in req.filter)) {
+            req.sql.where('pib_quintis.estado_id = 0')
+        } 
+        if ('city' in req.filter) {
+            req.sql.join('ibge_pib', null, 'ibge_pib.nivel_renda_brasil=pib_quintis.tipo AND ibge_pib.ano_censo=pib_quintis.ano_censo')
+        }
+    }
+
+    else if (("city" in req.dims) || ("city" in req.filter)) {
+      req.sql.from('ibge_pib')
+      .field('ibge_pib.pib_per_capita', 'total')
+      .field('ibge_pib.ano_censo', 'year')
+      .group('ibge_pib.ano_censo')
+      .group('ibge_pib.pib_per_capita')
+      .order('ibge_pib.ano_censo')
+  } else  {
+      req.sql.from('ibge_pib')
+      .field('SUM(ibge_pib.pib)/SUM(ibge_pib.populacao)', 'total')
+      .field('ibge_pib.ano_censo', 'year')
+      .group('ibge_pib.ano_censo')
+      .order('ibge_pib.ano_censo')
+   }
+   next();
+}, rqf.build(), query, (req, res, next) => {
+     req.result.forEach((i) => {
+        let value = parseFloat(i.total);
+        let isnum = /^\d+$/.test(value);
+        if (isnum == true) {
+            value = value.toFixed(2)
+        }
+        // console.log(i.total);
+
+        let res = value.toString().split(".");
+        //rounding decimal.
+        let decimal = Math.round(res[1].toString().substring(0,2) + (".") + res[1].toString().substring(2,3));
+        //case 0 after comma
+        if (res[1].toString().substring(0,1) == 0) {
+            i.total = parseFloat(res[0] + "." + "0" + decimal);
+        } else {
+            i.total = parseFloat(res[0] + "." +  decimal);
+        }
+        // console.log(i.total);
+     });
+     next();
+ }, id2str.transform(false), response("pibpercapita"));
+
+
+module.exports = pibpercapitaApp;
diff --git a/src/libs/routes_v2/population.js b/src/libs/routes_v2/population.js
new file mode 100644
index 00000000..fca3b0d6
--- /dev/null
+++ b/src/libs/routes_v2/population.js
@@ -0,0 +1,163 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const populationApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+populationApp.use(cache('15 day'));
+
+populationApp.get('/year_range', (req, res, next) => {
+    req.sql.from('ibge_pib')
+    .field('MIN(ibge_pib.ano_censo)', 'start_year')
+    .field('MAX(ibge_pib.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+populationApp.get('/years', (req, res, next) => {
+    req.sql.from('ibge_pib').
+    field('DISTINCT ibge_pib.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+// populationApp.get('/city_size', (req, res, next) => {
+//     req.result = [
+//         {id: 1, name: "0 - 5000"},
+//         {id: 2, name: "5001 - 10000"},
+//         {id: 3, name: "10001 - 20000"},
+//         {id: 4, name: "20001 - 50000"},
+//         {id: 5, name: "50001 - 100000"},
+//         {id: 6, name: "100001 - 500000"},
+//         {id: 7, name: "mais que 500000"}
+//     ];
+//     next();
+// }, response('city_size'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'ibge_pib'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'ibge_pib'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'ibge_pib'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'ibge_pib'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'regiao_id',
+        table: 'ibge_pib'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'ibge_pib'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'ibge_pib',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'ibge_pib',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+});
+
+populationApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+  req.sql.from('ibge_pib')
+    .field('SUM(ibge_pib.populacao)', 'total')
+    .field('ibge_pib.ano_censo', 'year')
+    .group('ibge_pib.ano_censo')
+    .order('ibge_pib.ano_censo')
+
+   next();
+}, query, addMissing(rqf), id2str.transform(false), response('population'));
+
+module.exports = populationApp;
diff --git a/src/libs/routes_v2/portalMec.js b/src/libs/routes_v2/portalMec.js
new file mode 100644
index 00000000..e0e5353e
--- /dev/null
+++ b/src/libs/routes_v2/portalMec.js
@@ -0,0 +1,136 @@
+const express = require('express');
+
+const portalMecApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'day',
+    table: 'docente',
+    tableField: 'nasc_dia',
+    resultField: 'born_day_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'docente',
+        field: 'nasc_dia'
+    }
+}).addValue({
+    name: 'month',
+    table: 'docente',
+    tableField: 'nasc_mes_id',
+    resultField: 'born_month',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'docente',
+        field: 'nasc_mes'
+    }
+}).addValue({
+    name: 'year',
+    table: 'docente',
+    tableField: 'nasc_ano_id',
+    resultField: 'born_year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'docente',
+        field: 'nasc_ano'
+    }
+}).addValue({
+    name: 'teacher',
+    table: 'docente',
+    tableField: 'id',
+    resultField: 'teacher_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'docente',
+        field: 'id'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'docente',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'docente',
+        field: 'ano_censo'
+    }
+});
+
+portalMecApp.get('/', rqf.parse(), (req, res, next) => {
+
+    req.sql.field('docente.id_docente')
+    .field('docente.ano_censo', 'year')
+    .field('docente.cod_quimica', 'Química')
+    .field('docente.cod_fisica', 'Física')
+    .field('docente.cod_matematica', 'Matemática')
+    .field('docente.cod_biologia', 'Biologia')
+    .field('docente.cod_ciencias', 'Ciências')
+    .field('docente.cod_literat_port', 'Língua/ Literatura Portuguesa')
+    .field('docente.cod_literat_ing', 'Língua/ Literatura estrangeira - Inglês')
+    .field('docente.cod_literat_esp', 'Língua/ Literatura estrangeira - Espanhol')
+    .field('docente.cod_literat_frances', 'Língua/ Literatura estrangeira - Francês')
+    .field('docente.literat_outra', 'Língua/ Literatura estrangeira - Outra')
+    .field('docente.cod_literat_indigena', 'Língua/ Literatura estrangeira - Língua Indígena')
+    .field('docente.cod_artes', 'Artes (Educação Artística, Teatro, Dança, Música, Artes Plásticas e outras)')
+    .field('docente.cod_ed_fisica', 'Educação Física')
+    .field('docente.cod_hist', 'História')
+    .field('docente.cod_geo', 'Geografia')
+    .field('docente.cod_filos', 'Filosofia')
+    .field('docente.cod_ensino_religioso', 'Ensino Religioso')
+    .field('docente.cod_estudos_sociais', 'Estudos Sociais')
+    .field('docente.cod_sociologia', 'Sociologia')
+    .field('docente.cod_inf_comp', 'Informática/ Computação')
+    .field('docente.cod_profissionalizante', 'Disciplinas profissionalizantes')
+    .field('docente.cod_disc_atendimento_especiais', 'Disciplinas voltadas ao atendimento às necessidades educacionais específicas dos alunos que são público alvo da educação especial e às práticas educacionais inclusivas')
+    .field('docente.cod_disc_diversidade_socio_cult', 'Disciplinas voltadas à diversidade sociocultural (Disciplinas pedagógicas)')
+    .field('docente.cod_libras', 'Libras')
+    .field('docente.cod_disciplina_pedag', 'Disciplinas pedagógicas')
+    .field('docente.cod_outras_disciplina', 'Outras disciplinas')
+    .from('docente')
+    next();
+
+}, rqf.build(), query, response('portalMec'));
+
+module.exports = portalMecApp;
diff --git a/src/libs/routes_v2/portalMecInep.js b/src/libs/routes_v2/portalMecInep.js
new file mode 100644
index 00000000..2b235891
--- /dev/null
+++ b/src/libs/routes_v2/portalMecInep.js
@@ -0,0 +1,60 @@
+const express = require('express');
+
+const portalMecInepApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'school_cod',
+    table: 'escola',
+    tableField: 'id',
+    resultField: 'school_cod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'escola',
+        field: 'id'
+    }
+});
+
+portalMecInepApp.get('/', rqf.parse(), (req, res, next) => {
+
+    req.sql.field('DISTINCT escola.id', 'id')
+    .field('escola.nome_escola', 'name')
+    .from('escola')
+    .join('estado', null, 'estado.id=escola.estado_id')
+    .field('estado.nome', 'state_name')
+    .join('municipio', null, 'municipio.id=escola.municipio_id')
+    .field('municipio.nome', 'city_name')
+
+    next();
+
+}, rqf.build(), query, response('portalMec_inep'));
+
+module.exports = portalMecInepApp;
diff --git a/src/libs/routes_v2/rateSchool.js b/src/libs/routes_v2/rateSchool.js
new file mode 100644
index 00000000..a6a91b7b
--- /dev/null
+++ b/src/libs/routes_v2/rateSchool.js
@@ -0,0 +1,337 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const rateSchoolApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+rateSchoolApp.use(cache('15 day'));
+
+let rqf = new ReqQueryFields();
+
+// Complete range of the enrollments dataset.
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+rateSchoolApp.get('/year_range', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('MIN(pnad.ano_censo)', 'start_year')
+    .field('MAX(pnad.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+rateSchoolApp.get('/years', (req, res, next) => {
+    req.sql.from('pnad')
+    .field('DISTINCT pnad.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+rateSchoolApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'pnad\'');
+    next();
+}, query, response('source'));
+
+rateSchoolApp.get('/ethnic_group_pnad', (req, res, next) => {
+    req.result = [
+        {id: 0, name: 'Indígena'},
+        {id: 1, name: 'Branca e amarela'},
+        {id: 2, name: 'Preta e parda'},
+        {id: 9, name: 'Sem declaração'}
+    ];
+    next();
+}, response('ethnic_group_pnad'));
+
+rateSchoolApp.get('/age_range', (req, res, next) => {
+    req.result = [
+        {id: 1, name: '0 a 3 anos'},
+        {id: 2, name: '4 a 5 anos'},
+        {id: 3, name: '6 a 10 anos'},
+        {id: 4, name: '11 a 14 anos'},
+        {id: 5, name: '15 a 17 anos'},
+        {id: 6, name: '18 a 24 anos'}
+    ];
+    next();
+}, response('age_range'));
+
+rateSchoolApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+rateSchoolApp.get('/location', (req, res, next) => {
+    req.result = [
+		{id: 1, name: 'Urbana'},
+		{id: 2, name: 'Rural'}
+	];
+    next();
+}, response('location'));
+
+rateSchoolApp.get('/fifth_household_income', (req, res, next) => {
+    req.result = [
+		{id: 1, name: '20% menores'},
+        {id: 2, name: '2o quinto'},
+        {id: 3, name: '3o quinto'},
+        {id: 4, name: '4o quinto'},
+		{id: 5, name: '20% maiores'},
+        {id: -1, name: 'Sem declaração'}
+	];
+    next();
+},response('fifth_household_income'));
+
+rateSchoolApp.get('/extremes_household_income', (req, res, next) => {
+    req.result = [
+		{id: 1, name: '10% menores'},
+        {id: 2, name: '10% maiores'},
+        {id: -1, name: 'Sem declaração'}
+	];
+    next();
+}, response('extremes_household_income'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'pnad'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'pnad'
+    }
+}).addValue({
+    name: 'ethnic_group_pnad',
+    table: 'pnad',
+    tableField: 'cor_raca',
+    resultField: 'ethnic_group_pnad_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'pnad',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'pnad',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'pnad',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'pnad',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'age_range',
+    table: 'pnad',
+    tableField: 'faixa_etaria_31_03',
+    resultField: 'age_range_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'faixa_etaria_31_03'
+    }
+}).addValue({
+    name: 'gender',
+    table: 'pnad',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'location',
+    table: 'pnad',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'extremes_household_income',
+    table: 'pnad',
+    tableField: 'extremos_nivel_rendimento',
+    resultField: 'extremes_household_income_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'extremos_nivel_rendimento'
+    }
+}).addValue({
+    name: 'fifth_household_income',
+    table: 'pnad',
+    tableField: 'quintil_nivel_rendimento',
+    resultField: 'fifth_household_income_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'quintil_nivel_rendimento'
+    }
+});
+
+function matchQueries(queryTotal, queryPartial) {
+    let match = [];
+    queryTotal.forEach((result) => {
+        let newObj = {};
+        let keys = Object.keys(result);
+        keys.forEach((key) => {
+            newObj[key] = result[key];
+        });
+        // console.log('NEW OBJ');
+        // console.log(newObj);
+        let index = keys.indexOf('total');
+        if(index > -1) keys.splice(index, 1);
+        let objMatch = null;
+
+        for(let i = 0; i < queryPartial.length; ++i) {
+            let partial = queryPartial[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(partial[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                objMatch = partial;
+                break;
+            }
+        }
+
+        if(objMatch) {
+            // console.log(objMatch);
+            newObj.denominator = result.total;
+            newObj.partial = objMatch.total;
+            newObj.total = (objMatch.total / result.total) * 100;
+            match.push(newObj);
+        }
+    });
+
+    return match;
+}
+
+rateSchoolApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.querySet = [];
+    req.queryIndex = {};
+
+    log.debug(req.sql.toParam());
+    if ("age_range" in req.filter || "age_range" in req.dims) {
+        let freq_total = req.sql.clone();
+        freq_total.field('sum(pnad.peso)', 'total')
+        .field('pnad.ano_censo','year')
+        .from('pnad')
+        .group('pnad.ano_censo')
+        .order('pnad.ano_censo')
+        .where('pnad.faixa_etaria_31_03 < 7')
+        req.queryIndex.freq_total = req.querySet.push(freq_total) - 1;
+
+        let freq_nursery = req.sql.clone();
+        freq_nursery.field('sum(pnad.peso)', 'total')
+        .field('pnad.ano_censo','year')
+        .from('pnad')
+        .group('pnad.ano_censo')
+        .order('pnad.ano_censo')
+        .where('pnad.frequenta_escola_creche = 2')
+        .where('pnad.faixa_etaria_31_03 < 7')
+        req.queryIndex.freq_nursery = req.querySet.push(freq_nursery) - 1;
+    }
+     next();
+}, multiQuery, (req, res, next) => {
+    if ("age_range" in req.filter || "age_range" in req.dims) {
+        log.debug(req.result[req.queryIndex.freq_total]);
+        log.debug(req.result[req.queryIndex.freq_nursery])
+        let school_attendance_rate = matchQueries(req.result[req.queryIndex.freq_total], req.result[req.queryIndex.freq_nursery]);
+        req.result = school_attendance_rate;
+    } else {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Wrong/No filter specified'
+        });
+    }
+    log.debug(req.result)
+    next();
+}, id2str.transform(false), response('rateSchool'));
+
+module.exports = rateSchoolApp;
diff --git a/src/libs/routes_v2/region.js b/src/libs/routes_v2/region.js
new file mode 100644
index 00000000..5df65e72
--- /dev/null
+++ b/src/libs/routes_v2/region.js
@@ -0,0 +1,72 @@
+const express = require('express');
+
+const regionApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+regionApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'id',
+    table: '@',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id',
+        table: '@'
+    }
+}).addValue({
+    name: 'id_not',
+    table: '@',
+    tableField: 'id',
+    where: {
+        relation: '<>',
+        type: 'integer',
+        field: 'id',
+        table: '@'
+    }
+}).addField({
+    name: 'search',
+    field: false,
+    where: true
+}).addValueToField({
+    name: 'name',
+    table: '@',
+    tableField: 'nome',
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome',
+        table: '@'
+    }
+}, 'search');
+
+regionApp.get('/', rqf.parse(), (req, res, next) => {
+    req.sql.from('regiao')
+        .field('id')
+        .field('nome', 'name');
+    next();
+}, rqf.build(), query, response('region'));
+
+module.exports = regionApp;
diff --git a/src/libs/routes_v2/resetToken.js b/src/libs/routes_v2/resetToken.js
new file mode 100644
index 00000000..34ece845
--- /dev/null
+++ b/src/libs/routes_v2/resetToken.js
@@ -0,0 +1,81 @@
+const express = require('express');
+
+const resetTokenApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const ResetToken = require(`${libs}/models/resetToken`);
+
+const User = require(`${libs}/models/user`);
+
+resetTokenApp.get('/:token', (req, res, next) => {
+    let token = req.params.token;
+    ResetToken.findOne({token: token}, (err, rToken) => {
+        if(err) {
+            log.error(err);
+            return next(err);
+        }
+        if(!rToken) {
+            // TODO: generate new reset token
+            res.statusCode = 404;
+            return next({msg: 'Token not found', status:404});
+        }
+        if (rToken.hasExpired()) {
+            res.statusCode = 410;
+            ResetToken.remove({token: token}, (err) => {
+                if(err) {
+                    log.error(err);
+                    next(err);
+                }
+            })
+            return next({msg: 'Token expired', status: 410});
+        }
+        User.findById(rToken.userId, (err, user) => {
+            if(err) {
+                log.error(err);
+                next(err);
+            }
+            let u = user.toObject();
+            delete u.salt;
+            delete u.hashedPassword;
+            res.json({user: u});
+        });
+    });
+});
+resetTokenApp.post('/:token', (req, res, next) => {
+    let token = req.params.token;
+    ResetToken.findOne({token: token}, (err, rToken) => {
+        if(err) {
+            log.error(err);
+            return next(err);
+        }
+        if(!rToken) {
+            res.statusCode = 404;
+            return next({msg: 'Token not found', status:404});
+        }
+        User.findById(rToken.userId, (err, user) => {
+            if(err) {
+                log.error(err);
+                next(err);
+            }
+            user.password = req.body.password;
+            user.save((err) => {
+                if(err) {
+                    log.error(err);
+                    next(err);
+                }
+                ResetToken.remove({token: token}, (err) => {
+                    if(err) {
+                        log.error(err);
+                        next(err);
+                    }
+                })
+                res.json({msg: "Senha alterada com sucesso"});
+            })
+        });
+    });
+})
+
+module.exports = resetTokenApp;
diff --git a/src/libs/routes_v2/school.js b/src/libs/routes_v2/school.js
new file mode 100644
index 00000000..73b765e2
--- /dev/null
+++ b/src/libs/routes_v2/school.js
@@ -0,0 +1,663 @@
+const express = require('express');
+
+const schoolApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const passport = require('passport');
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+let rqfCount = new ReqQueryFields();
+
+// Return location
+schoolApp.get('/year_range', cache('15 day'), (req, res, next) => {
+    req.sql.from('escola')
+    .field('MIN(escola.ano_censo)', 'start_year')
+    .field('MAX(escola.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+schoolApp.get('/years', cache('15 day'), (req, res, next) => {
+    req.sql.from('escola').
+    field('DISTINCT escola.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+schoolApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'escola\'');
+    next();
+}, query, response('source'));
+
+schoolApp.get('/location', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+schoolApp.get('/diff_location', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: 0, name: "Não está em localidade diferenciada"},
+        {id: 1, name: "Área de assentamento"},
+        {id: 2, name: "Terra indígena"},
+        {id: 3, name: "Terra remanescente de quilombos"},
+    ];
+    next();
+}, response('diff_location'));
+
+schoolApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+schoolApp.get('/adm_dependency_detailed', cache('15 day'), (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 8; i++) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+schoolApp.get('/government_agreement', cache('15 day'), (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.governmentAgreement("null")
+    }];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.governmentAgreement(i)
+        });
+    };
+    next();
+}, response('government_agreement'));
+
+schoolApp.get('/agreement', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Municipal'},
+        {id: 2, name: 'Estadual'},
+        {id: 3, name: 'Estadual e Municipal'}
+    ];
+    next();
+}, response('agreement'));
+
+schoolApp.get('/education_day_care_child', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_day_care_child'));
+
+schoolApp.get('/education_preschool_child', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_preschool_child'));
+
+schoolApp.get('/education_begin_elementary_school', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_begin_elementary_school'));
+
+schoolApp.get('/education_end_elementary_school', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_end_elementary_school'));
+
+schoolApp.get('/education_middle_school', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_middle_school'));
+
+schoolApp.get('/education_professional', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_professional'));
+
+schoolApp.get('/education_eja', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não Declarado'},
+        {id: 0, name: 'Não'},
+        {id: 1, name: 'Sim'}
+    ];
+    next();
+}, response('education_eja'));
+
+
+schoolApp.get('/arrangement', cache('15 day'), (req, res, next) => {
+    req.result = [
+        {id: 0, name: 'Creche'},
+        {id: 1, name: 'Pré Escola'},
+        {id: 2, name: 'Ensino Fundamental - AI'},
+        {id: 3, name: 'Ensino Fundamental - AF'},
+        {id: 4, name: 'Ed. Infantil Unificada/Multietapa/Multissérie/Correção fluxo'},
+        {id: 5, name: 'Ensino Médio'},
+        {id: 6, name: 'Ensino EJA'},
+        {id: 7, name: 'Educação Profissional'},
+        {id: 8, name: 'Educação Especial Exclusiva'}
+    ];
+    next();
+}, response('arrangement'));
+
+schoolApp.get('/integral_time', cache('15 day'), (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.integralTime(i)
+        });
+    }
+    next();
+}, response('integral_time'));
+
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'id',
+    table: 'escola',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'ano_censo',
+        table: 'escola'
+    }
+}).addField({
+    name: 'search',
+    field: true,
+    where: true
+}).addValueToField({
+    name: 'city_name',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    dontGroup: true,
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome'
+    },
+    join: {
+      primary: 'id',
+      foreign: 'municipio_id',
+      foreignTable: 'escola'
+    }
+}, 'search')
+.addValueToField({
+    name: 'state_name',
+    table: 'estado',
+    tableField: 'nome',
+    resultField: 'state_name',
+    dontGroup: true,
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'sigla'
+    },
+    join: {
+      primary: 'id',
+      foreign: 'estado_id',
+      foreignTable: 'escola'
+    }
+}, 'search').addValue({
+    name: 'diff_location',
+    table: 'escola',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+});
+
+rqfCount.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'id',
+    table: 'escola',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'filter').addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'ano_censo',
+        table: 'escola'
+    }
+}).addValue({
+    name: 'location',
+    table: 'escola',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: 'escola',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'escola',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'arrangement',
+    table: 'escola',
+    tableField: 'arranjo',
+    resultField: 'arrangement_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'arranjo'
+    }
+}).addValue({
+    name: 'adm_dependency',
+    table: 'escola',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'escola',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'government_agreement',
+    table: 'escola',
+    tableField: 'dependencia_convenio_publico',
+    resultField: 'government_agreement_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_convenio_publico'
+    }
+}).addValue({
+    name: 'integral_time',
+    table: 'escola',
+    tableField: 'tempo_integral',
+    resultField: 'integral_time_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'tempo_integral'
+    }
+}).addValue({
+    name: 'agreement',
+    table: 'escola',
+    tableField: 'tipo_convenio_pp',
+    resultField: 'agreement_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'tipo_convenio_pp'
+    }
+}).addValue({
+    name: 'education_day_care_child',
+    table: 'escola',
+    tableField: 'reg_infantil_creche_t1',
+    resultField: 'education_day_care_child_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'reg_infantil_creche_t1'
+    }
+}).addValue({
+    name: 'education_preschool_child',
+    table: 'escola',
+    tableField: 'reg_infantil_preescola_t1',
+    resultField: 'education_preschool_child_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'reg_infantil_preescola_t1'
+    }
+}).addValue({
+    name: 'education_begin_elementary_school',
+    table: 'escola',
+    tableField: 'reg_fund_ai_t1',
+    resultField: 'education_begin_elementary_school_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'reg_fund_ai_t1'
+    }
+}).addValue({
+    name: 'education_end_elementary_school',
+    table: 'escola',
+    tableField: 'reg_fund_af_t1',
+    resultField: 'education_end_elementary_school_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'reg_fund_af_t1'
+    }
+}).addValue({
+    name: 'education_middle_school',
+    table: 'escola',
+    tableField: 'reg_medio_medio_t1',
+    resultField: 'education_middle_school_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'reg_medio_medio_t1'
+    }
+}).addValue({
+    name: 'education_professional',
+    table: 'escola',
+    tableField: 'educacao_profissional',
+    resultField: 'education_professional_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'educacao_profissional'
+    }
+}).addValue({
+    name: 'education_eja',
+    table: 'escola',
+    tableField: 'ensino_eja',
+    resultField: 'education_eja_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'ensino_eja'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'school_building',
+    table: 'escola',
+    tableField: 'local_func_predio_escolar',
+    resultField: 'school_building',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'local_func_predio_escolar'
+    }
+});
+schoolApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+
+    req.sql.from('escola')
+        .field('escola.id')
+        .field('escola.ano_censo', 'year')
+        .field('escola.nome_escola', 'name')
+        .field('escola.estado_id', 'state_id')
+        .field('escola.municipio_id', 'city_id');
+    next();
+}, query, response('school'));
+
+schoolApp.get('/count', cache('15 day'), rqfCount.parse(), (req, res, next) => {
+	let arrang = ["arranjo_creche", "arranjo_pre", "arranjo_fundamental_ai", "arranjo_fundamental_af", "arranjo_multietapa", "arranjo_ensino_medio", "ensino_eja", "educacao_profissional", "ensino_especial"];
+
+    req.sql.from('escola')
+        .field('COUNT(escola.id)', 'total')
+        .field("'Brasil'", 'name')
+        .field('escola.ano_censo', 'year')
+        .group('escola.ano_censo')
+        .order('escola.ano_censo')
+        .where('escola.situacao_funcionamento_pareada = 1 AND (escola.ensino_regular = 1 OR escola.ensino_eja=1 or escola.educacao_profissional=1)')
+
+	//Transforma a query em OR se tiver o filtro do arranjo
+	if (req.filter.arrangement) {
+		let arrangementQuery = "";
+		for (let i = 0; i < req.filter.arrangement.length - 1; i++) {
+			arrangementQuery += 'escola.' + arrang[req.filter.arrangement[i]] + ' = 1 OR ';
+		}
+		// o ultimo elemento precisa ser sem o OR
+		arrangementQuery += 'escola.' + arrang[req.filter.arrangement[req.filter.arrangement.length - 1]] + ' = 1';
+		req.sql.where('' + arrangementQuery);
+	}
+	delete req.filter.arrangement
+    next();
+}, rqfCount.build(), query, id2str.transform(), addMissing(rqfCount), response('school'));
+
+module.exports = schoolApp;
diff --git a/src/libs/routes_v2/schoolInfrastructure.js b/src/libs/routes_v2/schoolInfrastructure.js
new file mode 100644
index 00000000..c74f0e30
--- /dev/null
+++ b/src/libs/routes_v2/schoolInfrastructure.js
@@ -0,0 +1,741 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const infrastructureApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+infrastructureApp.use(cache('15 day'));
+
+infrastructureApp.get('/year_range', (req, res, next) => {
+    req.sql.from('escola')
+    .field('MIN(escola.ano_censo)', 'start_year')
+    .field('MAX(escola.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+infrastructureApp.get('/years', (req, res, next) => {
+    req.sql.from('escola')
+    .field('DISTINCT escola.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+infrastructureApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'escola\'');
+    next();
+}, query, response('source'));
+
+infrastructureApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+infrastructureApp.get('/rural_location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: "Urbana"},
+        {id: 2, name: "Rural"},
+        {id: 3, name: "Rural - Área de assentamento"},
+        {id: 4, name: "Rural - Terra indígena"},
+        {id: 5, name: "Rural - Área remanescente de quilombos"},
+        {id: 6, name: "Rural - Unidade de uso sustentável"}
+    ];
+    next();
+}, response('rural_location'));
+
+infrastructureApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+infrastructureApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'id',
+    resultField: 'city_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'escola'
+    }
+}, 'filter').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}, 'dims').addValueToField({
+    name: 'state',
+    table: 'estado',
+    tableField: 'id',
+    resultField: 'state_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'escola'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'escola'
+    }
+}, 'filter').addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'escola'
+    }
+}).addValue({
+    name: 'location',
+    table: 'escola',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'escola',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'adm_dependency',
+    table: 'escola',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'escola',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'escola',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+});
+
+function matchQueries(queryTotal, queryPartial, queryNeeded, zeroPercentage=false) {
+    let match = [];
+    queryTotal.forEach((result) => {
+        let newObj = {};
+        let keys = Object.keys(result);
+        keys.forEach((key) => {
+            newObj[key] = result[key];
+        });
+        let index = keys.indexOf('total');
+        if(index > -1) keys.splice(index, 1);
+        let partialMatch = null;
+        let needMatch = null;
+
+        for(let i = 0; i < queryPartial.length; ++i) {
+            let partial = queryPartial[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(partial[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                partialMatch = partial;
+                break;
+            }
+        }
+        if(queryPartial.length == 0) {
+            partialMatch = JSON.parse(JSON.stringify(result));
+            partialMatch.total = 0;
+        }
+
+        for(let i = 0; i < queryNeeded.length; ++i) {
+            let needed = queryNeeded[i];
+            let foundMatch = true;
+            for(let j = 0; j < keys.length; ++j) {
+                let key = keys[j];
+                if(needed[key] !== result[key]) {
+                    foundMatch = false;
+                    break;
+                }
+            }
+            if(foundMatch) {
+                needMatch = needed;
+                break;
+            }
+        }
+
+        if(queryNeeded.length == 0) {
+            needMatch = JSON.parse(JSON.stringify(result));
+            needMatch.total = 0;
+        }
+
+        newObj.percentage = ( ( partialMatch ? partialMatch.total : 0) / result.total) * 100;
+		if(zeroPercentage) newObj.percentage = 0;
+		newObj.partial = ( partialMatch ? partialMatch.total : 0);
+		newObj.total = result.total;
+		newObj.need_adaptation = needMatch ? needMatch.total : 0;
+		match.push(newObj);
+    });
+
+    return match;
+}
+
+infrastructureApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.querySet = [];
+    req.queryIndex = {};
+
+    // Local de funcionamento
+    let allSchools = req.sql.clone();
+    allSchools.from('escola').field('COUNT(escola.id)', 'total')
+    .field("'Brasil'", 'name')
+    .field('escola.ano_censo', 'year')
+    .group('escola.ano_censo')
+    .where('escola.situacao_de_funcionamento = 1')
+    .where('escola.ensino_regular = 1 OR escola.ensino_eja = 1 OR escola.educacao_profissional = 1')
+    .where('escola.local_func_predio_escolar = 1')
+    .where('escola.dependencia_adm_id <= 3')
+    .order('escola.ano_censo');
+    req.queryIndex.allSchools = req.querySet.push(allSchools) - 1;
+
+    let allUrbanSchools = allSchools.clone();
+    allUrbanSchools.where('escola.localizacao_id = 1');
+    req.queryIndex.allUrbanSchools = req.querySet.push(allUrbanSchools) - 1;
+
+    let allCountrySchools = allSchools.clone();
+    allCountrySchools.where('escola.localizacao_id = 2');
+    req.queryIndex.allCountrySchools = req.querySet.push(allCountrySchools) - 1;
+
+    let allSchoolsNotSchoolBuilding = req.sql.clone();
+    allSchoolsNotSchoolBuilding.from('escola').field('COUNT(escola.id)', 'total')
+    .field("'Brasil'", 'name')
+    .field('escola.ano_censo', 'year')
+    .group('escola.ano_censo')
+    .where('escola.situacao_de_funcionamento = 1')
+    .where('escola.ensino_regular = 1 OR escola.ensino_eja = 1 OR escola.educacao_profissional = 1')
+    .where('escola.local_func_predio_escolar = 0')
+    .where('escola.dependencia_adm_id <= 3')
+    .order('escola.ano_censo');
+    req.queryIndex.allSchoolsNotSchoolBuilding = req.querySet.push(allSchoolsNotSchoolBuilding) - 1;
+
+    // Bibliotecas
+    req.queryIndex.allLibraries = req.queryIndex.allUrbanSchools;
+
+    let haveLibraries = allUrbanSchools.clone();
+    haveLibraries.where('escola.biblioteca = 1');
+    req.queryIndex.haveLibraries = req.querySet.push(haveLibraries) - 1;
+
+    let needLibraries = allUrbanSchools.clone();
+    needLibraries.where('escola.biblioteca = 0');
+    req.queryIndex.needLibraries = req.querySet.push(needLibraries) - 1;
+
+    // Sala de leitura
+    req.queryIndex.allLibrariesReadingRoom = req.queryIndex.allCountrySchools;
+
+    let haveLibrariesReadingRoom = allCountrySchools.clone();
+    haveLibrariesReadingRoom.where('escola.biblioteca_sala_leitura = true');
+    req.queryIndex.haveLibrariesReadingRoom = req.querySet.push(haveLibrariesReadingRoom) - 1;
+
+    let needLibrariesReadingRoom = allCountrySchools.clone();
+    needLibrariesReadingRoom.where('escola.biblioteca_sala_leitura = false');
+    req.queryIndex.needLibrariesReadingRoom = req.querySet.push(needLibrariesReadingRoom) - 1;
+
+    // Laboratório de informática
+    // Se (situacao_de_funcionamento=1) & (dependencia_adm_id<=3) & (CEBES027P1=1) & 
+    // (ensino_regular=1 | ensino_eja=1 |    educacao_profissional=1) & 
+    // ( reg_fund_ai_t1=1 | reg_fund_af_t1=1 | reg_medio_medio_t1=1 | ensino_eja_fund= 1 |     ensino_eja_medio= 1 | ensino_eja_prof= 1 | esp_eja_fund=1 | 
+    // esp_eja_medio=1 | ensino_esp_exclusiva_eja_prof=1) então conta id
+    let allInfLab = allSchools.clone();
+    allInfLab.where('reg_fund_ai_t1=1 OR reg_fund_af_t1=1 OR reg_medio_medio_t1=1 OR ensino_eja_fund=1 OR ensino_eja_medio=1 OR ensino_eja_prof=1 OR esp_eja_fund=1 OR esp_eja_medio=1 OR ensino_esp_exclusiva_eja_prof=1');
+    req.queryIndex.allInfLab = req.querySet.push(allInfLab) - 1;
+
+    let haveInfLab = allInfLab.clone();
+    haveInfLab.where('escola.lab_informatica = 1');
+    req.queryIndex.haveInfLab = req.querySet.push(haveInfLab) - 1;
+
+    let needInfLab = allInfLab.clone();
+    needInfLab.where('escola.lab_informatica = 0');
+    req.queryIndex.needInfLab = req.querySet.push(needInfLab) - 1;
+
+    // Laboratório de ciências
+    let allScienceLab = allSchools.clone();
+    allScienceLab.where('reg_fund_af_t1=1 OR reg_medio_medio_t1=1 OR ensino_eja_fund=1 OR ensino_eja_medio=1 OR ensino_eja_prof=1 OR esp_eja_fund=1 OR esp_eja_medio=1 OR ensino_esp_exclusiva_eja_prof=1');
+    req.queryIndex.allScienceLab = req.querySet.push(allScienceLab) - 1;
+
+    let haveScienceLab = allScienceLab.clone();
+    haveScienceLab.where('escola.lab_ciencias = true');
+    req.queryIndex.haveScienceLab = req.querySet.push(haveScienceLab) - 1;
+
+    let needScienceLab = allScienceLab.clone();
+    needScienceLab.where('escola.lab_ciencias = false');
+    req.queryIndex.needScienceLab = req.querySet.push(needScienceLab) - 1;
+
+    // Parque infantil
+    // Se (situacao_de_funcionamento=1) and (ensino_regular=1 OR ensino_eja=1 OR educacao_profissional=1) and 
+    // (local_func_predio_escolar=1) and (dependencia_adm_id<=3) and (reg_infantil_creche_t1=1 or reg_infantil_preescola_t1=1 or reg_fund_ai_t1=1) então conta id
+    let allKidsPark = allSchools.clone();
+    allKidsPark.where('reg_infantil_creche_t1=1 OR reg_infantil_preescola_t1=1 OR reg_fund_ai_t1=1');
+    req.queryIndex.allKidsPark = req.querySet.push(allKidsPark) - 1;
+
+    let haveKidsPark = allKidsPark.clone();
+    haveKidsPark.where('escola.parque_infantil = 1');
+    req.queryIndex.haveKidsPark = req.querySet.push(haveKidsPark) - 1;
+
+    let needKidsPark = allKidsPark.clone();
+    needKidsPark.where('escola.parque_infantil = 0');
+    req.queryIndex.needKidsPark = req.querySet.push(needKidsPark) - 1;
+
+    // // Berçário
+    // let allCribs = allSchools.clone();
+    // allCribs.where('escola.reg_infantil_creche_t1 = 1');
+    // req.queryIndex.allCribs = req.querySet.push(allCribs) - 1;
+
+    // let haveCribs = allCribs.clone();
+    // haveCribs.where('escola.bercario = 1');
+    // req.queryIndex.haveCribs = req.querySet.push(haveCribs) - 1;
+
+    // let needCribs = allCribs.clone();
+    // needCribs.where('escola.bercario = 0');
+    // req.queryIndex.needCribs = req.querySet.push(needCribs) - 1;
+
+    // Quadra de esportes
+    let allSportsCourt = allSchools.clone();
+    allSportsCourt.where('reg_fund_ai_t1=1  or reg_fund_af_t1=1 or  reg_medio_medio_t1=1  or  ensino_eja_fund= 1 or ensino_eja_medio= 1 or ensino_eja_prof= 1 or esp_eja_fund=1 or esp_eja_medio=1 or ensino_esp_exclusiva_eja_prof=1');
+    req.queryIndex.allSportsCourt = req.querySet.push(allSportsCourt) - 1;
+
+    let haveSportsCourt = allSportsCourt.clone();
+    haveSportsCourt.where('escola.quadra_esportes = 1');
+    req.queryIndex.haveSportsCourt = req.querySet.push(haveSportsCourt) - 1;
+
+    let needSportsCourt = allSportsCourt.clone();
+    needSportsCourt.where('escola.quadra_esportes = 0');
+    req.queryIndex.needSportsCourt = req.querySet.push(needSportsCourt) - 1;
+
+    // Quadras a serem cobertas
+    // Se (situacao_de_funcionamento=1) and (ensino_regular=1 OR ensino_eja=1 OR educacao_profissional=1) and (local_func_predio_escolar=1) and 
+    // (dependencia_adm_id<=3) and (reg_fund_ai_t1=1  or reg_fund_af_t1=1 or  reg_medio_medio_t1=1  or  ensino_eja_fund= 1 or ensino_eja_medio= 1 or 
+    // ensino_eja_prof= 1 or esp_eja_fund=1 or esp_eja_medio=1 or ensino_esp_exclusiva_eja_prof=1) and (quadra_esportes_descoberta=1) então conta id
+    let allSportsCourtCoverage = allSportsCourt.clone();
+    allSportsCourtCoverage.where('escola.quadra_esportes_descoberta = 1');
+    req.queryIndex.allSportsCourtCoverage = req.querySet.push(allSportsCourtCoverage) -1;
+
+    req.queryIndex.haveSportsCourtCoverage = req.queryIndex.allSportsCourtCoverage; // It must be []
+
+    req.queryIndex.needSportsCourtCoverage = req.queryIndex.allSportsCourtCoverage;
+
+    // Pátio
+    req.queryIndex.allCourtyard = req.queryIndex.allSchools;
+
+    let haveCourtyard = allSchools.clone();
+    haveCourtyard.where('escola.patio = 1 OR escola.patio = 2');
+    req.queryIndex.haveCourtyard = req.querySet.push(haveCourtyard) - 1;
+
+    let needCourtyard = allSchools.clone();
+    needCourtyard.where('escola.patio = 0');
+    req.queryIndex.needCourtyard = req.querySet.push(needCourtyard) - 1;
+
+    // Pátios a serem cobertos
+    let allCourtyardCoverage = allSchools.clone();
+    allCourtyardCoverage.where('escola.patio = 1');
+    req.queryIndex.allCourtyardCoverage = req.querySet.push(allCourtyardCoverage) - 1;
+
+    req.queryIndex.haveCourtyardCoverage = req.queryIndex.allCourtyardCoverage; // It must be []
+
+    req.queryIndex.needCourtyardCoverage = req.queryIndex.allCourtyardCoverage;
+
+    // Sala de direção
+    req.queryIndex.allDirectorRoom = req.queryIndex.allUrbanSchools;
+
+    let haveDirectorRoom = allUrbanSchools.clone();
+    haveDirectorRoom.where('escola.sala_diretoria = 1');
+    req.queryIndex.haveDirectorRoom = req.querySet.push(haveDirectorRoom) - 1;
+
+    let needDirectorRoom = allUrbanSchools.clone();
+    needDirectorRoom.where('escola.sala_diretoria = 0');
+    req.queryIndex.needDirectorRoom = req.querySet.push(needDirectorRoom) - 1;
+
+    // Secretaria
+    req.queryIndex.allSecretary = req.queryIndex.allSchools;
+
+    let haveSecretary = allSchools.clone();
+    haveSecretary.where('escola.secretaria = 1');
+    req.queryIndex.haveSecretary = req.querySet.push(haveSecretary) - 1;
+
+    let needSecretary = allSchools.clone();
+    needSecretary.where('escola.secretaria = 0');
+    req.queryIndex.needSecretary = req.querySet.push(needSecretary) - 1;
+
+    // Sala de professores
+    req.queryIndex.allTeacherRoom = req.queryIndex.allSchools;
+
+    let haveTeacherRoom = allSchools.clone();
+    haveTeacherRoom.where('escola.sala_professor = 1');
+    req.queryIndex.haveTeacherRoom = req.querySet.push(haveTeacherRoom) - 1;
+
+    let needTeacherRoom = allSchools.clone();
+    needTeacherRoom.where('escola.sala_professor = 0');
+    req.queryIndex.needTeacherRoom = req.querySet.push(needTeacherRoom) - 1;
+
+    // Cozinha
+    req.queryIndex.allKitchen = req.queryIndex.allSchools;
+
+    let haveKitchen = allSchools.clone();
+    haveKitchen.where('escola.cozinha = 1');
+    req.queryIndex.haveKitchen = req.querySet.push(haveKitchen) - 1;
+
+    let needKitchen = allSchools.clone();
+    needKitchen.where('escola.cozinha = 0');
+    req.queryIndex.needKitchen = req.querySet.push(needKitchen) - 1;
+
+    // Despensa
+    req.queryIndex.allStoreroom = req.queryIndex.allSchools;
+
+    let haveStoreroom = allSchools.clone();
+    haveStoreroom.where('escola.despensa = 1');
+    req.queryIndex.haveStoreroom = req.querySet.push(haveStoreroom) - 1;
+
+    let needStoreroom = allSchools.clone();
+    needStoreroom.where('escola.despensa = 0');
+    req.queryIndex.needStoreroom = req.querySet.push(needStoreroom) - 1;
+
+    // Almoxarifado
+    req.queryIndex.allWarehouse = req.queryIndex.allSchools;
+
+    let haveWarehouse = allSchools.clone();
+    haveWarehouse.where('escola.almoxarifado = 1');
+    req.queryIndex.haveWarehouse = req.querySet.push(haveWarehouse) - 1;
+
+    let needWarehouse = allSchools.clone();
+    needWarehouse.where('escola.almoxarifado = 0');
+    req.queryIndex.needWarehouse = req.querySet.push(needWarehouse) - 1;
+
+    // Internet
+    req.queryIndex.allInternet = req.queryIndex.allCountrySchools;
+
+    let haveInternet = allCountrySchools.clone();
+    haveInternet.where('escola.internet = 1');
+    req.queryIndex.haveInternet = req.querySet.push(haveInternet) - 1;
+
+    let needInternet = allCountrySchools.clone();
+    needInternet.where('escola.internet = 0');
+    req.queryIndex.needInternet = req.querySet.push(needInternet) - 1;
+
+    // Internet banda larga
+    // Se (situacao_de_funcionamento=1) and (ensino_regular=1 OR ensino_eja=1 OR educacao_profissional=1) and (local_func_predio_escolar=1) and 
+    // (dependencia_adm_id<=3) and (localizacao_id=2) então conta id
+    req.queryIndex.allBroadbandInternet = req.queryIndex.allUrbanSchools;
+
+    let haveBroadbandInternet = allUrbanSchools.clone();
+    haveBroadbandInternet.where('escola.internet_banda_larga = 1');
+    req.queryIndex.haveBroadbandInternet = req.querySet.push(haveBroadbandInternet) - 1;
+
+    let needBroadbandInternet = allUrbanSchools.clone();
+    needBroadbandInternet.where('escola.internet_banda_larga = 0');
+    req.queryIndex.needBroadbandInternet = req.querySet.push(needBroadbandInternet) - 1;
+
+    // Banheiro
+    req.queryIndex.allInsideBathroom = req.queryIndex.allSchools;
+
+    let haveInsideBathroom = allSchools.clone();
+    haveInsideBathroom.where('escola.banheiro = 1');
+    req.queryIndex.haveInsideBathroom = req.querySet.push(haveInsideBathroom) - 1;
+
+    let needInsideBathroom = allSchools.clone();
+    needInsideBathroom.where('escola.banheiro = 0');
+    req.queryIndex.needInsideBathroom = req.querySet.push(needInsideBathroom) - 1;
+
+    // Banheiro adequado para educação infantil dentro do prédio
+    // Se (situacao_de_funcionamento=1) and (ensino_regular=1 OR ensino_eja=1 OR educacao_profissional=1) and (local_func_predio_escolar=1) and
+    // (dependencia_adm_id<=3) and (reg_infantil_creche_t1=1 or reg_infantil_preescola_t1=1 or  reg_fund_ai_t1=1) então conta id
+    let allInsideKidsBathroom = allSchools.clone();
+    allInsideKidsBathroom.where('reg_infantil_creche_t1=1 OR reg_infantil_preescola_t1=1 OR reg_fund_ai_t1=1');
+    req.queryIndex.allInsideKidsBathroom = req.querySet.push(allInsideKidsBathroom) - 1;
+
+    let haveInsideKidsBathroom = allInsideKidsBathroom.clone();
+    haveInsideKidsBathroom.where('escola.sanitario_ei = 1');
+    req.queryIndex.haveInsideKidsBathroom = req.querySet.push(haveInsideKidsBathroom) - 1;
+
+    let needInsideKidsBathroom = allInsideKidsBathroom.clone();
+    needInsideKidsBathroom.where('escola.sanitario_ei = 0');
+    req.queryIndex.needInsideKidsBathroom = req.querySet.push(needInsideKidsBathroom) - 1;
+
+    // Fornecimento de energia
+    req.queryIndex.allEletricPower = req.queryIndex.allSchools;
+
+    let haveEletricPower = allSchools.clone();
+    haveEletricPower.where('escola.energia_inexistente = 0');
+    req.queryIndex.haveEletricPower = req.querySet.push(haveEletricPower) - 1;
+
+    let needEletricPower = allSchools.clone();
+    needEletricPower.where('escola.energia_inexistente = 1');
+    req.queryIndex.needEletricPower = req.querySet.push(needEletricPower) - 1;
+
+    // Abastecimento de água
+    req.queryIndex.allWaterSupply = req.queryIndex.allSchools;
+
+    let haveWaterSupply = allSchools.clone();
+    haveWaterSupply.where('escola.agua_inexistente = 0');
+    req.queryIndex.haveWaterSupply = req.querySet.push(haveWaterSupply) - 1;
+
+    let needWaterSupply = allSchools.clone();
+    needWaterSupply.where('escola.agua_inexistente = 1');
+    req.queryIndex.needWaterSupply = req.querySet.push(needWaterSupply) - 1;
+
+    // Água Potável
+    req.queryIndex.allFilteredWater = req.queryIndex.allSchools;
+
+    let haveFilteredWater = allSchools.clone();
+    haveFilteredWater.where('escola.agua_potavel = 1');
+    req.queryIndex.haveFilteredWater = req.querySet.push(haveFilteredWater) - 1;
+
+    let needFilteredWater = allSchools.clone();
+    needFilteredWater.where('escola.agua_potavel = 0');
+    req.queryIndex.needFilteredWater = req.querySet.push(needFilteredWater) - 1;
+
+    // Coleta de esgoto
+    req.queryIndex.allSewage = req.queryIndex.allSchools;
+
+    let haveSewage = allSchools.clone();
+    haveSewage.where('escola.esgoto_rede_publica = 1 OR escola.esgoto_fossa_septica = 1');
+    req.queryIndex.haveSewage = req.querySet.push(haveSewage) - 1;
+
+    let needSewage = allSchools.clone();
+    needSewage.where('escola.esgoto_rede_publica = 0 AND escola.esgoto_fossa_septica = 0');
+    req.queryIndex.needSewage = req.querySet.push(needSewage) - 1;
+
+    // Dependências adaptada para pessoas com deficiências
+    req.queryIndex.allAdaptedBuilding = req.queryIndex.allSchools;
+
+    let haveAdaptedBuilding = allSchools.clone();
+    haveAdaptedBuilding.where('escola.acessibilidade_inexistente = 0');
+    req.queryIndex.haveAdaptedBuilding = req.querySet.push(haveAdaptedBuilding) - 1;
+
+    let needAdaptedBuilding = allSchools.clone();
+    needAdaptedBuilding.where('escola.acessibilidade_inexistente = 1');
+    req.queryIndex.needAdaptedBuilding = req.querySet.push(needAdaptedBuilding) - 1;
+
+    // Banheiros adaptados para pessoas com deficiências
+    req.queryIndex.allSpecialBathroom = req.queryIndex.allSchools;
+
+    let haveSpecialBathroom = allSchools.clone();
+    haveSpecialBathroom.where('escola.sanitario_pne = 1');
+    req.queryIndex.haveSpecialBathroom = req.querySet.push(haveSpecialBathroom) - 1;
+
+    let needSpecialBathroom = allSchools.clone();
+    needSpecialBathroom.where('escola.sanitario_pne = 0');
+    req.queryIndex.needSpecialBathroom = req.querySet.push(needSpecialBathroom) - 1;
+
+
+    next();
+}, multiQuery, (req, res, next) => {
+
+    let schools_in_school_buildings = req.result[req.queryIndex.allSchools];
+    let urban_schools_in_school_buildings = req.result[req.queryIndex.allUrbanSchools];
+    let country_schools_in_school_buildings = req.result[req.queryIndex.allCountrySchools];
+    let schools_not_in_school_buildings = req.result[req.queryIndex.allSchoolsNotSchoolBuilding];
+
+    // Faz o matching entre os resultados
+    let libraries = matchQueries(req.result[req.queryIndex.allLibraries], req.result[req.queryIndex.haveLibraries], req.result[req.queryIndex.needLibraries]);
+    let libraries_reading_room = matchQueries(req.result[req.queryIndex.allLibrariesReadingRoom], req.result[req.queryIndex.haveLibrariesReadingRoom], req.result[req.queryIndex.needLibrariesReadingRoom]);
+    let computer_lab = matchQueries(req.result[req.queryIndex.allInfLab], req.result[req.queryIndex.haveInfLab], req.result[req.queryIndex.needInfLab]);
+    let science_lab = matchQueries(req.result[req.queryIndex.allScienceLab], req.result[req.queryIndex.haveScienceLab], req.result[req.queryIndex.needScienceLab]);
+    let kids_park = matchQueries(req.result[req.queryIndex.allKidsPark], req.result[req.queryIndex.haveKidsPark], req.result[req.queryIndex.needKidsPark]);
+    // let nursery = matchQueries(req.result[req.queryIndex.allCribs], req.result[req.queryIndex.haveCribs], req.result[req.queryIndex.needCribs]);
+    let sports_court = matchQueries(req.result[req.queryIndex.allSportsCourt], req.result[req.queryIndex.haveSportsCourt], req.result[req.queryIndex.needSportsCourt]);
+    let sports_court_coverage = matchQueries(req.result[req.queryIndex.allSportsCourtCoverage], [], req.result[req.queryIndex.needSportsCourtCoverage], true); // have = []
+    let courtyard = matchQueries(req.result[req.queryIndex.allCourtyard], req.result[req.queryIndex.haveCourtyard], req.result[req.queryIndex.needCourtyard]);
+    let courtyard_coverage = matchQueries(req.result[req.queryIndex.allCourtyardCoverage], [], req.result[req.queryIndex.needCourtyardCoverage], true); // have = []
+    let director_room = matchQueries(req.result[req.queryIndex.allDirectorRoom], req.result[req.queryIndex.haveDirectorRoom], req.result[req.queryIndex.needDirectorRoom]);
+    let secretary = matchQueries(req.result[req.queryIndex.allSecretary], req.result[req.queryIndex.haveSecretary], req.result[req.queryIndex.needSecretary]);
+    let teacher_room = matchQueries(req.result[req.queryIndex.allTeacherRoom], req.result[req.queryIndex.haveTeacherRoom], req.result[req.queryIndex.needTeacherRoom]);
+    let kitchen = matchQueries(req.result[req.queryIndex.allKitchen], req.result[req.queryIndex.haveKitchen], req.result[req.queryIndex.needKitchen]);
+    let storeroom = matchQueries(req.result[req.queryIndex.allStoreroom], req.result[req.queryIndex.haveStoreroom], req.result[req.queryIndex.needStoreroom]);
+    let warehouse = matchQueries(req.result[req.queryIndex.allWarehouse], req.result[req.queryIndex.haveWarehouse], req.result[req.queryIndex.needWarehouse]);
+    let internet = matchQueries(req.result[req.queryIndex.allInternet], req.result[req.queryIndex.haveInternet], req.result[req.queryIndex.needInternet]);
+    let broadband_internet = matchQueries(req.result[req.queryIndex.allBroadbandInternet], req.result[req.queryIndex.haveBroadbandInternet], req.result[req.queryIndex.needBroadbandInternet]);
+    let inside_bathroom = matchQueries(req.result[req.queryIndex.allInsideBathroom], req.result[req.queryIndex.haveInsideBathroom], req.result[req.queryIndex.needInsideBathroom]);
+    let inside_kids_bathroom = matchQueries(req.result[req.queryIndex.allInsideKidsBathroom], req.result[req.queryIndex.haveInsideKidsBathroom], req.result[req.queryIndex.needInsideKidsBathroom]);
+    let eletrical_power = matchQueries(req.result[req.queryIndex.allEletricPower], req.result[req.queryIndex.haveEletricPower], req.result[req.queryIndex.needEletricPower]);
+    let water_supply = matchQueries(req.result[req.queryIndex.allWaterSupply], req.result[req.queryIndex.haveWaterSupply], req.result[req.queryIndex.needWaterSupply]);
+    let filtered_water = matchQueries(req.result[req.queryIndex.allFilteredWater], req.result[req.queryIndex.haveFilteredWater], req.result[req.queryIndex.needFilteredWater]);
+    let sewage = matchQueries(req.result[req.queryIndex.allSewage], req.result[req.queryIndex.haveSewage], req.result[req.queryIndex.needSewage]);
+    let adapted_building = matchQueries(req.result[req.queryIndex.allAdaptedBuilding], req.result[req.queryIndex.haveAdaptedBuilding], req.result[req.queryIndex.needAdaptedBuilding]);
+    let adapted_bathroom = matchQueries(req.result[req.queryIndex.allSpecialBathroom], req.result[req.queryIndex.haveSpecialBathroom], req.result[req.queryIndex.needSpecialBathroom]);
+
+    req.result = [{
+        schools_in_school_buildings,
+        urban_schools_in_school_buildings,
+        country_schools_in_school_buildings,
+        schools_not_in_school_buildings,
+        libraries,
+        libraries_reading_room,
+        computer_lab,
+        science_lab,
+        kids_park,
+        sports_court,
+        sports_court_coverage,
+        courtyard,
+        courtyard_coverage,
+        director_room,
+        secretary,
+        teacher_room,
+        kitchen,
+        storeroom,
+        warehouse,
+        internet,
+        broadband_internet,
+        inside_bathroom,
+        inside_kids_bathroom,
+        eletric_energy: eletrical_power,
+        water_supply,
+        filtered_water,
+        sewage_treatment: sewage,
+        adapted_building,
+        special_bathroom: adapted_bathroom
+    }];
+
+    next();
+}, id2str.multitransform(false), response('infrastructure'));
+
+module.exports = infrastructureApp;
diff --git a/src/libs/routes_v2/schoolLocation.js b/src/libs/routes_v2/schoolLocation.js
new file mode 100644
index 00000000..5c0d98ff
--- /dev/null
+++ b/src/libs/routes_v2/schoolLocation.js
@@ -0,0 +1,114 @@
+const express = require('express');
+
+const schoolLocationApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const passport = require('passport');
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'id',
+    table: 'localizacao_escolas',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'localizacao_escolas'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'localizacao_escolas'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: '@'
+    }
+}, 'dims');
+
+schoolLocationApp.get('/', rqf.parse(), (req, res, next) => {
+    req.dims.city=true;
+    req.dims.mesoregion=true;
+    req.dims.microregion=true;
+
+    req.sql.from('localizacao_escolas')
+        .field('localizacao_escolas.nome', 'name')
+        .field('localizacao_escolas.id', 'id')
+        .field('localizacao_escolas.latitude', 'latitude')
+        .field('localizacao_escolas.longitude', 'longitude')
+        .group('localizacao_escolas.nome')
+        .group('localizacao_escolas.id')
+        .group('localizacao_escolas.latitude')
+        .group('localizacao_escolas.longitude');
+    next();
+}, rqf.build(), query, id2str.transform(), response('school_location'));
+
+module.exports = schoolLocationApp;
diff --git a/src/libs/routes_v2/simulation.js b/src/libs/routes_v2/simulation.js
new file mode 100644
index 00000000..4b2e40c3
--- /dev/null
+++ b/src/libs/routes_v2/simulation.js
@@ -0,0 +1,167 @@
+const express = require('express');
+
+const simulationApp = express();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const Simulation = require(`${libs}/models/simulation`);
+
+const PQR = require(`${libs}/models/pqr`);
+
+const passport = require('passport');
+
+simulationApp.get('/time', (req, res, next) => {
+    const maxTime = parseInt(req.query.max_time, 10);
+    if(isNaN(maxTime)) {
+        res.status(400);
+        next({
+            status: 400,
+            message: 'Invalid value for mandatory parameter max_time'
+        });
+    }
+    res.json({
+        result: Array.apply(null, {length: maxTime}).map(Number.call, Number).map((i)=>i+1)
+    });
+});
+
+simulationApp.get('/pqr', (req, res) => {
+    PQR.findOne((err, pqr) => {
+        if(err) {
+            log.error(err);
+            return next({err});
+        }
+
+        res.json(pqr);
+    });
+});
+
+simulationApp.put('/pqr', passport.authenticate('bearer', { session: false }), (req, res, next) => {
+    let user = req.user.toObject();
+
+    PQR.findOne((err, pqr) => {
+        if(err) {
+            log.error(err)
+            return next({err});
+        }
+
+        if(!user.admin) {
+            log.info(`Usuário ${user.email} tentou alterar o PQR, mas não tem privilégio`);
+            res.statusCode = 401;
+            return next({err: { msg: 'Unauthorized'}});
+        }
+        pqr.content = req.body.content || pqr.content;
+        pqr.save((err) => {
+            if(err) {
+                log.error(err);
+                return next({err});
+            }
+            res.json({msg: 'PQR updated'})
+        });
+    });
+});
+
+simulationApp.get('/', passport.authenticate('bearer', { session: false }), (req, res) => {
+    let user = req.user.toObject();
+    let query = Simulation.find({userId: user._id}).select('userId name createdAt updatedAt');
+    query.exec((err, simulations) => {
+        if(err) {
+            log.error(err);
+            return next({err});
+        }
+
+        res.json(simulations);
+    });
+
+    // Simulation.find({userId: user._id}, (err, simulations) => {
+    //     if(err) {
+    //         log.error(err);
+    //         return next({err});
+    //     }
+
+    //     res.json(simulations);
+    // });
+});
+
+simulationApp.post('/', passport.authenticate('bearer', { session: false }), (req, res, next) => {
+    let user = req.user.toObject();
+
+    let simulation = new Simulation({
+        userId: user._id,
+        content: req.body.content,
+        name: req.body.name
+    });
+
+    simulation.save((err) => {
+        if(err) {
+            log.error(err);
+            return next({err});
+        }
+
+        res.json({msg: 'Simulation created', simulation});
+    })
+});
+
+simulationApp.get('/:id', passport.authenticate('bearer', { session: false }), (req, res) => {
+    let user = req.user.toObject();
+
+    Simulation.findOne({_id: req.params.id, userId: user._id}, (err, simulation) => {
+        if(err) {
+            log.error(err);
+            return next({err});
+        }
+
+        res.json(simulation);
+    });
+});
+
+simulationApp.put('/:id', passport.authenticate('bearer', { session: false }), (req, res, next) => {
+    let user = req.user.toObject();
+
+    Simulation.findOne({_id: req.params.id, userId: user._id}, (err, simulation) => {
+        if(err) {
+            log.error(err);
+            return next({err});
+        }
+
+        if(!simulation) {
+            res.statusCode = 404;
+            return next({err: { msg: 'Simulation not found'}});
+        }
+
+        simulation.content = req.body.content || simulation.content;
+        simulation.name = req.body.name || simulation.name;
+        simulation.updatedAt = Date.now();
+
+        simulation.save((err) => {
+            if(err) {
+                log.error(err);
+                return next(err);
+            }
+
+            res.json(simulation);
+        });
+    });
+});
+
+simulationApp.delete('/:id', passport.authenticate('bearer', { session: false }), (req, res, next) => {
+    let user = req.user.toObject();
+
+    Simulation.remove({_id: req.params.id, userId: user._id}, (err, simulation) => {
+        if(err) {
+            log.error(err);
+            return next({err});
+        }
+
+        res.json({msg: 'Simulation removed'});
+    });
+});
+
+module.exports = simulationApp;
diff --git a/src/libs/routes_v2/siope.js b/src/libs/routes_v2/siope.js
new file mode 100644
index 00000000..1ef9f101
--- /dev/null
+++ b/src/libs/routes_v2/siope.js
@@ -0,0 +1,186 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const siopeApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const multiQuery = require(`${libs}/middlewares/multiQuery`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+siopeApp.use(cache('15 day'));
+
+siopeApp.get('/years', (req, res, next) => {
+    req.sql.from('siope_mun')
+    .field('DISTINCT siope_mun.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    req.oldResult = req.result;
+
+    req.sql = squel.select();
+
+    req.sql.from('siope_uf')
+    .field('DISTINCT siope_uf.ano_censo', 'year');
+    next();
+}, query, (req, res, next) => {
+    let result = Object.assign(req.oldResult, req.result);
+    req.result = result;
+    next();
+}, response('years'));
+
+rqf.addField({
+    name: 'filter',
+    field: true,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'siope_mun'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'siope_mun'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['id','nome','sigla'],
+    resultField: ['state_id','state_name','state_abbreviation'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+});
+
+
+
+siopeApp.get('/', rqf.parse(), (req, res, next) => {
+    req.querySet = [];
+    req.queryIndex = {};
+
+    let siopeUf = req.sql.clone();
+    siopeUf.from('siope_uf')
+    .field('siope_uf.ano_censo', 'year')
+    .field('siope_uf.estado_id', 'state_id')
+    .field('siope_uf.fundeb', 'fundeb')
+    .field('siope_uf.total_impostos', 'impostos')
+    .field('siope_uf.total_mde', 'MDE')
+    .group('siope_uf.ano_censo')
+    .group('siope_uf.estado_id')
+    .group('siope_uf.fundeb')
+    .group('siope_uf.total_impostos')
+    .group('siope_uf.total_mde')
+    .order('siope_uf.ano_censo');
+    req.queryIndex.siopeUf = req.querySet.push(siopeUf) - 1;
+
+    let siopeMun = req.sql.clone();
+    siopeMun.from('siope_mun')
+    .field('siope_mun.ano_censo', 'year')
+    .field('siope_mun.estado_id', 'state_id')
+    .field('siope_mun.municipio_id', 'city_id')
+    .field('siope_mun.fundeb', 'fundeb')
+    .field('siope_mun.total_impostos', 'impostos')
+    .field('siope_mun.total_mde', 'MDE')
+    .group('siope_mun.ano_censo')
+    .group('siope_mun.estado_id')
+    .group('siope_mun.municipio_id')
+    .group('siope_mun.fundeb')
+    .group('siope_mun.total_impostos')
+    .group('siope_mun.total_mde')
+    .order('siope_mun.ano_censo');
+    req.queryIndex.siopeMun = req.querySet.push(siopeMun) - 1;
+
+    next();
+}, rqf.multibuild(), multiQuery, (req, res, next) => {
+
+    let result = []
+
+    req.result[req.queryIndex.siopeUf].forEach((item) => {
+        result.push(item)
+    });
+    req.result[req.queryIndex.siopeMun].forEach((item) => {
+        result.push(item)
+    });
+
+    req.result = result;
+    next();
+
+}, response('siope'));
+
+module.exports = siopeApp;
diff --git a/src/libs/routes_v2/spatial.js b/src/libs/routes_v2/spatial.js
new file mode 100644
index 00000000..d4f48fe8
--- /dev/null
+++ b/src/libs/routes_v2/spatial.js
@@ -0,0 +1,373 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const log = require(`${libs}/log`)(module);
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const sqlQuery = require(`${libs}/db/query_exec`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const spatialApp = express();
+
+let rqf = new ReqQueryFields();
+
+function processResultSet(querySet, querySetLabels = ["result"], singleResult = false) {
+    const resultMap = new Map();
+    let resultIdx = 0;
+    // loop relies on the fact that Promise.all maintains the order of the original iterable
+    for(let result of querySet) {
+        const resultLbl = querySetLabels[resultIdx];
+        resultMap[resultLbl] = [];
+        if (singleResult) {
+            resultMap[resultLbl] = result[0];
+        } else {
+            for(let row of result) {
+                resultMap[resultLbl].push(row);
+            }
+        }
+        resultIdx++;
+    }
+    return resultMap;
+}
+
+function dbExecAll(querySet = []) {
+    // Issue all queries concurrently to the database, for every query object in the iterable
+    // NOTE: Array.map() returns a copy of the original array with each object 'mapped'.
+    return querySet.map((qry) => { return sqlQuery(qry.toString()); });
+}
+
+rqf.addField({
+    name: 'filter',
+    field: true,
+    where: true
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id'
+    }
+}).addValue({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+});
+
+spatialApp.get('/sociodemographic', rqf.parse(), rqf.build(), (req, res, next) => {
+    const populationYearQry = squel.select()
+        .field('MAX(ibge_populacao.ano_censo)')
+        .from('ibge_populacao');
+
+    const populationQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('SUM(populacao)', 'population')
+        .field('ibge_populacao.ano_censo', 'census_year')
+        .from('ibge_populacao')
+        .where(`ibge_populacao.ano_censo IN (${populationYearQry.toString()})`)
+        .group('ibge_populacao.ano_censo');
+
+    const pibYearQry = squel.select()
+        .field('MAX(ibge_pib.ano_censo)')
+        .from('ibge_pib');
+
+    const pibQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('AVG(ibge_pib.pib_per_capita)', 'gdp_per_capita')
+        .field('ibge_pib.ano_censo', 'census_year')
+        .from('ibge_pib')
+        .where(`ibge_pib.ano_censo IN (${pibYearQry.toString()})`)
+        .group('ibge_pib.ano_censo');
+
+    const idhYearQry = squel.select()
+        .field('MAX(adh_idh.ano_censo)')
+        .from('adh_idh');
+
+    const idhQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('AVG(idhm)', 'idhm')
+        .field('adh_idh.ano_censo', 'census_year')
+        .from('adh_idh')
+        .where(`adh_idh.ano_censo IN (${idhYearQry.toString()})`)
+        .group('adh_idh.ano_censo');
+
+    const analfabYearQry = squel.select()
+        .field('MAX(adh_analfabetismo.ano_censo)')
+        .from('adh_analfabetismo');
+
+    const analfabQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('AVG(t_analf15m)', 'analfabetism')
+        .field('adh_analfabetismo.ano_censo', 'census_year')
+        .from('adh_analfabetismo')
+        .where(`adh_analfabetismo.ano_censo IN (${analfabYearQry.toString()})`)
+        .group('adh_analfabetismo.ano_censo');
+
+    const giniYearQry = squel.select()
+        .field('MAX(adh_gini.ano_censo)')
+        .from('adh_gini');
+
+    const giniQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('AVG(gini)', 'gini')
+        .field('adh_gini.ano_censo', 'census_year')
+        .from('adh_gini')
+        .where(`adh_gini.ano_censo IN (${giniYearQry.toString()})`)
+        .group('adh_gini.ano_censo');
+
+    // map query objects to their respective response labels
+    const queryLabels = [ "population", "gdp", "idh", "analfab", "gini" ];
+    const querySet = [ populationQry, pibQry, idhQry, analfabQry, giniQry ];
+    // wait until all queries finish or one of them fail
+    Promise.all(dbExecAll(querySet)).then((queryResults) => {
+        req.result = processResultSet(queryResults, queryLabels, true);
+        next();
+    }).catch((error) => {
+        log.error(`[SQL query error] ${error}`);
+        next(error);
+    });
+}, response('spatial'));
+
+spatialApp.get('/educational', rqf.parse(), rqf.build(), (req, res, next) => {
+    const censusYearQry = squel.select()
+        .field('MAX(escola.ano_censo)', 'ano_censo')
+        .from('escola')
+        .toString();
+
+    const totalSchoolsQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('\'Total\'', 'location_name')
+        .field('COUNT(DISTINCT(escola.id))', 'total')
+        .field('escola.ano_censo', 'census_year')
+        .from('turma')
+        .from('escola')
+        .where('escola.ano_censo=turma.ano_censo AND escola.id=turma.escola_id')
+        .where(`escola.ano_censo IN (${censusYearQry})`)
+        .where('turma.tipo_turma_id = 0')
+        .group('escola.ano_censo');
+
+    const schoolsPerLocationQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('COUNT(DISTINCT(escola.id))', 'total')
+        .field('escola.ano_censo', 'census_year')
+        .field('localizacao.descricao', 'location_name')
+        .from('localizacao')
+        .from('turma')
+        .from('escola')
+        .where('escola.localizacao_id=localizacao.id')
+        .where('escola.ano_censo=turma.ano_censo AND escola.id=turma.escola_id')
+        .where(`escola.ano_censo IN (${censusYearQry})`)
+        .where('turma.tipo_turma_id = 0')
+        .group('escola.localizacao_id')
+        .group('escola.ano_censo')
+        .group('localizacao.descricao')
+        .order('localizacao.descricao');
+
+    const schoolsPerAdmDependencyQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('COUNT(DISTINCT(escola.id))', 'total')
+        .field('escola.ano_censo', 'census_year')
+        .field('dependencia_adm.nome', 'adm_dependency_name')
+        .from('dependencia_adm')
+        .from('escola')
+        .where('escola.dependencia_adm_id=dependencia_adm.id')
+        .where(`escola.ano_censo IN (${censusYearQry})`)
+        .group('escola.ano_censo')
+        .group('dependencia_adm.nome')
+        .order('escola.ano_censo')
+        .order('dependencia_adm.nome');
+
+    const enrollmentsQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('COALESCE(SUM(uc201.matriculas), 0)', 'total')
+        .field('uc201.ano_censo', 'census_year')
+        .from('uc201')
+        .group('uc201.ano_censo');
+
+    const enrollmentsPerAdmDepQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('COALESCE(SUM(uc201.matriculas), 0)', 'total')
+        .field('uc201.ano_censo', 'census_year')
+        .field('dependencia_adm.nome', 'adm_dependency_name')
+        .from('dependencia_adm')
+        .from('uc201')
+        .where('uc201.dependencia_adm_id=dependencia_adm.id')
+        .group('dependencia_adm.nome')
+        .group('uc201.ano_censo');
+
+    const enrollmentsPerSchoolLevelQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('COALESCE(SUM(uc201.matriculas), 0)', 'total')
+        .field('uc201.ano_censo', 'census_year')
+        .field('etapa_ensino.desc_etapa', 'school_level_name')
+        .field('dependencia_adm.nome', 'adm_dependency_name')
+        .from('etapa_ensino')
+        .from('dependencia_adm')
+        .from('uc201')
+        .where('uc201.etapas_mod_ensino_segmento_id=etapa_ensino.id')
+        .where('uc201.dependencia_adm_id=dependencia_adm.id')
+        .group('etapa_ensino.desc_etapa')
+        .group('dependencia_adm.nome')
+        .group('uc201.ano_censo');
+
+    const enrollmentsPerLocationQry = req.sql.clone()
+        .field('\'Brasil\'', 'name')
+        .field('COALESCE(SUM(uc201.matriculas), 0)', 'total')
+        .field('uc201.ano_censo', 'census_year')
+        .field('localizacao.descricao', 'location_name')
+        .from('localizacao')
+        .from('uc201')
+        .where('uc201.localizacao=localizacao.id')
+        .group('localizacao.descricao')
+        .group('uc201.ano_censo');
+
+    const queryLabels = [ "school", "school_per_location", "school_per_adm_dependency", "enrollment", "enrollment_per_adm_dep",
+        "enrollment_per_school_level", "enrollment_per_location" ];
+    const querySet = [ totalSchoolsQry, schoolsPerLocationQry, schoolsPerAdmDependencyQry, enrollmentsQry,
+        enrollmentsPerAdmDepQry, enrollmentsPerSchoolLevelQry, enrollmentsPerLocationQry];
+    // wait until all queries finish or one of them fail
+    Promise.all(dbExecAll(querySet)).then((queryResults) => {
+        req.result = processResultSet(queryResults, queryLabels);
+        next();
+    }).catch((error) => {
+        log.error(`[SQL query error] ${error}`);
+        next(error);
+    });
+}, response('spatial'));
+
+spatialApp.get('/educational/school_level', rqf.parse(), rqf.build(), (req, res, next) => {
+    const enrollmentsPerSchoolLevelYearQry = squel.select()
+        .field('MAX(matricula.ano_censo)', 'census_year')
+        .from('matricula');
+
+    const enrollmentsPerSchoolLevelQry = req.sql.clone()
+        .field('COALESCE(COUNT(matricula.id), 0)', 'total')
+        .field('matricula.ano_censo', 'census_year')
+        .field('matricula.serie_ano_id', 'school_year')
+        .field('etapa_ensino.desc_etapa', 'school_level')
+        .from('etapa_ensino')
+        .from('matricula')
+        .where(`matricula.ano_censo IN (${enrollmentsPerSchoolLevelYearQry.toString()})`)
+        .where('matricula.etapa_ensino_id = etapa_ensino.id')
+        .where('matricula.tipo <= 3')
+        .group('etapa_ensino.desc_etapa')
+        .group('etapa_ensino.id')
+        .group('matricula.serie_ano_id')
+        .group('matricula.ano_censo')
+        .order('etapa_ensino.id')
+        .order('matricula.serie_ano_id')
+        .order('matricula.ano_censo');
+
+    const queryLabels = [ 'enrollment_per_school_level', 'enrollment_census_year' ];
+    const querySet = [ enrollmentsPerSchoolLevelQry, enrollmentsPerSchoolLevelYearQry ];
+    // wait until all queries finish or one of them fail
+    Promise.all(dbExecAll(querySet, enrollmentsPerSchoolLevelYearQry)).then((queryResults) => {
+        const result = queryResults[0];
+        const censusYear = queryResults[1][0]['census_year'];
+
+        let school_levels = {};
+        for(let i = 0; i < result.length; ++i) {
+            const school_year  = id2str.schoolYear(result[i].school_year);
+            const school_level = result[i].school_level;
+            const census_year = result[i].census_year;
+            if (typeof school_levels[school_level] === 'undefined') {
+                school_levels[school_level] = {};
+            }
+            school_levels[school_level][school_year] = parseInt(result[i].total, 10);
+        }
+
+        let response = [];
+        for(let level in school_levels) {
+            if (school_levels.hasOwnProperty(level)) {
+                let sclevel = {};
+                sclevel["degree"] = level;
+                sclevel["census_year"] = parseInt(censusYear, 10);
+                sclevel["table"] = [];
+                for(let school_year in school_levels[level]) {
+                    if (school_levels[level].hasOwnProperty(school_year)) {
+                        let enrollment = { 'title' : school_year,
+                                           'value' : school_levels[level][school_year] };
+                        sclevel["table"].push(enrollment);
+                    }
+                }
+                response.push(sclevel);
+            }
+        }
+        req.result = response;
+        next();
+    }).catch((error) => {
+        log.error(`[SQL query error] ${error}`);
+        next(error);
+    });
+}, response('spatial'));
+
+module.exports = spatialApp;
diff --git a/src/libs/routes_v2/state.js b/src/libs/routes_v2/state.js
new file mode 100644
index 00000000..c9830b20
--- /dev/null
+++ b/src/libs/routes_v2/state.js
@@ -0,0 +1,108 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const stateApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`); 
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+stateApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'id',
+    table: 'estado',
+    tableField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValue({
+    name: 'id_not',
+    table: 'estado',
+    tableField: 'id',
+    where: {
+        relation: '<>',
+        type: 'integer',
+        field: 'id'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'regiao_id',
+        table: 'estado'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'estado'
+    }
+}).addField({
+    name: 'search',
+    field: false,
+    where: true
+}).addValueToField({
+    name: 'name',
+    table: 'estado',
+    tableField: 'nome',
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome'
+    }
+}, 'search');
+
+stateApp.get('/', rqf.parse(), rqf.build(), (req, res, next) => {
+    req.sql.from('estado')
+        .field('estado.id').group('estado.id')
+        .field('regiao_id', 'region_id').group('regiao_id')
+        .field('estado.nome', 'name').group('estado.nome')
+        .field('estado.sigla', 'abbreviation').group('estado.sigla')
+        .field('estado.longitude', 'longitude').group('estado.longitude')
+        .field('estado.latitude', 'latitude').group('estado.latitude')
+				.where('estado.id <> 99');
+    next();
+}, query, response('state'));
+
+module.exports = stateApp;
diff --git a/src/libs/routes_v2/studentsAee.js b/src/libs/routes_v2/studentsAee.js
new file mode 100644
index 00000000..2a45a2f5
--- /dev/null
+++ b/src/libs/routes_v2/studentsAee.js
@@ -0,0 +1,219 @@
+const express = require('express');
+
+const studentsAeeApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const passport = require('passport');
+
+const download = require(`${libs}/middlewares/downloadDatabase`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+let rqfCount = new ReqQueryFields();
+
+// cubApp.get('/year_range', (req, res, next) => {
+
+//     req.sql.from('cub')
+//     .field('MIN(cub.ano_censo)', 'start_year')
+//     .field('MAX(cub.ano_censo)', 'end_year');
+//     next();
+// }, query, response('range'));
+
+// cubApp.get('/years', (req, res, next) => {
+//     req.sql.from('cub')
+//     .field('DISTINCT cub.ano_censo', 'year');
+//     next();
+// }, query, response('years'));
+
+// cubApp.get('/months', (req, res, next) => {
+//     req.sql.from('cub')
+//     .field('DISTINCT cub.mes_censo', 'month');
+//     next();
+// }, query, response('months'));
+
+// cubApp.get('/years_months', (req, res, next) => {
+//     req.sql.from('cub')
+//     .field('DISTINCT cub.ano_censo AS "year", cub.mes_censo AS "month"');
+//     next();
+// }, query, response('years_months'));
+
+// cubApp.get('/price_type', (req, res, next) => {
+//     req.sql.from('cub')
+//     .field('DISTINCT cub.tipo_preco', 'price_type');
+//     next();
+// }, query, response('price_type'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValueToField({ 
+    name: 'state', // working
+    table: 'estado',
+    tableField: ['sigla', 'id'],
+    resultField: ['sigla_uf', 'cod_uf'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'numero_estudantes_aee'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'estado_id',
+        foreignTable: 'numero_estudantes_aee'
+    }
+}, 'filter').addValueToField({
+    name: 'city', // working
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'numero_estudantes_aee'
+    }
+}, 'filter').addValue({
+    name: 'region', // working
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'numero_estudantes_aee'
+    }
+}).addValueToField({
+    name: 'school', // working
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'numero_estudantes_aee'
+    }
+}, 'filter').addValueToField({
+    name: 'locale_id', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'localidade_area_rural',
+    resultField: 'locale_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}, 'filter').addValue({
+    name: 'ethnic_group', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'cor_raca_id',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca_id'
+    }
+}).addValue({
+    name: 'adm_dependency', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name:'age_range_all', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'faixa_etaria_31_03',
+    resultField: 'age_range_all_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'faixa_etaria_31_03'
+    }
+  }).addValue({
+    name: 'gender', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'activity_days', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'dias_atividade',
+    resultField: 'activity_days_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dias_atividade'
+    }
+}).addField({
+    name: 'special_service', // working
+    table: 'numero_estudantes_aee',
+    tableField: 'disc_atendimento_especiais',
+    resultField: 'special_service_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'disc_atendimento_especiais'
+    }
+});
+
+studentsAeeApp.get('/', rqf.parse(), (req, res, next) => {
+    req.sql.from('numero_estudantes_aee')
+    .field('numero_estudantes_aee.ano_censo')
+    .field('COUNT(distinct numero_estudantes_aee.id_aluno)', 'total')
+    .group('numero_estudantes_aee.ano_censo')
+    .order('numero_estudantes_aee.ano_censo')
+    next();
+}, rqf.build(), (req, res, next) => {
+    console.log(req.sql.toString());
+    next();
+}, query, id2str.transform(), response('studentsAee'));
+
+module.exports = studentsAeeApp;
diff --git a/src/libs/routes_v2/teacher.js b/src/libs/routes_v2/teacher.js
new file mode 100644
index 00000000..94e6d86c
--- /dev/null
+++ b/src/libs/routes_v2/teacher.js
@@ -0,0 +1,576 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const teacherApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const config = require(`${libs}/config`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]}  }).middleware;
+
+let rqf = new ReqQueryFields();
+
+teacherApp.use(cache('15 day'));
+
+// Returns a tuple of start and ending years of the complete enrollments dataset.
+teacherApp.get('/year_range', (req, res, next) => {
+    req.sql.from('docente')
+    .field('MIN(docente.ano_censo)', 'start_year')
+    .field('MAX(docente.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+teacherApp.get('/years', (req, res, next) => {
+    req.sql.from('docente').
+    field('DISTINCT docente.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+teacherApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'docente\'');
+    next();
+}, query, response('source'));
+
+teacherApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+teacherApp.get('/diff_location', (req, res, next) => {
+    req.result = [
+        {id: 0, name: "A escola não está em localidade diferenciada"},
+        {id: 1, name: "Área de assentamento"},
+        {id: 2, name: "Terra indígena"},
+        {id: 3, name: "Terra remanescente de quilombos"},
+    ];
+    next();
+}, response('diff_location'));
+
+teacherApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 8; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+teacherApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+teacherApp.get('/education_level_mod', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 12; ++i) {   
+        req.result.push({
+            id: i,
+            name: id2str.educationLevelMod(i)
+        });
+    }
+    req.result.push({
+        id: 99,
+        name: id2str.educationLevelMod(99)
+    });
+    next();
+}, response('education_level_mod'));
+
+teacherApp.get('/education_level_short', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Creche'},
+        {id: 2, name: 'Pré-Escola'},
+        {id: 3, name: 'Ensino Fundamental - anos iniciais'},
+        {id: 4, name: 'Ensino Fundamental - anos finais'},
+        {id: 5, name: 'Ensino Médio'},
+        {id: 6, name: 'EJA'},
+        {id: 7, name: 'EE exclusiva'}
+    ];
+    next();
+}, response('education_level_short'));
+
+teacherApp.get('/education_type', (req, res, next) => {
+    req.sql.from('docente')
+    .field('DISTINCT nivel_tipo_formacao', 'id')
+    .order('id');
+    next();
+}, query, (req, res, next) => {
+    req.result.forEach((result) => {
+        result.name = id2str.educationType(result.id);
+    });
+    next();
+}, response('education_type'));
+
+teacherApp.get('/gender', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Masculino'},
+        {id: 2, name: 'Feminino'}
+    ];
+    next();
+}, response('gender'));
+
+
+teacherApp.get('/contract_type', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        contractType: id2str.contractType("null")
+    }];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.contractType(i)
+        });
+    }
+    next();
+}, response('contract_type'));
+
+teacherApp.get('/ethnic_group', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <=5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ethnicGroup(i)
+        });
+    }
+    next();
+}, response('ethnic_group'));
+
+teacherApp.get('/initial_training', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <=5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.initialTraining(i)
+        });
+  }
+  next();
+}, response('initial_training'));
+
+teacherApp.get('/pos_training', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.posTraining(i)
+        });
+  }
+  next();
+}, response('pos_training'));
+
+teacherApp.get('/licentiate_degree', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 3; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.licentiateDegree(i)
+        });
+  }
+  next();
+}, response('licentiate_degree'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'adm_dependency',
+    table: 'docente',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'docente',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'contract_type',
+    table: 'docente',
+    tableField: 'tipo_contratacao',
+    resultField: 'contract_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'tipo_contratacao'
+    }
+}).addValue({
+    name: 'education_level_mod',
+    table: 'docente',
+    tableField: 'etapas_mod_ensino_segmento_id',
+    resultField: 'education_level_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapas_mod_ensino_segmento_id'
+    }
+}).addValue({
+    name: 'education_level_short',
+    table: 'docente',
+    tableField: 'etapa_resumida',
+    resultField: 'education_level_short_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapa_resumida'
+    }
+}).addValue({
+    name: 'education_type',
+    table: 'docente',
+    tableField: 'nivel_tipo_formacao',
+    resultField: 'education_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'nivel_tipo_formacao'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_regiao_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_estado_id',
+        foreignTable: 'docente'
+    }
+}).addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}, 'dims').addValueToField({
+    name: 'city',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'escola_municipio_id',
+        foreignTable: 'docente'
+    }
+}, 'filter').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'docente'
+    }
+}, 'dims').addValueToField({
+    name: 'school',
+    table: 'escola',
+    tableField: 'nome_escola',
+    resultField: 'school_name',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'docente'
+    }
+}, 'filter').addValue({
+    name: 'location',
+    table: 'docente',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name: 'diff_location',
+    table: 'docente',
+    tableField: 'localizacao_diferenciada_par',
+    resultField: 'diff_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_diferenciada_par'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'docente',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'docente',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'gender',
+    table: 'docente',
+    tableField: 'sexo',
+    resultField: 'gender_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo'
+    }
+}).addValue({
+    name: 'ethnic_group',
+    table: 'docente',
+    tableField: 'cor_raca',
+    resultField: 'ethnic_group_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cor_raca'
+    }
+}).addValue({
+    name: 'initial_training',
+    table: 'docente',
+    tableField: 'formacao_inicial_docente',
+    resultField: 'initial_training_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'formacao_inicial_docente'
+    }
+}).addValue({
+    name: 'pos_training',
+    table: 'docente',
+    tableField: 'formacao_pos_docente',
+    resultField: 'pos_training_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'formacao_pos_docente'
+    }
+}).addValue({
+    name: 'licentiate_degree',
+    table: 'docente',
+    tableField: 'formacao_licenciatura_docente',
+    resultField: 'licentiate_degree_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'formacao_licenciatura_docente'
+    }
+});
+
+const sortYearPtid = (a, b) => {
+    if (a.year < b.year)
+        return -1
+    if (a.year > b.year)
+        return 1
+    
+    if (a.pos_training_id < b.pos_training_id)
+        return -1
+    if (a.pos_training_id > b.pos_training_id)
+        return 1
+    return 0
+}
+
+teacherApp.get('/', rqf.parse(), (req, res, next) => {
+    req.sql.field('COUNT(DISTINCT docente.id_docente)', 'total')
+    .field("'Brasil'", 'name')
+    .field('docente.ano_censo', 'year')
+    .from('docente')
+    .join('turma', null, 'docente.turma_id=turma.id AND docente.ano_censo=turma.ano_censo')
+    .group('docente.ano_censo')
+    .order('docente.ano_censo')
+    .where('(docente.tipo_docente = 1 OR docente.tipo_docente = 5) AND \
+        ((docente.tipo_turma_id >= 0 AND docente.tipo_turma_id <= 3 AND docente.tipo_turma_atendimento_id is NULL) \
+        OR ((docente.tipo_turma_atendimento_id = 1 OR docente.tipo_turma_atendimento_id = 2) AND docente.tipo_turma_id is NULL)) \
+        AND (docente.ano_censo <> 2009 or (docente.escola_estado_id <> 42 and docente.escola_estado_id <> 43) )');     // não devemos trazer SC em 2009.
+
+    // if("education_level_mod" in req.dims) {
+    //     req.hadEducationLevelMod = true;
+    //     req.sql.where('docente.etapas_mod_ensino_segmento_id < 11');
+    // }
+
+    next();
+}, rqf.build(), query, addMissing(rqf), (req, res, next) => {
+    // Função criada para preencher valores faltantes no gráfico do indicador de 
+    // formação em pós graduação do MapFOR.
+    if (req.dims.pos_training){
+        var year = req.result[0].year;
+        var posTrainingIds = req.result.map(obj => {
+            if (year == obj.year) 
+                return obj.pos_training_id
+        }).filter(num => num !== undefined)
+        
+        var missingValues = [];
+        for(let i = 1; i <= 4; ++i) {
+            if ( !posTrainingIds.includes(i) ){
+                missingValues.push(i);
+            }
+        }
+        
+        for (let curYear = 2012; curYear <= 2020; ++curYear){
+            for (let ptId of missingValues){
+                req.result.push({
+                    total:0,
+                    name:"Brasil",
+                    year:curYear,
+                    pos_training_id:ptId,
+                    pos_training_name:id2str.posTraining(ptId)
+                })
+            }
+        }
+        req.result.sort(sortYearPtid)
+        
+    }
+    next();
+}, id2str.transform(), response('teacher'));
+
+module.exports = teacherApp;
diff --git a/src/libs/routes_v2/transport.js b/src/libs/routes_v2/transport.js
new file mode 100644
index 00000000..74c1b095
--- /dev/null
+++ b/src/libs/routes_v2/transport.js
@@ -0,0 +1,400 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const transportApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+transportApp.use(cache('15 day'));
+
+transportApp.get('/year_range', (req, res, next) => {
+    req.sql.from('transporte')
+    .field('MIN(transporte.ano_censo)', 'start_year')
+    .field('MAX(transporte.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+transportApp.get('/years', (req, res, next) => {
+    req.sql.from('transporte')
+    .field('DISTINCT transporte.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+transportApp.get('/adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependency(i)
+        });
+    };
+    next();
+}, response('adm_dependency'));
+
+transportApp.get('/adm_dependency_detailed', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.admDependencyPriv(i)
+        });
+    };
+    next();
+}, response('adm_dependency_detailed'));
+
+transportApp.get('/location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: 'Urbana'},
+        {id: 2, name: 'Rural'}
+    ];
+    next();
+}, response('location'));
+
+transportApp.get('/rural_location', (req, res, next) => {
+    req.result = [
+        {id: 1, name: "Urbana"},
+        {id: 2, name: "Rural"},
+        {id: 3, name: "Rural - Área de assentamento"},
+        {id: 4, name: "Rural - Terra indígena"},
+        {id: 5, name: "Rural - Área remanescente de quilombos"},
+        {id: 6, name: "Rural - Unidade de uso sustentável"}
+    ];
+    next();
+}, response('rural_location'));
+
+transportApp.get('/education_level_mod', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Creche'},
+        {id: 2, name: 'Pré-Escola'},
+        {id: 4, name: 'Ensino Fundamental - anos iniciais'},
+        {id: 5, name: 'Ensino Fundamental - anos finais'},
+        {id: 6, name: 'Ensino Médio'},
+        {id: 7, name: 'Turmas multiseriadas e multietapas'},
+        {id: 8, name: 'EJA - Ensino Fundamental'},
+        {id: 9, name: 'EJA - Ensino Médio'},
+        {id: 10, name: 'Educação Profissional'}
+    ];
+    next();
+}, response('education_level_mod'));
+
+transportApp.get('/service_type', (req, res, next) => {
+    req.result = [
+        {id: 0, name: 'Não se aplica'},
+        {id: 1, name: 'Classe hospitalar'},
+        {id: 2, name: 'Unidade de Atendimento Socioeducativo'},
+        {id: 3, name: 'Unidade Prisional'},
+        {id: 4, name: 'Atividade Complementar '},
+        {id: 5, name: 'Atendimento Educacional Especializado (AEE)'}
+    ];
+    next();
+}, response('service_type'));
+
+transportApp.get('/transportation_manager', (req, res, next) => {
+    req.result = [
+        {id: null, name: 'Não classificada'},
+        {id: 1, name: 'Estadual'},
+        {id: 2, name: 'Municipal'},
+    ];
+    next();
+}, response('transportation_manager'));
+
+transportApp.get('/source', (req, res, next) => {
+    req.sql.from('fonte')
+    .field('fonte', 'source')
+    .where('tabela = \'matricula\'');
+    next();
+}, query, response('source'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'school',
+    table: 'escola',
+    tableField: ['nome_escola', 'id'],
+    resultField: ['school_name', 'school_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: ['id', 'ano_censo'],
+        foreign: ['escola_id', 'ano_censo'],
+        foreignTable: 'transporte'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'regiao_id',
+        foreignTable: 'transporte'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'municipio_id',
+        table: 'transporte'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'municipio_id',
+        foreignTable: 'transporte'
+    }
+}).addValue({
+    name: 'state',
+    table: 'transporte',
+    tableField: ['estado_nome', 'estado_id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'estado_id',
+        table: 'transporte'
+    }
+}).addValue({
+    name: 'rural_location',
+    table: 'transporte',
+    tableField: 'localidade_area_rural',
+    resultField: 'rural_location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localidade_area_rural'
+    }
+}).addValue({
+    name: 'location',
+    table: 'transporte',
+    tableField: 'localizacao_id',
+    resultField: 'location_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localizacao_id'
+    }
+}).addValue({
+    name:'adm_dependency',
+    table: 'transporte',
+    tableField: 'dependencia_adm_id',
+    resultField: 'adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_id'
+    }
+}).addValue({
+    name: 'adm_dependency_detailed',
+    table: 'transporte',
+    tableField: 'dependencia_adm_priv',
+    resultField: 'adm_dependency_detailed_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'dependencia_adm_priv'
+    }
+}).addValue({
+    name: 'transportation_manager',
+    table: 'transporte',
+    tableField: 'responsavel_transp',
+    resultField: 'transportation_manager_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'responsavel_transp'
+    }
+}).addValue({
+    name: 'education_level_mod',
+    table: 'transporte',
+    tableField: 'etapas_mod_ensino_segmento_id',
+    resultField: 'education_level_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'etapas_mod_ensino_segmento_id'
+    }
+}).addValue({
+    name: 'service_type',
+    table: 'transporte',
+    tableField: 'tipo',
+    resultField: 'service_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'tipo'
+    }
+}).addValue({
+    name: 'min_year',
+    table: 'transporte',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'transporte',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'year',
+    table: 'transporte',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+});
+
+transportApp.get('/', rqf.parse(), (req, res, next) => {
+    req.dims.year = true;
+    req.sql
+        .field('sum(transporte.total)', 'total')
+        .field("'Brasil'", 'name')
+        .from('transporte')
+        .where('transporte.transporte_id=0')
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.total = req.result;
+    req.resetSql();
+    next();
+}, rqf.parse(), (req, res, next) => {
+    req.dims.year = true;
+    req.sql
+        .field('sum(transporte.total)', 'total')
+        .field("'Brasil'", 'name')
+        .from('transporte')
+        .where('transporte.transporte_id=1')
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    req.public_total = req.result;
+    req.resetSql();
+    next();
+
+}, rqf.parse(), (req, res, next) => {
+    req.dims.year = true;
+    req.sql
+        .field('sum(transporte.total)', 'total')
+        .field("'Brasil'", 'name')
+        .field('transporte.transporte_id', 'id')
+        .from('transporte')
+        .where('transporte.transporte_id>0')
+        .group('transporte.transporte_id')
+        .order('transporte.transporte_id')
+    next();
+}, rqf.build(), query, id2str.transform(), (req, res, next) => {
+    let transports = req.result;
+
+    let results = [];
+    let obj = {};
+    for (let i = 1; i < 13; i++) {
+        obj[id2str.transport(i)] = [];
+    }
+
+    let i = 0
+    while (i < transports.length) {
+        let result = [];
+        let j = 0;
+        let transport = transports[i];
+        let totalArray = (transport.id  == 1) ? req.total : req.public_total;
+        let match;
+        obj[id2str.transport(transport.id)] = result;
+        while (j < totalArray.length && i < transports.length) {
+            transport = transports[i];
+            let transportTotal = totalArray[j];
+
+            let currentTransport = {};
+            delete transport.id;
+            match = true;
+            Object.keys(transport).forEach(function(key) {
+                currentTransport[key] = transportTotal[key];
+                if (key != 'total' && transport[key] != transportTotal[key]) {
+                    match = false;
+                    return;
+                }
+            })
+
+            if (match) {
+                currentTransport.partial = (match) ? transport.total : 0;
+                currentTransport.percentage = (match) ? transport.total/transportTotal.total * 100 : 0;
+                result.push(currentTransport);
+                i++;
+                j++;
+            }
+            else
+                j++;
+        }
+    }
+    results.push(obj)
+    req.result = results;
+
+    next();
+}, response('transports'));
+
+module.exports = transportApp;
diff --git a/src/libs/routes_v2/university.js b/src/libs/routes_v2/university.js
new file mode 100644
index 00000000..be7fe771
--- /dev/null
+++ b/src/libs/routes_v2/university.js
@@ -0,0 +1,347 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const universityApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+let rqf = new ReqQueryFields();
+
+let rqfCount = new ReqQueryFields();
+
+universityApp.use(cache('15 day'));
+
+universityApp.get('/upper_adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 7; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperAdmDependency(i)
+        });
+    };
+    next();
+}, response('upper_adm_dependency'));
+
+universityApp.get('/years', (req, res, next) => {
+    req.sql.from('ies_ens_superior')
+    .field('DISTINCT ies_ens_superior.ano_censo', 'year')
+    next();
+}, query, response('years'));
+
+universityApp.get('/year_range', (req, res, next) => {
+    req.sql.from('ies_ens_superior')
+    .field('MIN(ies_ens_superior.ano_censo)', 'start_year')
+    .field('MAX(ies_ens_superior.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+universityApp.get('/academic_organization', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.academicOrganization(i)
+        });
+    };
+    next();
+}, response('academic_organization'));
+
+universityApp.get('/capital', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('capital'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addValue({
+    name: 'id',
+    table: 'ies_ens_superior',
+    tableField: 'cod_ies',
+    resultField: 'id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_municipio_ies',
+        table: 'ies_ens_superior'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_uf_ies',
+        table: 'ies_ens_superior'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_uf_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'id'
+    },
+    join: {
+        primary: 'nome',
+        foreign: 'nome_regiao_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+}).addValue({
+    name: 'year',
+    table: 'ies_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'ano_censo',
+        table: 'ies_ens_superior'
+    }
+}).addField({
+    name: 'search',
+    field: true,
+    where: true
+}).addValueToField({
+    name: 'city_name',
+    table: 'municipio',
+    tableField: 'nome',
+    resultField: 'city_name',
+    dontGroup: true,
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'nome'
+    },
+    join: {
+      primary: 'id',
+      foreign: 'cod_municipio_ies',
+      foreignTable: 'ies_ens_superior'
+    }
+}, 'search')
+.addValueToField({
+    name: 'state_name',
+    table: 'estado',
+    tableField: 'nome',
+    resultField: 'state_name',
+    dontGroup: true,
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'sigla'
+    },
+    join: {
+      primary: 'id',
+      foreign: 'cod_uf_ies',
+      foreignTable: 'ies_ens_superior'
+    }
+}, 'search');
+
+
+rqfCount.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_municipio_ies',
+        table: 'ies_ens_superior'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'id'
+    },
+    join: {
+        primary: 'nome',
+        foreign: 'nome_regiao_ies',
+        foreignTable: 'ies_ens_superior'
+    }
+
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_uf_ies',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_uf_ies',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'upper_adm_dependency',
+    table: 'ies_ens_superior',
+    tableField: 'par_categoria_administrativa',
+    resultField: 'upper_adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'ies_ens_superior',
+        field: 'cod_categoria_administrativa'
+    }
+}).addValue({
+    name: 'academic_organization',
+    table: 'ies_ens_superior',
+    tableField: 'cod_organizacao_academica',
+    resultField: 'academic_organization_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'ies_ens_superior',
+        field: 'cod_organizacao_academica'
+    }
+}).addValue({
+    name: 'capital',
+    table: 'ies_ens_superior',
+    tableField: 'tfd_capital_ies',
+    resultField: 'capital_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'ies_ens_superior',
+        field: 'capital_ies'
+    }
+});
+
+universityApp.get('/', rqf.parse(),  rqf.build(), (req, res, next) => {
+    req.sql.from('ies_ens_superior')
+        .field('ies_ens_superior.cod_ies', 'id')
+        .field('ies_ens_superior.ano_censo', 'year')
+        .field('ies_ens_superior.nome_ies', 'name')
+        .field('ies_ens_superior.cod_uf_ies', 'state_id')
+        .field('ies_ens_superior.cod_municipio_ies', 'city_id')
+        .field('ies_ens_superior.cod_uf_ies/10', 'region_id');
+    next();
+
+}, query, response('university'));
+
+universityApp.get('/count', rqfCount.parse(), (req, res, next) => {
+    req.sql.field('COUNT(*)', 'total')
+    .field("'Brasil'", 'name')
+    .field('ies_ens_superior.ano_censo', 'year')
+    .from('ies_ens_superior')
+    .group('ies_ens_superior.ano_censo')
+    .order('ies_ens_superior.ano_censo')
+
+    next();
+}, rqfCount.build(), query, addMissing(rqfCount), id2str.transform(), response('university'));
+
+module.exports = universityApp;
diff --git a/src/libs/routes_v2/universityEnrollment.js b/src/libs/routes_v2/universityEnrollment.js
new file mode 100644
index 00000000..8df5aded
--- /dev/null
+++ b/src/libs/routes_v2/universityEnrollment.js
@@ -0,0 +1,848 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const universityEnrollmentApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+universityEnrollmentApp.get('/years', (req, res, next) => {
+    req.sql.from('aluno_ens_superior')
+    .field('DISTINCT aluno_ens_superior.ano_censo', 'year')
+		.where('aluno_ens_superior.ano_censo > 2010');
+    next();
+}, query, response('years'));
+
+universityEnrollmentApp.get('/year_range', (req, res, next) => {
+    req.sql.from('aluno_ens_superior')
+    .field('MIN(aluno_ens_superior.ano_censo)', 'start_year')
+    .field('MAX(aluno_ens_superior.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+universityEnrollmentApp.get('/academic_organization', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.academicOrganization(i)
+        });
+    };
+    next();
+}, response('academic_organization'));
+
+universityEnrollmentApp.get('/upper_adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 7; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperAdmDependency(i)
+        });
+    };
+    next();
+}, response('upper_adm_dependency'));
+
+universityEnrollmentApp.get('/ocde_geral', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <= 8; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ocdeGeral(i)
+        });
+    };
+    next();
+}, response('ocde_geral'));
+
+
+universityEnrollmentApp.get('/finish', (req, res, next) => {
+    req.result = []
+    for (let i = 0; i <= 1; ++i){
+        req.result.push({
+            id: i,
+            name: id2str.finishUniversity(i)
+        })
+    }
+    next();
+
+}, response('finish'));
+
+universityEnrollmentApp.get('/ocde_specific', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 1; i <= 86; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.ocdeSpecific(i)
+        };
+        if (obj.name !== id2str.ocdeSpecific(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.ocdeSpecific(defaultCase)
+    });
+    next();
+}, response('ocde_specific'));
+
+universityEnrollmentApp.get('/ocde_detailed', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 142; i <= 863; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.ocdeDetailed(i)
+        };
+        if (obj.name !== id2str.ocdeDetailed(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.ocdeDetailed(defaultCase)
+    });
+    next();
+}, response('ocde_detailed'));
+
+universityEnrollmentApp.get('/cine_geral', (req, res, next) => {
+    req.result = [];
+    for(let i = 0; i <= 10; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.cineGeral(i)
+        });
+    };
+    next();
+}, response('cine_geral'));
+
+universityEnrollmentApp.get('/cine_specific', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 1; i <= 104; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.cineSpecific(i)
+        };
+        if (obj.name !== id2str.cineSpecific(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.cineSpecific(defaultCase)
+    });
+    next();
+}, response('cine_specific'));
+
+universityEnrollmentApp.get('/cine_detailed', (req, res, next) => {
+    req.result = [];
+    const defaultCase = null;
+    for(let i = 11; i <= 1041; ++i) {
+        let obj = {
+            id: i,
+            name: id2str.cineDetailed(i)
+        };
+        if (obj.name !== id2str.cineDetailed(defaultCase)){
+            req.result.push(obj);
+        }
+    };
+    req.result.push({
+        id: defaultCase,
+        name: id2str.cineDetailed(defaultCase)
+    });
+    next();
+}, response('cine_detailed'));
+
+universityEnrollmentApp.get('/upper_turn', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.upperTurn("null")
+    }];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperTurn(i)
+        });
+    };
+    next();
+}, response('upper_turn'));
+
+universityEnrollmentApp.get('/student_deficiency', (req, res, next) => {
+    req.result = [{
+        id: 9,
+        name: id2str.studentDeficiency(9)
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.studentDeficiency(i)
+        });
+    };
+    next();
+}, response('student_deficiency'));
+
+universityEnrollmentApp.get('/ethnic_group_ies', (req, res, next) => {
+    req.result = [{
+        id: 9,
+        name: id2str.ethnicGroupIES(9)
+    }];
+    for(let i = 0; i <=5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ethnicGroupIES(i)
+        });
+    }
+    next();
+}, response('ethnic_group_ies'));
+
+universityEnrollmentApp.get('/school_type', (req, res, next) => {
+    req.result = [{
+        id: 9,
+        name: id2str.schoolType(9)
+    }];
+    for(let i = 1; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.schoolType(i)
+        });
+    };
+    next();
+}, response('school_type'));
+
+universityEnrollmentApp.get('/university', (req, res, next) => {
+    req.sql.from('aluno_ens_superior')
+    .field('DISTINCT aluno_ens_superior.nome_ies', 'nome')
+    .field('aluno_ens_superior.cod_ies', 'cod')
+    next();
+}, query, response('university'));
+
+universityEnrollmentApp.get('/academic_level', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.academicLevel(i)
+        });
+    };
+    next();
+}, response('academic_level'));
+
+universityEnrollmentApp.get('/gender_ies', function (req, res, next) {
+    req.result = [];
+    for (var i = 1; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.genderIES(i)
+        });
+    };
+    next();
+}, response('gender_ies'));
+
+universityEnrollmentApp.get('/upper_education_mod', function (req, res, next) {
+    req.result = [];
+    for (var i = 1; i <= 3; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperEducationMod(i)
+        });
+    };
+    next();
+}, response('upper_education_mod'));
+
+universityEnrollmentApp.get('/age_student_code', function (req, res, next) {
+    req.result = [];
+    for (var i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ageStudentCode(i)
+        });
+    };
+    next();
+}, response('age_student_code'));
+
+
+universityEnrollmentApp.get('/enter_situation/student_enter_situation', function (req, res, next) {
+    req.result = [];
+    for (var i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.enterSituation(i)
+        });
+    };
+    next();
+}, response('student_enter_situation'));
+
+universityEnrollmentApp.get('/enrollment_situation/student_enter_situation', function (req, res, next) {
+    req.result = [];
+    for (var i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.enterSituation(i)
+        });
+    };
+    next();
+}, response('student_enter_situation'));
+
+
+universityEnrollmentApp.get('/enter_situation/student_enrollment_situation', function (req, res, next) {
+    req.result = [];
+    for (var i = 1; i <= 3; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.enrollmentSituation(i)
+        });
+    };
+    next();
+}, response('student_enrollment_situation'));
+
+universityEnrollmentApp.get('/university', (req, res, next) => {
+    req.sql.from('aluno_ens_superior')
+    .field('DISTINCT aluno_ens_superior.nome_ies', 'nome')
+    .field('aluno_ens_superior.cod_ies', 'cod')
+    next();
+}, query, response('university'));
+
+universityEnrollmentApp.get('/localoffer', (req, res, next) => {
+    req.sql.from('localoferta_ens_superior', 'l')
+    .field('DISTINCT l.nome', 'localoffer_name')
+    .field('l.cod_local_oferta', 'localoffer_id');
+    next();
+}, query, response('localoffer'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localoferta_cod_municipio',
+        table: 'localoferta_ens_superior_matricula'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'localoferta_cod_municipio',
+        foreignTable: 'localoferta_ens_superior_matricula'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localoferta_cod_uf',
+        table: 'localoferta_ens_superior_matricula'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'localoferta_cod_uf',
+        foreignTable: 'localoferta_ens_superior_matricula'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'localoferta_cod_regiao',
+        table: 'localoferta_ens_superior_matricula'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'localoferta_cod_regiao',
+        foreignTable: 'localoferta_ens_superior_matricula'
+    }
+}).addValue({
+    name: 'localoffer',
+    table: 'localoferta_ens_superior_matricula',
+    tableField: ['cod_local_oferta', 'localoferta_nome'],
+    resultField: ['localoffer_id', 'localoffer_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_local_oferta'
+    }
+}).addValue({
+    name: 'campi',
+    table: 'localoferta_ens_superior_matricula',
+    tableField: ['cod_local_oferta', 'localoferta_nome'],
+    resultField: ['campi_id', 'campi_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_local_oferta'
+    }
+}).addValue({
+    name: 'university',
+    table: '@',
+    tableField: ['cod_ies', 'nome_ies'],
+    resultField: ['university_id', 'university_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies'
+    }
+}).addValue({
+    name: 'universityLocalOffer',
+    table: '@',
+    tableField: ['cod_ies', 'nome_ies'],
+    resultField: ['university_id', 'university_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies'
+    }
+}).addValue({
+    name:'upper_adm_dependency',
+    table: '@',
+    tableField: 'par_categoria_administrativa',
+    resultField: 'upper_adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_categoria_administrativa'
+    }
+}).addValue({
+    name:'academic_organization',
+    table: '@',
+    tableField: 'cod_organizacao_academica',
+    resultField: 'academic_organization_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_organizacao_academica'
+    }
+}).addValue({
+    name:'ocde_specific',
+    table: '@',
+    tableField: ['par_cod_ocde_area_especifica'],
+    resultField: ['ocde_specific_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_cod_ocde_area_especifica'
+    }
+}).addValue({
+    name:'ocde_geral',
+    table: '@',
+    tableField: ['par_cod_ocde_area_geral'],
+    resultField: ['ocde_geral_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_cod_ocde_area_geral'
+    }
+}).addValue({
+    name:'ocde_detailed',
+    table: '@',
+    tableField: ['par_cod_ocde_area_detalhada'],
+    resultField: ['ocde_detailed_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_cod_ocde_area_detalhada'
+    }
+}).addValue({
+    name:'cine_specific',
+    table: '@',
+    tableField: ['cod_cine_area_especifica', 'nome_cine_area_especifica'],
+    resultField: ['cine_specific_id', 'cine_specific_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cine_area_especifica'
+    }
+}).addValue({
+    name:'cine_geral',
+    table: '@',
+    tableField: ['cod_cine_area_geral', 'nome_cine_area_geral'],
+    resultField: ['cine_geral_id', 'cine_geral_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cine_area_geral'
+    }
+}).addValue({
+    name:'cine_detailed',
+    table: '@',
+    tableField: ['cod_cine_area_detalhada', 'nome_cine_area_detalhada'],
+    resultField: ['cine_detailed_id', 'cine_detailed_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cine_area_detalhada'
+    }
+}).addValue({
+    name:'academic_level',
+    table: '@',
+    tableField: 'cod_grau_academico',
+    resultField: 'academic_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_grau_academico'
+    }
+}).addValue({
+    name:'upper_education_mod',
+    table: '@',
+    tableField: 'cod_modalidade_ensino',
+    resultField: 'upper_education_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_modalidade_ensino'
+    }
+}).addValue({
+    name:'is_free',
+    table: '@',
+    tableField: 'gratuito',
+    resultField: 'is_free_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'gratuito'
+    }
+}).addValue({
+    name:'night_time',
+    table: '@',
+    tableField: 'noturno_curso_t',
+    resultField: 'night_time_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'noturno_curso_t'
+    }
+}).addValue({
+    name:'situation',
+    table: '@',
+    tableField: 'cod_situacao_curso',
+    resultField: 'situacao_curso_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_situacao_curso'
+    }
+}).addValue({
+    name:'finish',
+    table: '@',
+    tableField: 'concluinte',
+    resultField: 'finish_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'concluinte'
+    }
+}).addValue({
+    name:'year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'age_student_code',
+    table: '@',
+    tableField: 'idade_aluno_codigo',
+    resultField: 'age_student_code_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'idade_aluno_codigo'
+    }
+}).addValue({
+    name:'upper_turn',
+    table: '@',
+    tableField: 'cod_turno_aluno',
+    resultField: 'upper_turn_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_turno_aluno'
+    }
+}).addValue({
+    name:'ethnic_group_ies',
+    table: '@',
+    tableField: 'par_cod_cor_raca_aluno',
+    resultField: 'ethnic_group_ies_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_cod_cor_raca_aluno'
+    }
+}).addValue({
+    name:'student_deficiency',
+    table: '@',
+    tableField: 'par_aluno_deficiencia_transtorno_superdotacao',
+    resultField: 'student_deficiency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_aluno_deficiencia_transtorno_superdotacao'
+    }
+}).addValue({
+    name:'school_type',
+    table: '@',
+    tableField: 'par_tipo_escola_ensino_medio',
+    resultField: 'school_type_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_tipo_escola_ensino_medio'
+    }
+}).addValue({
+    name: 'gender_ies',
+    table: '@',
+    tableField: 'par_genero_aluno',
+    resultField: 'gender_ies_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_genero_aluno'
+    }
+}).addValue({
+    name: 'mesoregion',
+    table: 'municipio',
+    tableField: ['nome_mesorregiao', 'mesorregiao_id'],
+    resultField: ['mesoregion_name', 'mesoregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'mesorregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'localoferta_cod_municipio',
+        foreignTable: 'localoferta_ens_superior_matricula'
+    }
+}).addValue({
+    name: 'microregion',
+    table: 'municipio',
+    tableField: ['nome_microrregiao', 'microrregiao_id'],
+    resultField: ['microregion_name', 'microregion_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'microrregiao_id',
+        table: 'municipio'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'localoferta_cod_municipio',
+        foreignTable: 'localoferta_ens_superior_matricula'
+    }
+}).addValue({
+    name: 'course',
+    table: 'curso_ens_superior',
+    tableField: 'nome_curso',
+    resultField: 'course_name',
+    where:{
+        relation: '=',
+        type: 'string',
+        table:  'curso_ens_superior',
+        field: 'nome_curso'
+    },
+    join:{
+        primary: ['ano_censo', 'cod_curso'],
+        foreign: ['ano_censo', 'cod_curso'],
+        foreignTable: 'localoferta_ens_superior_matricula'
+    }
+});
+
+universityEnrollmentApp.get('/', rqf.parse(), (req, res, next) => {
+    if ("localoffer" in req.dims) {
+        if ("university" in req.dims || "universityLocalOffer" in req.dims) {
+            req.sql.from('localoferta_ens_superior_matricula')
+                .field('curso_ens_superior.ano_censo', 'year')
+                .field('COUNT(localoferta_ens_superior.cod_local_oferta)', 'total')
+                .group('localoferta_ens_superior_matricula.ano_censo')
+                .where('localoferta_ens_superior_matricula.cod_aluno_situacao = 2 OR localoferta_ens_superior_matricula.cod_aluno_situacao = 6 OR localoferta_ens_superior_matricula.matriculado = 1')
+                .where('localoferta_ens_superior_matricula.cod_nivel_academico = 1')
+                .order('localoferta_ens_superior_matricula.ano_censo')
+                .order('localoferta_ens_superior.cod_local_oferta');
+        } else {
+            req.sql.from('localoferta_ens_superior_matricula')
+                .field('localoferta_ens_superior_matricula.ano_censo', 'year')
+                .field('COUNT(*)', 'total')
+                .field('localoferta_ens_superior_matricula.cod_ies', 'university_id')
+                .field('localoferta_ens_superior_matricula.nome_ies', 'university_name')
+                .where('localoferta_ens_superior_matricula.cod_aluno_situacao = 2 OR localoferta_ens_superior_matricula.cod_aluno_situacao = 6 OR localoferta_ens_superior_matricula.matriculado = 1')
+                .where('localoferta_ens_superior_matricula.cod_nivel_academico = 1')
+                .group('localoferta_ens_superior_matricula.ano_censo')
+                .group('localoferta_ens_superior_matricula.cod_ies')
+                .group('localoferta_ens_superior_matricula.nome_ies')
+                .order('localoferta_ens_superior_matricula.ano_censo')
+                .order('localoferta_ens_superior_matricula.cod_local_oferta');
+        }
+    } else if (("state" in req.dims) || ("city" in req.dims) || ("region" in req.dims) || ("mesoregion" in req.dims) || ("microregion" in req.dims) ||
+        ("state" in req.filter) || ("city" in req.filter) || ("region" in req.filter) || ("mesoregion" in req.filter) || ("microregion" in req.filter)) {
+        req.sql.from('localoferta_ens_superior_matricula')
+        .field('DISTINCT COUNT(*)', 'total')
+        .field("'Brasil'", 'name')
+        .field('localoferta_ens_superior_matricula.ano_censo', 'year')
+        .where('localoferta_ens_superior_matricula.cod_aluno_situacao = 2 OR localoferta_ens_superior_matricula.cod_aluno_situacao = 6 OR localoferta_ens_superior_matricula.matriculado = 1')
+        .where('localoferta_ens_superior_matricula.cod_nivel_academico = 1')
+        .group('localoferta_ens_superior_matricula.ano_censo')
+        .order('localoferta_ens_superior_matricula.ano_censo')
+    } else if ("university" in req.dims || "universityLocalOffer" in req.dims) {
+        req.sql.from('aluno_ens_superior')
+        .field('COUNT(*)', 'total')
+        .field("'Brasil'", 'name')
+        .field('aluno_ens_superior.ano_censo', 'year')
+        .group('aluno_ens_superior.cod_ies')
+        .group('aluno_ens_superior.ano_censo')
+        .where('aluno_ens_superior.cod_aluno_situacao = 2 OR aluno_ens_superior.cod_aluno_situacao = 6 OR aluno_ens_superior.matriculado = 1')
+        .where('aluno_ens_superior.cod_nivel_academico = 1')
+        .order('aluno_ens_superior.cod_ies')
+        .order('aluno_ens_superior.ano_censo')
+    } else {
+        req.sql.from('localoferta_ens_superior_matricula')
+        .field('COUNT(*)', 'total')
+        .field("'Brasil'", 'name')
+        .field('localoferta_ens_superior_matricula.ano_censo', 'year')
+        .where('localoferta_ens_superior_matricula.cod_aluno_situacao = 2 OR localoferta_ens_superior_matricula.cod_aluno_situacao = 6 OR localoferta_ens_superior_matricula.matriculado = 1')
+        .where('localoferta_ens_superior_matricula.cod_nivel_academico = 1')
+        .group('localoferta_ens_superior_matricula.ano_censo')
+        .order('localoferta_ens_superior_matricula.ano_censo')
+    }
+    next();
+}, rqf.build(), query, (req, res, next) =>{ console.log(req.sql.toString()); next()}, id2str.transform(), addMissing(rqf), (req, res, next) => {
+    if ('course' in req.dims){
+        var total_course = req.result.reduce((total, cur) => {return total += cur.total}, 0)
+        for (var course of req.result){
+            course.percentage = Number((( course.total / total_course ) * 100).toFixed(2))
+        }
+    }
+    next();
+}, response('universityEnrollment'));
+
+universityEnrollmentApp.get('/enter_situation', rqf.parse(), (req, res, next) => {
+    req.sql.from('localoferta_ens_superior_matricula')
+    .field('SUM(CASE WHEN localoferta_ens_superior_matricula.cod_aluno_situacao=2 AND localoferta_ens_superior_matricula.ingressante=1 THEN 1 ELSE 0 END)', 'cursando')
+    .field('SUM(CASE WHEN localoferta_ens_superior_matricula.cod_aluno_situacao=6 AND localoferta_ens_superior_matricula.ingressante=1 THEN 1 ELSE 0 END)', 'concluinte')
+    .field('SUM(CASE WHEN (localoferta_ens_superior_matricula.cod_aluno_situacao=4 OR localoferta_ens_superior_matricula.cod_aluno_situacao=5 OR localoferta_ens_superior_matricula.cod_aluno_situacao=7) AND localoferta_ens_superior_matricula.ingressante=1 THEN 1 ELSE 0 END)', 'evadido')
+    .field('SUM(CASE WHEN localoferta_ens_superior_matricula.cod_aluno_situacao=3 AND localoferta_ens_superior_matricula.ingressante=1 THEN 1 ELSE 0 END)', 'trancado')
+    .field('COUNT(*)', 'total')
+    .field("'Brasil'", 'name')
+    .field('localoferta_ens_superior_matricula.ano_censo', 'year')
+    .where('localoferta_ens_superior_matricula.cod_nivel_academico=1')
+    .where('localoferta_ens_superior_matricula.cod_grau_academico=2 OR localoferta_ens_superior_matricula.cod_grau_academico=4')
+    .group('localoferta_ens_superior_matricula.ano_censo')
+    .order('localoferta_ens_superior_matricula.ano_censo')
+    next()
+}, rqf.build(), query, (req, res, next) => {
+    for (var res of req.result){
+        res.cursando = Number(res.cursando);
+        res.concluinte = Number(res.concluinte);
+        res.evadido = Number(res.evadido);
+        res.trancado = Number(res.trancado);
+        res.total = res.cursando + res.concluinte + res.evadido + res.trancado
+        res.taxa_evasao = Number( ((res.evadido/res.total) * 100).toFixed(2) )
+    }
+    next();
+}, id2str.transform(), response('enterSituation'));
+
+universityEnrollmentApp.get('/enrollment_situation', rqf.parse(), (req, res, next) => {
+    req.sql.from('localoferta_ens_superior_matricula')
+    .field('SUM(CASE WHEN localoferta_ens_superior_matricula.cod_aluno_situacao=2 THEN 1 ELSE 0 END)', 'cursando')
+    .field('SUM(CASE WHEN localoferta_ens_superior_matricula.cod_aluno_situacao=6 THEN 1 ELSE 0 END)', 'concluinte')
+    .field('SUM(CASE WHEN (localoferta_ens_superior_matricula.cod_aluno_situacao=4 OR localoferta_ens_superior_matricula.cod_aluno_situacao=5 OR localoferta_ens_superior_matricula.cod_aluno_situacao=7) THEN 1 ELSE 0 END)', 'evadido')
+    .field('SUM(CASE WHEN localoferta_ens_superior_matricula.cod_aluno_situacao=3 THEN 1 ELSE 0 END)', 'trancado')
+    .field('localoferta_ens_superior_matricula.ano_censo', 'year')
+    .field("'Brasil'", 'name')
+    .where('localoferta_ens_superior_matricula.cod_nivel_academico=1')
+    .where('localoferta_ens_superior_matricula.cod_grau_academico=2 OR localoferta_ens_superior_matricula.cod_grau_academico=4')
+    .group('localoferta_ens_superior_matricula.ano_censo')
+    .order('localoferta_ens_superior_matricula.ano_censo')
+    next()
+}, rqf.build(),query, (req, res, next) => {
+    for (var res of req.result){
+        res.cursando = Number(res.cursando);
+        res.concluinte = Number(res.concluinte);
+        res.evadido = Number(res.evadido);
+        res.trancado = Number(res.trancado);
+        res.total = res.cursando + res.concluinte + res.evadido + res.trancado
+    }
+    
+    next();
+}, id2str.transform(), response('enrollmentSituation'));
+
+
+
+module.exports = universityEnrollmentApp;
diff --git a/src/libs/routes_v2/universityLocalOffer.js b/src/libs/routes_v2/universityLocalOffer.js
new file mode 100644
index 00000000..535094c4
--- /dev/null
+++ b/src/libs/routes_v2/universityLocalOffer.js
@@ -0,0 +1,170 @@
+''/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const universityLocalOfferApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const request = require(`request`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+let rqf = new ReqQueryFields();
+
+universityLocalOfferApp.use(cache('15 day'));
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_municipio',
+        table: 'localoferta_ens_superior'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio',
+        foreignTable: 'localoferta_ens_superior'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: 'LIKE',
+        type: 'string',
+        field: 'id'
+    },
+    join: {
+        primary: 'nome',
+        foreign: 'nome_regiao_ies',
+        foreignTable: 'localoferta_ens_superior'
+    }
+
+}).addValue({
+    name: 'min_year',
+    table: 'localoferta_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: 'localoferta_ens_superior',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: 'localoferta_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: 'localoferta_ens_superior',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_uf',
+        table: 'localoferta_ens_superior'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_uf',
+        foreignTable: 'localoferta_ens_superior'
+    }
+}).addValueToField({
+    name: 'university',
+    table: 'localoferta_ens_superior',
+    tableField: 'cod_ies',
+    resultField: 'university_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies',
+        table: 'localoferta_ens_superior'
+    }
+}, 'filter').addValue({
+    name: 'year',
+    table: 'localoferta_ens_superior',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'localoferta_ens_superior',
+        field: 'ano_censo'
+    }
+});
+
+
+universityLocalOfferApp.get('/', rqf.parse(),  rqf.build(), (req, res, next) => {
+    req.sql.from('localoferta_ens_superior')
+        .field('distinct localoferta_ens_superior.cod_ies', 'id')
+        .field('localoferta_ens_superior.cod_local_oferta', 'localoffer_id')
+        .field('localoferta_ens_superior.ano_censo', 'year')
+        .field('ies_ens_superior.nome_ies', 'name')
+        .field('localoferta_ens_superior.nome', 'localoffer_name')
+        .field('localoferta_ens_superior.cod_uf', 'state_id')
+        .field('localoferta_ens_superior.cod_municipio', 'city_id')
+        .field('localoferta_ens_superior.cod_regiao', 'region_id')
+        .join('ies_ens_superior', null, 'localoferta_ens_superior.cod_ies=ies_ens_superior.cod_ies AND localoferta_ens_superior.ano_censo=ies_ens_superior.ano_censo')
+        .where('localoferta_ens_superior.nome IS NOT NULL AND ies_ens_superior.nome_ies IS NOT NULL');
+    next();
+
+}, query, response('universityLocalOfferApp'));
+
+module.exports = universityLocalOfferApp;
diff --git a/src/libs/routes_v2/universityTeacher.js b/src/libs/routes_v2/universityTeacher.js
new file mode 100644
index 00000000..f65a29b2
--- /dev/null
+++ b/src/libs/routes_v2/universityTeacher.js
@@ -0,0 +1,517 @@
+/*
+Copyright (C) 2016 Centro de Computacao Cientifica e Software Livre
+Departamento de Informatica - Universidade Federal do Parana - C3SL/UFPR
+
+This file is part of simcaq-node.
+
+simcaq-node is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+simcaq-node is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with simcaq-node.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+const express = require('express');
+
+const teacherEnrollmentApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const squel = require('squel');
+
+const query = require(`${libs}/middlewares/query`).query;
+
+const response = require(`${libs}/middlewares/response`);
+
+const ReqQueryFields = require(`${libs}/middlewares/reqQueryFields`);
+
+const id2str = require(`${libs}/middlewares/id2str`);
+
+const addMissing = require(`${libs}/middlewares/addMissing`);
+
+const config = require(`${libs}/config`);
+
+const cache = require('apicache').options({ debug: config.debug, statusCodes: {include: [200]} }).middleware;
+
+let rqf = new ReqQueryFields();
+
+teacherEnrollmentApp.get('/years', (req, res, next) => {
+    req.sql.from('docente_ens_superior')
+    .field('DISTINCT docente_ens_superior.ano_censo', 'year');
+    next();
+}, query, response('years'));
+
+teacherEnrollmentApp.get('/year_range', (req, res, next) => {
+    req.sql.from('docente_ens_superior')
+    .field('MIN(docente_ens_superior.ano_censo)', 'start_year')
+    .field('MAX(docente_ens_superior.ano_censo)', 'end_year');
+    next();
+}, query, response('range'));
+
+teacherEnrollmentApp.get('/academic_organization', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.academicOrganization(i)
+        });
+    };
+    next();
+}, response('academic_organization'));
+
+teacherEnrollmentApp.get('/upper_adm_dependency', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 7; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.upperAdmDependency(i)
+        });
+    };
+    next();
+}, response('upper_adm_dependency'));
+
+teacherEnrollmentApp.get('/teacher_situation', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 6; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.teacherSituation(i)
+        });
+    };
+    next();
+}, response('teacher_situation'));
+
+teacherEnrollmentApp.get('/work_regime', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.workRegime("null")
+    }];
+    for(let i = 1; i <= 4; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.workRegime(i)
+        });
+    };
+    next();
+}, response('work_regime'));
+
+teacherEnrollmentApp.get('/substitute', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.booleanVariable("null")
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('substitute'));
+
+teacherEnrollmentApp.get('/visitor', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.booleanVariable("null")
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('visitor'));
+
+teacherEnrollmentApp.get('/ead_teacher', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.booleanVariable("null")
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('ead_teacher'));
+
+teacherEnrollmentApp.get('/graduation_presential', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.booleanVariable("null")
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('graduation_presential'));
+
+teacherEnrollmentApp.get('/postgraduate_ead_teacher', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.booleanVariable("null")
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('postgraduate_ead_teacher'));
+
+teacherEnrollmentApp.get('/postgraduate_presential_teacher', (req, res, next) => {
+    req.result = [{
+        id: "null",
+        name: id2str.booleanVariable("null")
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.booleanVariable(i)
+        });
+    };
+    next();
+}, response('postgraduate_presential_teacher'));
+
+teacherEnrollmentApp.get('/deficiency', (req, res, next) => {
+    req.result = [{
+        id: 9,
+        name: id2str.studentDeficiency(9)
+    }];
+    for(let i = 0; i <= 1; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.studentDeficiency(i)
+        });
+    };
+    next();
+}, response('deficiency'));
+
+teacherEnrollmentApp.get('/ethnic_group_teacher_ies', (req, res, next) => {
+    req.result = [{
+        id: 9,
+        name: id2str.ethnicGroupTeacherIES(9)
+    }];
+    for(let i = 0; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.ethnicGroupTeacherIES(i)
+        });
+    };
+    next();
+}, response('ethnic_group_teacher_ies'));
+
+teacherEnrollmentApp.get('/teacher_schooling', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 5; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.teacherSchooling(i)
+        });
+    };
+    next();
+}, response('teacher_schooling'));
+
+teacherEnrollmentApp.get('/gender_ies', (req, res, next) => {
+    req.result = [];
+    for(let i = 1; i <= 2; ++i) {
+        req.result.push({
+            id: i,
+            name: id2str.genderIES(i)
+        });
+    };
+    next();
+}, response('gender_ies'));
+
+
+rqf.addField({
+    name: 'filter',
+    field: false,
+    where: true
+}).addField({
+    name: 'dims',
+    field: true,
+    where: false
+}).addValue({
+    name: 'min_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '>=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'max_year',
+    table: '@',
+    tableField: 'ano_censo',
+    resultField: 'year',
+    where: {
+        relation: '<=',
+        type: 'integer',
+        table: '@',
+        field: 'ano_censo'
+    }
+}).addValue({
+    name: 'state',
+    table: 'estado',
+    tableField: ['nome', 'id'],
+    resultField: ['state_name', 'state_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_uf_ies',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_uf_ies',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'city',
+    table: 'municipio',
+    tableField: ['nome', 'id'],
+    resultField: ['city_name', 'city_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_municipio_ies',
+        table: '@'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_municipio_ies',
+        foreignTable: '@'
+    }
+}).addValue({
+    name: 'region',
+    table: 'regiao',
+    tableField: ['nome', 'id'],
+    resultField: ['region_name', 'region_id'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'id'
+    },
+    join: {
+        primary: 'id',
+        foreign: 'cod_regiao_ies',
+        foreignTable: 'docente_ens_superior'
+    }
+}).addValue({
+    name: 'university',
+    table: 'docente_ens_superior',
+    tableField: ['cod_ies', 'nome_ies'],
+    resultField: ['university_id', 'university_name'],
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_ies'
+    }
+}).addValue({
+    name: 'upper_adm_dependency',
+    table: 'docente_ens_superior',
+    tableField: 'par_categoria_administrativa',
+    resultField: 'upper_adm_dependency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'docente_ens_superior',
+        field: 'par_categoria_administrativa'
+    }
+}).addValue({
+    name: 'academic_organization',
+    table: 'docente_ens_superior',
+    tableField: 'cod_organizacao_academica',
+    resultField: 'academic_organization_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        table: 'docente_ens_superior',
+        field: 'cod_organizacao_academica'
+    }
+}).addValue({
+    name:'academic_level',
+    table: 'docente_ens_superior',
+    tableField: 'cod_grau_academico',
+    resultField: 'academic_level_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_grau_academico'
+    }
+}).addValue({
+    name:'upper_education_mod',
+    table: 'docente_ens_superior',
+    tableField: 'cod_modalidade_ensino',
+    resultField: 'upper_education_mod_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_modalidade_ensino'
+    }
+}).addValue({
+    name:'teacher_situation',
+    table: 'docente_ens_superior',
+    tableField: 'par_situacao_docente',
+    resultField: 'teacher_situation_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_situacao_docente'
+    }
+}).addValue({
+    name:'work_regime',
+    table: 'docente_ens_superior',
+    tableField: 'cod_regime_trabalho',
+    resultField: 'work_regime_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_regime_trabalho'
+    }
+}).addValue({
+    name:'substitute',
+    table: 'docente_ens_superior',
+    tableField: 'docente_substituto',
+    resultField: 'substitute_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'docente_substituto'
+    }
+}).addValue({
+    name:'visitor',
+    table: 'docente_ens_superior',
+    tableField: 'docente_visitante',
+    resultField: 'visitor_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'docente_visitante'
+    }
+}).addValue({
+    name:'ead_teacher',
+    table: 'docente_ens_superior',
+    tableField: 'ministra_aula_ead',
+    resultField: 'ead_teacher_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'ministra_aula_ead'
+    }
+}).addValue({
+    name:'graduation_presential',
+    table: 'docente_ens_superior',
+    tableField: 'atua_atividade_graduacao_presencial',
+    resultField: 'graduation_presential_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'atua_atividade_graduacao_presencial'
+    }
+}).addValue({
+    name:'postgraduate_ead_teacher',
+    table: 'docente_ens_superior',
+    tableField: 'atua_atividade_posgraduacao_distancia',
+    resultField: 'postgraduate_ead_teacher_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'atua_atividade_posgraduacao_distancia'
+    }
+}).addValue({
+    name:'postgraduate_presential_teacher',
+    table: 'docente_ens_superior',
+    tableField: 'atua_atividade_posgraduacao_presencial',
+    resultField: 'postgraduate_presential_teacher_id',
+    where: {
+        relation: '=',
+        type: 'boolean',
+        field: 'atua_atividade_posgraduacao_presencial'
+    }
+}).addValue({
+    name:'teacher_schooling',
+    table: 'docente_ens_superior',
+    tableField: 'cod_escolaridade_docente',
+    resultField: 'teacher_schooling_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_escolaridade_docente'
+    }
+}).addValue({
+    name:'ethnic_group_teacher_ies',
+    table: 'docente_ens_superior',
+    tableField: 'cod_cor_raca_docente',
+    resultField: 'ethnic_group_teacher_ies_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'cod_cor_raca_docente'
+    }
+}).addValue({
+    name:'gender_ies',
+    table: 'docente_ens_superior',
+    tableField: 'sexo_docente',
+    resultField: 'gender_ies_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'sexo_docente'
+    }
+}).addValue({
+    name:'deficiency',
+    table: 'docente_ens_superior',
+    tableField: 'par_docente_deficiencia',
+    resultField: 'deficiency_id',
+    where: {
+        relation: '=',
+        type: 'integer',
+        field: 'par_docente_deficiencia'
+    }
+});
+
+teacherEnrollmentApp.get('/', rqf.parse(), (req, res, next) => {
+
+    if ("university" in req.dims) {
+        req.sql.field('COUNT(*)', 'total')
+        .field("'Brasil'", 'name')
+        .field('docente_ens_superior.ano_censo', 'year')
+        .from('docente_ens_superior')
+        .group('docente_ens_superior.cod_ies')
+        .group('docente_ens_superior.ano_censo')
+        .order('docente_ens_superior.cod_ies')
+        .order('docente_ens_superior.ano_censo')
+    }
+    else {
+        req.sql.field('COUNT(*)', 'total')
+        .field("'Brasil'", 'name')
+        .field('docente_ens_superior.ano_censo', 'year')
+        .from('docente_ens_superior')
+        .group('docente_ens_superior.ano_censo')
+        .order('docente_ens_superior.ano_censo')
+    }
+
+   next();
+},  rqf.build(), query, addMissing(rqf), id2str.transform(false), response('teacherEnrollment'));
+
+module.exports = teacherEnrollmentApp;
diff --git a/src/libs/routes_v2/user.js b/src/libs/routes_v2/user.js
new file mode 100644
index 00000000..7ff088ee
--- /dev/null
+++ b/src/libs/routes_v2/user.js
@@ -0,0 +1,305 @@
+const express = require('express');
+
+const userApp = express();
+
+const libs = `${process.cwd()}/libs`;
+
+const config = require(`${libs}/config`);
+
+const log = require(`${libs}/log`)(module);
+
+const User = require(`${libs}/models/user`);
+
+const VerificationToken = require(`${libs}/models/verificationToken`);
+
+const ResetToken = require(`${libs}/models/resetToken`);
+
+const response = require(`${libs}/middlewares/response`);
+
+const email = require(`${libs}/middlewares/email`);
+
+const passport = require('passport');
+
+function emailSyntax(email) {
+  const regex = /^(([^<>()\[\]\.,;:\s@\"]+(\.[^<>()\[\]\.,;:\s@\"]+)*)|(\".+\"))@(([^<>()[\]\.,;:\s@\"]+\.)+[^<>()[\]\.,;:\s@\"]{2,})$/i;
+  return regex.test(email);
+}
+
+userApp.get('/schooling', (req, res, next) => {
+  req.result = [
+    'Não estudou',
+    'Ensino Fundamental Incompleto',
+    'Ensino Fundamental Completo',
+    'Ensino Médio',
+    'Graduação',
+    'Mestrado',
+    'Doutorado'
+  ];
+  next();
+}, response('schooling'));
+
+userApp.get('/segment', (req, res, next) => {
+  req.result = [
+    'Gestores e equipe gestora das secretarias e ministério da Educação',
+    'Gestores dos órgãos de planejamento e finanças (das três esferas de governo)',
+    'Agentes do poder legislativo',
+    'Agentes dos conselhos de educação',
+    'Profissionais da educação',
+    'Sindicato',
+    'Sociedade civil interessada no financiamento da Educação Básica de qualidade',
+    'Comunidade acadêmica',
+    'Imprensa',
+    'Outro [citar segmento]'
+  ];
+  next();
+}, response('segment'));
+
+userApp.get('/role', (req, res, next) => {
+  req.result = [
+    {"Gestores e equipe gestora das secretarias e ministério da Educação" : ["Dirigente municipal, estadual e federal", "Secretário do MEC", "Servidor da área de planejamento educacional", "Membro de associação de gestores (Ex. Undime, Consed, etc)", "Outro [citar função]"]},
+    {"Gestores dos órgãos de planejamento e finanças (das três esferas de governo)" : ["Equipe gestora dos órgãos de planejamento", "Equipe gestora dos órgãos de finanças", "Outro [citar função]"]},
+    {"Agentes do poder legislativo" : ["Parlamentar", "Assessor/a parlamentar", "Auditor/a dos tribunais de conta", "Conselheiro/a de tribunais de conta.", "Outro [citar função]"]},
+    {"Agentes dos conselhos de educação" : ["Conselheiro/a municipais, estaduais e federais", "Conselheiro/a do Fundeb", "Outro [citar função]"]},
+    {"Profissionais da educação" : ["Professor/a da Educação Básica", "Profissional da educação não-docente", "Outro [citar função]"]},
+    {"Sindicato" : ["Agente de sindicatos"]},
+    {"Sociedade civil interessada no financiamento da Educação Básica de qualidade" : ["Membro de fóruns educacionais", "Membro de ONGs e demais entidades sem fins lucrativos", "Estudante da educação básica e membro de entidades estudantis", "Pais e membros de entidades de pais", "Outro [citar função]"]},
+    {"Comunidade acadêmica" : ["Pesquisador/a", "Estudantes de graduação e pós-graduação", "Representantes de entidades de pesquisa (Ex.: ANPED, ANPAE e FINEDUCA)", "Outro [citar função]"]},
+    {"Imprensa" : ["Jornalista", "Outro [citar função]"]},
+    {"Outro [citar segmento]" : []}
+  ]
+  next();
+}, response('role'));
+
+/*
+userApp.get('/', passport.authenticate('bearer', {session: false}), (req, res, next) => {
+  User.find((err, users) => {
+    if(err) {
+      log.error(err);
+      return next(err);
+    }
+
+    let result = [];
+    users.forEach((user) => {
+      let u = user.toObject();
+      delete u.hashedPassword;
+      delete u.salt;
+      result.push(u);
+    });
+    req.result = result;
+    next();
+  });
+}, response('users'));
+*/
+
+userApp.get('/me', passport.authenticate('bearer', { session: false }), (req, res, next) => {
+  let user = req.user.toObject();
+  delete user.hashedPassword;
+  delete user.salt;
+  req.result = user;
+  next();
+}, response('user'));
+
+userApp.get('/:id', (req, res, next) => {
+  User.findById(req.params.id, (err, user) => {
+    if(err) {
+      log.error(err);
+      return next(err);
+    }
+    if(!user) {
+      req.statusCode = 404;
+      next({msg: 'User not found'});
+    } else {
+      let u = user.toObject;
+      delete u.hashedPassword;
+      delete u.salt;
+      req.result = u;
+      next();
+    }
+  });
+}, response('user'));
+
+userApp.post('/', (req, res, next) => {
+  let user = new User({
+    email: req.body.email,
+    password: req.body.password,
+    name: req.body.name,
+    nickname: req.body.nickname,
+    cpf: req.body.cpf,
+    cep: req.body.cep,
+    complement: req.body.complement,
+    address: req.body.address,
+    phone: req.body.phone,
+    schooling: req.body.schooling,
+    course: req.body.course,
+    segment: req.body.segment,
+    role: req.body.role,
+    institutionName: req.body.institutionName,
+    state: req.body.state,
+    city: req.body.city,
+    receiveEmails: false || req.body.receiveEmails,
+    origin: req.body.origin,
+    citesegment: req.body.citesegment,
+    citerole: req.body.citerole,
+    admin: false
+  });
+
+  if (typeof req.body.password === 'undefined' || !req.body.password) {
+    res.statusCode = 400;
+    return res.json({errors: ["O campo senha é obrigatório"]});
+  } else {
+    user.save((err) => {
+      if(err) {
+        log.error(err);
+        let errors = [];
+        for(let errName in err.errors) {
+          errors.push(err.errors[errName].message);
+        }
+        log.error(errors);
+        res.statusCode = 400;
+        return res.json({err, errors});
+      }
+
+      // Create verification token
+      let verificationToken = new VerificationToken({
+        userId: user._id
+      });
+
+      verificationToken.createVerificationToken((err, token) => {
+        if(err) {
+          log.error(err);
+          return next(err);
+        }
+        let url = config.default.lde.url + '/verify';
+        let text = `Olá, ${user.name}, seja bem vindo/a ao Laboratório de Dados Educacionais.\n\nClique neste link para confirmar sua conta: ${url}/${token}`;
+        // Send confirmation email
+        let mailOptions = {
+          to: `"${user.name} <${user.email}>"`,
+          subject: "Confirme seu cadastro - Laboratório de Dados Educacionais",
+          text
+        }
+        email(mailOptions, (err, info) => {
+          if(err) {
+            log.error(err);
+            res.json({msg: 'User created'});
+          }
+          if(info) {
+            log.info(`Message ${info.messageId} sent: ${info.response}`);
+            log.info(`Usuário ${user.email} foi criado`);
+          }
+          res.json({msg: 'User created'});
+        });
+      });
+    });
+  }
+
+});
+
+userApp.put('/:id', passport.authenticate('bearer', { session: false }), (req, res, next) => {
+  User.findById(req.params.id, (err, user) => {
+    if (err) {
+      log.error(err);
+      return next({err});
+    }
+
+    if(!user) {
+      res.statusCode = 404;
+      return next({err: {
+        message: 'Usuário não encontrado'
+      }});
+    }
+
+    user.email = req.body.email || user.email;
+    user.name = req.body.name || user.name;
+    user.nickname = req.body.nickname || user.nickname || user.name;
+    user.cep = req.body.cep || user.cep;
+    user.complement = req.body.complement || user.complement;
+    user.address = req.body.address || user.address;
+    user.phone = req.body.phone || user.phone;
+    user.schooling = req.body.schooling || user.schooling;
+    user.course = req.body.course || user.course;
+    user.segment = req.body.segment || user.segment;
+    user.role = req.body.role || user.role;
+    user.institutionName = req.body.institutionName || user.institutionName;
+    user.state = req.body.state || user.state;
+    user.city = req.body.city || user.city;
+    user.receiveEmails = req.body.receiveEmails || user.receiveEmails;
+    user.citesegment = req.body.citesegment || user.citesegment;
+    user.citerole = req.body.citerole || user.citerole;
+
+    if ((req.body.password) && (req.body.newpassword)) {
+        if (req.body.password != req.body.newpassword) {
+            if (user.checkPassword(req.body.password)) {
+                user.password = req.body.newpassword;
+            } else {
+                res.statusCode = 500;
+                return res.json({error: {
+                    message: 'A senha atual está incorreta'
+                }});
+            }
+        } else {
+            res.statusCode = 500;
+            return res.json({error: {
+                message: 'A nova senha é a mesma da senha atual'
+            }});
+        }
+    }
+
+    user.save(err => {
+      if(err) {
+        log.error(err);
+        return next({message: 'Erro ao atualizar usuário'});
+      }
+      let u = user.toObject();
+      delete u.hashedPassword;
+      delete u.salt;
+      res.json({user: u});
+    })
+  })
+});
+
+userApp.get('/reset/password', (req, res, next) => {
+  let emailAddress = req.query.email;
+  User.findOne({email: emailAddress}, (err, user)=> {
+    if(err) {
+      log.error(err);
+      let errors = [];
+      for(let errName in err.errors) {
+        errors.push(err.errors[errName].message);
+      }
+      res.statusCode = 400;
+      return res.json({err, errors});
+    }
+    if (!user) {
+      res.statusCode = 404;
+      res.json({msg: "O usuário não está cadastrado"});
+    }
+    else {
+      let resetToken = new ResetToken({
+        userId: user._id
+      });
+      resetToken.createResetToken((err, token) => {
+        if (err) {
+          log.error(err);
+          return next(err);
+        }
+        let url = config.default.lde.url + '/reset-password';
+        let text = `Olá, ${user.name}.\n\nRecebemos uma solicitação para redefinir sua senha do Laboratório de Dados Educacionais. Clique neste link para redefinir a sua senha: ${url}/${token}`;
+        let mailOptions = {
+          to: `"${user.name} <${user.email}>"`,
+          subject: "Redefinição de Senha - Laboratório de Dados Educacionais",
+          text
+        }
+        email(mailOptions, (err, info) => {
+          if(err) {
+            log.error(err);
+            res.json({msg: 'Undelivered Reset Password Mail'});
+          }
+          log.info(`Message ${info.messageId} sent: ${info.response}`);
+          res.json({msg: 'Reset Password Mail Successfully Delivered'});
+        });
+      })
+    }
+  })
+})
+
+module.exports = userApp;
diff --git a/src/libs/routes_v2/verifyToken.js b/src/libs/routes_v2/verifyToken.js
new file mode 100644
index 00000000..d54f64aa
--- /dev/null
+++ b/src/libs/routes_v2/verifyToken.js
@@ -0,0 +1,52 @@
+const express = require('express');
+
+const verifyTokenApp = express.Router();
+
+const libs = `${process.cwd()}/libs`;
+
+const log = require(`${libs}/log`)(module);
+
+const VerificationToken = require(`${libs}/models/verificationToken`);
+
+const User = require(`${libs}/models/user`);
+
+verifyTokenApp.get('/:token', (req, res, next) => {
+    let token = req.params.token;
+    VerificationToken.findOne({token: token}, (err, vToken) => {
+        if(err) {
+            log.error(err);
+            return next(err);
+        }
+        if(!vToken) {
+            // TODO: generate new verification token
+            res.statusCode = 404;
+            return next({msg: 'Token not found', status:404});
+        }
+        User.findById(vToken.userId, (err, user) => {
+            if(err) {
+                log.error(err);
+                next(err);
+            }
+            user.verified = true;
+            user.save((err) => {
+                if(err) {
+                    log.error(err);
+                    next(err);
+                }
+            });
+            let u = user.toObject();
+            delete u.salt;
+            delete u.hashedPassword;
+            vToken.verified = true;
+            vToken.save((err) => {
+                if(err) {
+                    log.error(err);
+                    next(err);
+                }
+            });
+            res.json({msg: 'User verified', user: u});
+        });
+    });
+});
+
+module.exports = verifyTokenApp;
-- 
GitLab