Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
node_modules
.env
16 changes: 15 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -1 +1,15 @@
# api-auth
# API BASIC
Demo app with basic REST API.

# REST API
List of basic routes:

| Route | HTTP | Description |
| ------ | ------ | ------ |
| /api/signup | GET | Sign up with new user info |
| /api/signin | GET | Sign in while get an access token based on credentials |
| /api/users | GET | Get all the users info (admin only) |
| /api/users/:id | GET | Get a single users info (admin only and authenticated user) |
| /api/users | POST | Create a users (admin only) |
| /api/users/:id | PUT | Update a user with new info (admin only and authenticated user) |
| /api/users/:id | DELETE | Delete a user (admin only) |
10 changes: 10 additions & 0 deletions config/config.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{
"development": {
"username": "edim",
"password": "edimdendy",
"database": "api_db",
"host": "127.0.0.1",
"dialect": "postgres",
"logging": false
}
}
57 changes: 57 additions & 0 deletions controllers/auth.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
let model = require('../models')
let passwordHash = require('password-hash');
let jwt = require('jsonwebtoken');
require('dotenv').config()
let methods = {}

methods.signup = function(req, res) {
req.body.password = passwordHash.generate(req.body.password)
model.User.create({
name: req.body.name,
username: req.body.username,
password: req.body.password,
email: req.body.email,
role: req.body.role
})
.then(data => {
res.json(data)
})
.catch(err => {
res.json(err)
})
}

methods.signin = function(req, res) {
model.User.find({
where: {
$or: [{
username: req.body.username
}, {
email: req.body.email
}]
}
})
.then(record => {
if (passwordHash.verify(req.body.password, record.password)) {
let data = Object.assign({}, record.toJSON())
delete data.id
delete data.password
delete data.createdAt
delete data.updatedAt
let token = jwt.sign(data, process.env.SECRET_KEY, {
expiresIn: '1h'
})
res.json({
message: 'Login is successful',
token,
data
})
} else {
res.json({
message: 'Your password is not matched'
})
}
})
}

module.exports = methods
81 changes: 81 additions & 0 deletions controllers/users.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
let model = require('../models');
let passwordHash = require('password-hash');
let methods = {}

methods.getAllUsers = function(req, res) {
model.User.findAll()
.then(dataAllUser => {
res.json(dataAllUser)
})
}

methods.getById = function(req, res) {
model.User.findOne({
where: {
id: req.params.id
}
})
.then(data => {
res.json(data)
})
.catch(err => {
res.json(err)
})
}

methods.createUser = function(req, res) {
req.body.password = passwordHash.generate(req.body.password)
model.User.create({
name: req.body.name,
username: req.body.username,
password: req.body.password,
email: req.body.email,
role: req.body.role
})
.then(data => {
res.json(data)
})
.catch(err => {
res.json(err)
})
}

methods.updateById = function(req, res) {
model.User.find({
where: {
id: req.params.id
}
})
.then(function(users) {
if (users) {
users.updateAttributes({
name: req.body.name,
username: req.body.username,
password: req.body.password,
email: req.body.email
})
.then(function(data) {
res.json(data)
})
.catch(err => {
res.json(err)
})
}
});
}

methods.deleteById = function(req, res) {
model.User.destroy({
where: {
id: req.params.id
}
})
.then(function(data) {
res.json(data)
})
.catch(err => {
res.json(err)
})
}

module.exports = methods
35 changes: 35 additions & 0 deletions helpers/verify_token.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
const jwt = require('jsonwebtoken');
const model = require('../models');
require('dotenv').config();

module.exports = {
auth: function(req, res, next) {
jwt.verify(req.headers.token, process.env.SECRET_KEY, function(err, decoded) {
if (decoded) {
next()
} else {
res.send('You must login!')
}
})
},
isAdmin: function(req, res, next) {
jwt.verify(req.headers.token, process.env.SECRET_KEY, function(err, decoded) {
if (decoded) {
model.User.find({
where : {
$or: [{ username: decoded.username }, { email: decoded.email }]
}
})
.then((user) => {
if (user.role == 'admin') {
next()
} else {
res.send('You are not admin!')
}
})
} else {
res.send(err)
}
})
}
}
39 changes: 39 additions & 0 deletions migrations/20170425042846-create-user.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
'use strict';
module.exports = {
up: function(queryInterface, Sequelize) {
return queryInterface.createTable('Users', {
id: {
allowNull: false,
autoIncrement: true,
primaryKey: true,
type: Sequelize.INTEGER
},
name: {
type: Sequelize.STRING
},
username: {
type: Sequelize.STRING
},
password: {
type: Sequelize.TEXT
},
email: {
type: Sequelize.STRING
},
role: {
type: Sequelize.STRING
},
createdAt: {
allowNull: false,
type: Sequelize.DATE
},
updatedAt: {
allowNull: false,
type: Sequelize.DATE
}
});
},
down: function(queryInterface, Sequelize) {
return queryInterface.dropTable('Users');
}
};
36 changes: 36 additions & 0 deletions models/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
'use strict';

var fs = require('fs');
var path = require('path');
var Sequelize = require('sequelize');
var basename = path.basename(module.filename);
var env = process.env.NODE_ENV || 'development';
var config = require(__dirname + '/../config/config.json')[env];
var db = {};

if (config.use_env_variable) {
var sequelize = new Sequelize(process.env[config.use_env_variable]);
} else {
var sequelize = new Sequelize(config.database, config.username, config.password, config);
}

fs
.readdirSync(__dirname)
.filter(function(file) {
return (file.indexOf('.') !== 0) && (file !== basename) && (file.slice(-3) === '.js');
})
.forEach(function(file) {
var model = sequelize['import'](path.join(__dirname, file));
db[model.name] = model;
});

Object.keys(db).forEach(function(modelName) {
if (db[modelName].associate) {
db[modelName].associate(db);
}
});

db.sequelize = sequelize;
db.Sequelize = Sequelize;

module.exports = db;
20 changes: 20 additions & 0 deletions models/user.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
'use strict';
module.exports = function(sequelize, DataTypes) {
var User = sequelize.define('User', {
name: DataTypes.STRING,
username: {
type: DataTypes.STRING,
allowNull: false
},
password: DataTypes.TEXT,
email: DataTypes.STRING,
role: DataTypes.STRING
}, {
classMethods: {
associate: function(models) {
// associations can be defined here
}
}
});
return User;
};
33 changes: 33 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
{
"name": "api-basic",
"version": "1.0.0",
"description": "build api basic ",
"main": "index.js",
"scripts": {
"start": "node server.js",
"dev": "nodemon server.js",
"test": "echo \"Error: no test specified\" && exit 1"
},
"repository": {
"type": "git",
"url": "git+https://github.com/edimdendidiwangga/api-basic.git"
},
"author": "",
"license": "ISC",
"bugs": {
"url": "https://github.com/edimdendidiwangga/api-basic/issues"
},
"homepage": "https://github.com/edimdendidiwangga/api-basic#readme",
"dependencies": {
"body-parser": "^1.17.1",
"dotenv": "^4.0.0",
"express": "^4.15.2",
"jsonwebtoken": "^7.4.0",
"password-hash": "^1.2.2",
"pg": "^6.1.5",
"sequelize": "^3.30.4"
},
"devDependencies": {
"nodemon": "^1.11.0"
}
}
18 changes: 18 additions & 0 deletions routes/api.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
let router = require('express').Router();
let userController = require('../controllers/users')
let authController = require('../controllers/auth')
let helper = require('../helpers/verify_token')

router.get('/', function(req, res, next) {
res.send('Hello World');
})
// routes user
router.get('/users', helper.isAdmin, userController.getAllUsers)
router.get('/users/:id', helper.auth, userController.getById)
router.post('/users', helper.isAdmin, userController.createUser)
router.put('/users/:id', helper.auth, userController.updateById)
router.delete('/users/:id', helper.isAdmin, userController.deleteById)
router.post('/signup', authController.signup)
router.post('/signin', authController.signin)

module.exports = router;
14 changes: 14 additions & 0 deletions server.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
const express = require('express');
const bodyParser = require('body-parser');
const app = express()

app.set('port', process.env.PORT || 3000)

app.use(require('body-parser').json());
app.use(require('body-parser').urlencoded({ extended: false }));

app.use('/api', require('./routes/api'));

app.listen(app.get('port'), function(){
console.log('listening on port '+app.get('port'))
})