diff --git a/app/abilities/species.js b/app/abilities/species.js
new file mode 100644
index 0000000..7900607
--- /dev/null
+++ b/app/abilities/species.js
@@ -0,0 +1,17 @@
+import { Ability } from 'ember-can';
+
+export default Ability.extend({
+ // Only admins and writers can create a new species
+ canAdd: function() {
+ let role = this.get('session.currentUser.role');
+ return (role === 'W') || (role === 'A');
+ }.property('session.currentUser.role'),
+
+ // Only admins and the person who created can edit
+ canEdit: function() {
+ let role = this.get('session.currentUser.role');
+ let id = this.get('session.currentUser.id');
+ let author = this.get('model.createdBy');
+ return (role === 'W' && (+id === author)) || (role === 'A');
+ }.property('session.currentUser.role', 'session.currentUser.id', 'model.createdBy')
+});
diff --git a/app/components/species/species-details.js b/app/components/species/species-details.js
new file mode 100644
index 0000000..3fd4f8e
--- /dev/null
+++ b/app/components/species/species-details.js
@@ -0,0 +1,20 @@
+import Ember from 'ember';
+
+export default Ember.Component.extend({
+ classNames: ['grid-1'],
+ isEditing: false,
+ isNew: false,
+ actions: {
+ editSpecies: function() {
+ this.get('species').get('errors').clear();
+ if (this.get('isNew')) {
+ this.get('species').destroyRecord().then(this.sendAction());
+ }
+ this.toggleProperty('isEditing');
+ this.get('species').rollback();
+ },
+ saveSpecies: function() {
+ this.get('species').save().then(this.toggleProperty('isEditing'));
+ }
+ }
+});
diff --git a/app/models/species.js b/app/models/species.js
new file mode 100644
index 0000000..f4906c8
--- /dev/null
+++ b/app/models/species.js
@@ -0,0 +1,16 @@
+import DS from 'ember-data';
+
+export default DS.Model.extend({
+ speciesName: DS.attr('string'),
+ typeSpecies: DS.attr('boolean'),
+ etymology: DS.attr('string'),
+ genusName: DS.attr('string'),
+ strains: DS.hasMany('strain'),
+ totalStrains: DS.attr('number'),
+ createdAt: DS.attr('date'),
+ updatedAt: DS.attr('date'),
+ deletedAt: DS.attr('date'),
+ createdBy: DS.attr('number'),
+ updatedBy: DS.attr('number'),
+ deletedBy: DS.attr('number'),
+});
diff --git a/app/router.js b/app/router.js
index 2c793b1..f82a92c 100644
--- a/app/router.js
+++ b/app/router.js
@@ -8,6 +8,10 @@ var Router = Ember.Router.extend({
Router.map(function() {
this.route('login');
this.route('about');
+ this.resource('species', function() {
+ this.route('show', { path: ':species_id' });
+ this.route('new');
+ });
this.resource('strains', function() {
this.route('new');
this.route('show', { path: ':strain_id' }, function() {
diff --git a/app/routes/species.js b/app/routes/species.js
new file mode 100644
index 0000000..5b14286
--- /dev/null
+++ b/app/routes/species.js
@@ -0,0 +1,4 @@
+import Ember from 'ember';
+import AuthenticatedRouteMixin from 'simple-auth/mixins/authenticated-route-mixin';
+
+export default Ember.Route.extend(AuthenticatedRouteMixin);
diff --git a/app/routes/species/index.js b/app/routes/species/index.js
new file mode 100644
index 0000000..8f05241
--- /dev/null
+++ b/app/routes/species/index.js
@@ -0,0 +1,7 @@
+import Ember from 'ember';
+
+export default Ember.Route.extend({
+ model: function() {
+ return this.store.findAll('species');
+ }
+});
diff --git a/app/routes/species/new.js b/app/routes/species/new.js
new file mode 100644
index 0000000..96b0d46
--- /dev/null
+++ b/app/routes/species/new.js
@@ -0,0 +1,12 @@
+import Ember from 'ember';
+
+export default Ember.Route.extend({
+ model: function() {
+ return this.store.createRecord('species');
+ },
+ actions: {
+ cancelSpecies: function() {
+ this.transitionTo('species.index');
+ }
+ }
+});
diff --git a/app/routes/species/show.js b/app/routes/species/show.js
new file mode 100644
index 0000000..26d9f31
--- /dev/null
+++ b/app/routes/species/show.js
@@ -0,0 +1,4 @@
+import Ember from 'ember';
+
+export default Ember.Route.extend({
+});
diff --git a/app/templates/application.hbs b/app/templates/application.hbs
index f70063e..686e66b 100644
--- a/app/templates/application.hbs
+++ b/app/templates/application.hbs
@@ -4,6 +4,9 @@
{{/link-to}}
{{#if session.isAuthenticated}}
+ {{#link-to 'species' tagName='li' href=false}}
+ {{#link-to 'species'}}Species{{/link-to}}
+ {{/link-to}}
{{#link-to 'strains' tagName='li' href=false}}
{{#link-to 'strains'}}Strains{{/link-to}}
{{/link-to}}
diff --git a/app/templates/components/species/species-details.hbs b/app/templates/components/species/species-details.hbs
new file mode 100644
index 0000000..ce628a8
--- /dev/null
+++ b/app/templates/components/species/species-details.hbs
@@ -0,0 +1,76 @@
+
+
+
diff --git a/app/templates/species.hbs b/app/templates/species.hbs
new file mode 100644
index 0000000..c24cd68
--- /dev/null
+++ b/app/templates/species.hbs
@@ -0,0 +1 @@
+{{outlet}}
diff --git a/app/templates/species/index.hbs b/app/templates/species/index.hbs
new file mode 100644
index 0000000..bafa0e8
--- /dev/null
+++ b/app/templates/species/index.hbs
@@ -0,0 +1,30 @@
+{{genus-name}} Species
+Total species: {{controller.length}}
+
+{{#if (can "add species")}}
+ {{! Does nothing ATM }}
+ {{#link-to 'species.new' class="button-gray smaller"}}
+ Add Species
+ {{/link-to}}
+{{/if}}
+
+
+
+
+ Name |
+ Strains |
+
+
+
+ {{#each species in controller}}
+
+
+ {{#link-to 'species.show' species}}
+ {{species.speciesName}}
+ {{/link-to}}
+ |
+ {{species.totalStrains}} |
+
+ {{/each}}
+
+
diff --git a/app/templates/species/new.hbs b/app/templates/species/new.hbs
new file mode 100644
index 0000000..06da4fc
--- /dev/null
+++ b/app/templates/species/new.hbs
@@ -0,0 +1 @@
+{{species/species-details species=model isEditing=true isNew=true action="cancelSpecies"}}
diff --git a/app/templates/species/show.hbs b/app/templates/species/show.hbs
new file mode 100644
index 0000000..9112582
--- /dev/null
+++ b/app/templates/species/show.hbs
@@ -0,0 +1 @@
+{{species/species-details species=model}}
diff --git a/package.json b/package.json
index d419e2f..2a3d925 100644
--- a/package.json
+++ b/package.json
@@ -21,6 +21,7 @@
"devDependencies": {
"body-parser": "^1.12.2",
"broccoli-asset-rev": "^2.0.2",
+ "connect-restreamer": "^1.0.2",
"ember-can": "^0.4.0",
"ember-cli": "0.2.3",
"ember-cli-app-version": "0.3.3",
@@ -37,9 +38,9 @@
"ember-cli-uglify": "1.0.1",
"ember-data": "1.0.0-beta.16.1",
"ember-export-application-global": "^1.0.2",
- "express": "^4.12.3",
+ "express": "^4.12.4",
"glob": "^4.5.3",
"jsonwebtoken": "^5.0.0",
- "morgan": "^1.5.2"
+ "morgan": "^1.5.3"
}
}
diff --git a/server/mocks/species.js b/server/mocks/species.js
new file mode 100644
index 0000000..af29aac
--- /dev/null
+++ b/server/mocks/species.js
@@ -0,0 +1,108 @@
+module.exports = function(app) {
+ var express = require('express');
+ var speciesRouter = express.Router();
+
+ var SPECIES = [
+ {
+ id: 1,
+ genusName: "Hymenobacter",
+ speciesName: "One",
+ typeSpecies: true,
+ etymology: "Test Etymology",
+ strains: [1],
+ totalStrains: 1,
+ createdAt: "0001-01-01T00:00:00Z",
+ updatedAt: "0001-01-01T00:00:00Z",
+ deletedAt: null,
+ createdBy: 1,
+ updatedBy: 1,
+ deletedBy: null,
+ },
+ {
+ id: 2,
+ genusName: "Hymenobacter",
+ speciesName: "Two",
+ typeSpecies: true,
+ etymology: "Test Etymology",
+ strains: [2],
+ totalStrains: 1,
+ createdAt: "0001-01-01T00:00:00Z",
+ updatedAt: "0001-01-01T00:00:00Z",
+ deletedAt: null,
+ createdBy: 1,
+ updatedBy: 1,
+ deletedBy: null,
+ },
+ {
+ id: 3,
+ genusName: "Hymenobacter",
+ speciesName: "Three",
+ typeSpecies: true,
+ etymology: "Test Etymology",
+ strains: [3],
+ totalStrains: 1,
+ createdAt: "0001-01-01T00:00:00Z",
+ updatedAt: "0001-01-01T00:00:00Z",
+ deletedAt: null,
+ createdBy: 1,
+ updatedBy: 1,
+ deletedBy: null,
+ },
+ {
+ id: 4,
+ genusName: "Hymenobacter",
+ speciesName: "Four",
+ typeSpecies: true,
+ etymology: "Test Etymology",
+ strains: [4],
+ totalStrains: 1,
+ createdAt: "0001-01-01T00:00:00Z",
+ updatedAt: "0001-01-01T00:00:00Z",
+ deletedAt: null,
+ createdBy: 1,
+ updatedBy: 1,
+ deletedBy: null,
+ }
+ ];
+
+ speciesRouter.get('/', function(req, res) {
+ var species;
+ if (req.query.ids) {
+ species = SPECIES.filter(function(s) {
+ return req.query.ids.indexOf(s.id.toString()) > -1;
+ });
+ } else {
+ species = SPECIES;
+ }
+ res.send({
+ 'species': species
+ });
+ });
+
+ speciesRouter.post('/', function(req, res) {
+ res.status(201).end();
+ });
+
+ speciesRouter.get('/:id', function(req, res) {
+ var species = SPECIES.filter(function(s) {
+ return s.id == req.params.id;
+ });
+ res.send({
+ 'species': species[0]
+ });
+ });
+
+ speciesRouter.put('/:id', function(req, res) {
+ res.send({
+ 'species': {
+ id: req.params.id
+ }
+ });
+ });
+
+ speciesRouter.delete('/:id', function(req, res) {
+ res.status(204).end();
+ });
+
+ app.use('/api/hymenobacter/species', speciesRouter);
+};
diff --git a/tests/unit/components/species/species-details-test.js b/tests/unit/components/species/species-details-test.js
new file mode 100644
index 0000000..b3c5c67
--- /dev/null
+++ b/tests/unit/components/species/species-details-test.js
@@ -0,0 +1,21 @@
+import {
+ moduleForComponent,
+ test
+} from 'ember-qunit';
+
+moduleForComponent('species/species-details', {
+ // Specify the other units that are required for this test
+ // needs: ['component:foo', 'helper:bar']
+});
+
+test('it renders', function(assert) {
+ assert.expect(2);
+
+ // Creates the component instance
+ var component = this.subject();
+ assert.equal(component._state, 'preRender');
+
+ // Renders the component to the page
+ this.render();
+ assert.equal(component._state, 'inDOM');
+});
diff --git a/tests/unit/models/species-test.js b/tests/unit/models/species-test.js
new file mode 100644
index 0000000..1f8a9d1
--- /dev/null
+++ b/tests/unit/models/species-test.js
@@ -0,0 +1,15 @@
+import {
+ moduleForModel,
+ test
+} from 'ember-qunit';
+
+moduleForModel('species', {
+ // Specify the other units that are required for this test.
+ needs: ['model:strain']
+});
+
+test('it exists', function(assert) {
+ var model = this.subject();
+ // var store = this.store();
+ assert.ok(!!model);
+});
diff --git a/tests/unit/routes/species-test.js b/tests/unit/routes/species-test.js
new file mode 100644
index 0000000..ed7e7ba
--- /dev/null
+++ b/tests/unit/routes/species-test.js
@@ -0,0 +1,14 @@
+import {
+ moduleFor,
+ test
+} from 'ember-qunit';
+
+moduleFor('route:species', {
+ // Specify the other units that are required for this test.
+ // needs: ['controller:foo']
+});
+
+test('it exists', function(assert) {
+ var route = this.subject();
+ assert.ok(route);
+});
diff --git a/tests/unit/routes/species/index-test.js b/tests/unit/routes/species/index-test.js
new file mode 100644
index 0000000..4e751cd
--- /dev/null
+++ b/tests/unit/routes/species/index-test.js
@@ -0,0 +1,14 @@
+import {
+ moduleFor,
+ test
+} from 'ember-qunit';
+
+moduleFor('route:species/index', {
+ // Specify the other units that are required for this test.
+ // needs: ['controller:foo']
+});
+
+test('it exists', function(assert) {
+ var route = this.subject();
+ assert.ok(route);
+});
diff --git a/tests/unit/routes/species/new-test.js b/tests/unit/routes/species/new-test.js
new file mode 100644
index 0000000..1c7fab4
--- /dev/null
+++ b/tests/unit/routes/species/new-test.js
@@ -0,0 +1,14 @@
+import {
+ moduleFor,
+ test
+} from 'ember-qunit';
+
+moduleFor('route:species/new', {
+ // Specify the other units that are required for this test.
+ // needs: ['controller:foo']
+});
+
+test('it exists', function(assert) {
+ var route = this.subject();
+ assert.ok(route);
+});
diff --git a/tests/unit/routes/species/show-test.js b/tests/unit/routes/species/show-test.js
new file mode 100644
index 0000000..d6ad3b2
--- /dev/null
+++ b/tests/unit/routes/species/show-test.js
@@ -0,0 +1,14 @@
+import {
+ moduleFor,
+ test
+} from 'ember-qunit';
+
+moduleFor('route:species/show', {
+ // Specify the other units that are required for this test.
+ // needs: ['controller:foo']
+});
+
+test('it exists', function(assert) {
+ var route = this.subject();
+ assert.ok(route);
+});