diff --git a/app/authenticators/application.js b/app/authenticators/application.js new file mode 100644 index 0000000..3ffd4ff --- /dev/null +++ b/app/authenticators/application.js @@ -0,0 +1,50 @@ +import Ember from 'ember'; +import BaseAuthenticator from 'ember-simple-auth/authenticators/base'; +import config from '../config/environment'; + +const { RSVP: { Promise }, $, get, isEmpty, run } = Ember; + +export default BaseAuthenticator.extend({ + serverTokenEndpoint: `${config.APP.API_HOST}/api/auth/login/`, + tokenAttributeName: 'data.token', + identificationAttributeName: 'username', + verificationAttributeName: 'password', + + restore(data) { + const tokenAttributeName = this.get('tokenAttributeName'); + const tokenAttribute = get(data, tokenAttributeName); + + if (!isEmpty(tokenAttribute)) { + return Promise.resolve(data); + } else { + return Promise.reject(); + } + }, + + authenticate(username, password) { + return new Promise((resolve, reject) => { + const { identificationAttributeName: id, verificationAttributeName: verify } = this.getProperties('identificationAttributeName', 'verificationAttributeName'); + const data = {}; + data[id] = username; + data[verify] = password; + return this.makeRequest(data).then( + (response) => run(null, resolve, response), + (xhr) => run(null, reject, xhr.responseJSON || xhr.responseText) + ); + }); + }, + + invalidate() { + return Promise.resolve(); + }, + + makeRequest(data, options) { + const serverTokenEndpoint = this.get('serverTokenEndpoint'); + const requestOptions = $.extend({}, { + url: serverTokenEndpoint, + type: 'POST', + data, + }, options || {}); + return $.ajax(requestOptions); + }, +}); diff --git a/app/authorizers/application.js b/app/authorizers/application.js new file mode 100644 index 0000000..a0d6c72 --- /dev/null +++ b/app/authorizers/application.js @@ -0,0 +1,13 @@ +import Ember from 'ember'; +import BaseAuthorizer from 'ember-simple-auth/authorizers/base'; + +const { isEmpty } = Ember; + +export default BaseAuthorizer.extend({ + authorize(data, block) { + const accessToken = data['token']; + if (!isEmpty(accessToken)) { + block('Authorization', `Token ${accessToken}`); + } + } +}); diff --git a/app/controllers/login.js b/app/controllers/login.js new file mode 100644 index 0000000..e5bb9c8 --- /dev/null +++ b/app/controllers/login.js @@ -0,0 +1,21 @@ +import Ember from 'ember'; + +const { Controller, inject: { service } } = Ember; + +export default Controller.extend({ + session: service(), + flashMessages: service(), + + actions: { + authenticate() { + this.transitionToRoute('loading').then(() => { + let { identification, password } = this.getProperties('identification', 'password'); + this.get('session').authenticate('authenticator:application', identification, password).catch((reason) => { + this.transitionToRoute('login').then(() => { + this.get('flashMessages').danger(reason.non_field_errors || reason); + }); + }); + }); + }, + }, +}); diff --git a/app/router.js b/app/router.js index cdc2578..185d60b 100644 --- a/app/router.js +++ b/app/router.js @@ -7,6 +7,7 @@ const Router = Ember.Router.extend({ }); Router.map(function() { + this.route('login'); }); export default Router; diff --git a/app/routes/application.js b/app/routes/application.js new file mode 100644 index 0000000..38bb6b3 --- /dev/null +++ b/app/routes/application.js @@ -0,0 +1,6 @@ +import Ember from 'ember'; +import ApplicationRouteMixin from 'ember-simple-auth/mixins/application-route-mixin'; + +const { Route } = Ember; + +export default Route.extend(ApplicationRouteMixin, {}); diff --git a/app/routes/index.js b/app/routes/index.js new file mode 100644 index 0000000..ed916b6 --- /dev/null +++ b/app/routes/index.js @@ -0,0 +1,6 @@ +import Ember from 'ember'; +import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin'; + +const { Route } = Ember; + +export default Route.extend(AuthenticatedRouteMixin, {}); diff --git a/app/routes/login.js b/app/routes/login.js new file mode 100644 index 0000000..a9687fd --- /dev/null +++ b/app/routes/login.js @@ -0,0 +1,6 @@ +import Ember from 'ember'; +import UnauthenticatedRouteMixin from 'ember-simple-auth/mixins/unauthenticated-route-mixin'; + +const { Route } = Ember; + +export default Route.extend(UnauthenticatedRouteMixin, {}); diff --git a/app/styles/app.css b/app/styles/app.css index e69de29..4eefa73 100644 --- a/app/styles/app.css +++ b/app/styles/app.css @@ -0,0 +1,39 @@ +.form-signin { + max-width: 330px; + padding: 15px; + margin: 0 auto; +} + +.form-signin .form-signin-heading, .form-signin .checkbox { + margin-bottom: 10px; +} + +.form-signin .checkbox { + font-weight: normal; +} + +.form-signin .form-control { + position: relative; + height: auto; + -webkit-box-sizing: border-box; + -moz-box-sizing: border-box; + box-sizing: border-box; + padding: 10px; + font-size: 16px; +} + +.form-signin .form-control:focus { + z-index: 2; +} + +.form-signin input[type="text"] { + margin-bottom: -1px; + border-bottom-right-radius: 0; + border-bottom-left-radius: 0; +} + +.form-signin input[type="password"] { + margin-bottom: 10px; + border-top-left-radius: 0; + border-top-right-radius: 0; +} diff --git a/app/templates/application.hbs b/app/templates/application.hbs index 103746f..c24cd68 100644 --- a/app/templates/application.hbs +++ b/app/templates/application.hbs @@ -1,11 +1 @@ -