feat(di): add metadata to Key

This commit is contained in:
vsavkin 2014-10-20 15:17:06 -04:00
parent 2a4b63b614
commit ea0df352be
7 changed files with 58 additions and 9 deletions

View File

@ -12,6 +12,8 @@ function constructResolvingPath(keys:List) {
} }
} }
export class KeyMetadataError extends Error {}
export class ProviderError extends Error { export class ProviderError extends Error {
constructor(key:Key, constructResolvingMessage:Function) { constructor(key:Key, constructResolvingMessage:Function) {
this.keys = [key]; this.keys = [key];

View File

@ -44,13 +44,13 @@ export class Injector {
_createListOfBindings(flattenBindings):List { _createListOfBindings(flattenBindings):List {
var bindings = ListWrapper.createFixedSize(Key.numberOfKeys() + 1); var bindings = ListWrapper.createFixedSize(Key.numberOfKeys + 1);
MapWrapper.forEach(flattenBindings, (v, keyId) => bindings[keyId] = v); MapWrapper.forEach(flattenBindings, (v, keyId) => bindings[keyId] = v);
return bindings; return bindings;
} }
_createInstances():List { _createInstances():List {
return ListWrapper.createFixedSize(Key.numberOfKeys() + 1); return ListWrapper.createFixedSize(Key.numberOfKeys + 1);
} }
_getByKey(key:Key, returnPromise:boolean, returnLazy:boolean) { _getByKey(key:Key, returnPromise:boolean, returnLazy:boolean) {

View File

@ -1,30 +1,44 @@
import {KeyMetadataError} from './exceptions';
import {MapWrapper, Map} from 'facade/collection'; import {MapWrapper, Map} from 'facade/collection';
import {FIELD, int, isPresent} from 'facade/lang'; import {FIELD, int, isPresent} from 'facade/lang';
var _allKeys = MapWrapper.create(); var _allKeys = MapWrapper.create();
var _id:int = 0;
export class Key { export class Key {
@FIELD('final token') @FIELD('final token')
@FIELD('final id:int') @FIELD('final id:int')
@FIELD('metadata:Object')
constructor(token, id:int) { constructor(token, id:int) {
this.token = token; this.token = token;
this.id = id; this.id = id;
this.metadata = null;
} }
static get(token) { static get(token):Key {
if (token instanceof Key) return token; if (token instanceof Key) return token;
if (MapWrapper.contains(_allKeys, token)) { if (MapWrapper.contains(_allKeys, token)) {
return MapWrapper.get(_allKeys, token); return MapWrapper.get(_allKeys, token);
} }
var newKey = new Key(token, ++_id); var newKey = new Key(token, Key.numberOfKeys);
MapWrapper.set(_allKeys, token, newKey); MapWrapper.set(_allKeys, token, newKey);
return newKey; return newKey;
} }
static numberOfKeys() { static setMetadata(key:Key, metadata):Key {
return _id; if (isPresent(key.metadata) && key.metadata !== metadata) {
throw new KeyMetadataError();
}
key.metadata = metadata;
return key;
}
static clear() {
_allKeys = MapWrapper.create();
}
static get numberOfKeys():int {
return MapWrapper.size(_allKeys);
} }
} }

View File

@ -1,8 +1,12 @@
import {describe, it, expect} from 'test_lib/test_lib'; import {describe, it, expect, beforeEach} from 'test_lib/test_lib';
import {Key} from 'di/di'; import {Key} from 'di/di';
export function main() { export function main() {
describe("key", function () { describe("key", function () {
beforeEach(function () {
Key.clear();
});
it('should be equal to another key if type is the same', function () { it('should be equal to another key if type is the same', function () {
expect(Key.get('car')).toBe(Key.get('car')); expect(Key.get('car')).toBe(Key.get('car'));
}); });
@ -14,5 +18,32 @@ export function main() {
it('should return the passed in key', function () { it('should return the passed in key', function () {
expect(Key.get(Key.get('car'))).toBe(Key.get('car')); expect(Key.get(Key.get('car'))).toBe(Key.get('car'));
}); });
describe("metadata", function () {
it("should assign metadata to a key", function () {
var key = Key.get('car');
Key.setMetadata(key, "meta");
expect(key.metadata).toEqual("meta");
});
it("should allow assigning the same metadata twice", function () {
var key = Key.get('car');
Key.setMetadata(key, "meta");
Key.setMetadata(key, "meta");
expect(key.metadata).toEqual("meta");
});
it("should throw when assigning different metadata", function () {
var key = Key.get('car');
Key.setMetadata(key, "meta1");
expect(() => Key.setMetadata(key, "meta2")).toThrowError();
});
});
}); });
} }

View File

@ -11,6 +11,7 @@ class MapWrapper {
static forEach(m, fn) { static forEach(m, fn) {
m.forEach((k,v) => fn(v,k)); m.forEach((k,v) => fn(v,k));
} }
static int size(m) {return m.length;}
} }
class ListWrapper { class ListWrapper {

View File

@ -10,6 +10,7 @@ export class MapWrapper {
static forEach(m, fn) { static forEach(m, fn) {
m.forEach(fn); m.forEach(fn);
} }
static size(m) {return m.size;}
} }

View File

@ -15,7 +15,7 @@ Expect expect(actual, [matcher]) {
class Expect extends gns.Expect { class Expect extends gns.Expect {
Expect(actual) : super(actual); Expect(actual) : super(actual);
void toThrowError(message) => this.toThrowWith(message: message); void toThrowError([message=""]) => this.toThrowWith(message: message);
void toBePromise() => _expect(actual is Future, equals(true)); void toBePromise() => _expect(actual is Future, equals(true));
Function get _expect => gns.guinness.matchers.expect; Function get _expect => gns.guinness.matchers.expect;
} }