# Copyright (C) 2008-2011 AMEE UK Ltd. - http://www.amee.com # Released as Open Source Software under the BSD 3-Clause license. See LICENSE.txt for details. require 'spec_helper.rb' TestSeriesOne=[[AMEE::Epoch,1],[AMEE::Epoch+1,2],[AMEE::Epoch+3,4]] TestSeriesTwo=[[AMEE::Epoch,2],[AMEE::Epoch+1,6],[AMEE::Epoch+5,7],[AMEE::Epoch+9,11]] MockResourcePath="/data/transport/plane/generic/AD63A83B4D41/kgCO2PerPassengerJourney" MockDataItemPath="/data/transport/plane/generic/AD63A83B4D41" MockResourceXML=''+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+AMEE::Epoch.xmlschema+'1kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+(AMEE::Epoch+1).xmlschema+'2kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+(AMEE::Epoch+3).xmlschema+'4kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ '' MockResourceJSON='{"dataItem":{"uid":"AD63A83B4D41"},"itemValues":['+ ' {"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+AMEE::Epoch.xmlschema+'","value":"1","uid":"127612FA4921","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ',{"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+(AMEE::Epoch+1).xmlschema+'","value":"2","uid":"127612FA4922","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ',{"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+(AMEE::Epoch+3).xmlschema+'","value":"4","uid":"127612FA4923","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ']}' MockResourceXMLTwo=''+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+AMEE::Epoch.xmlschema+'1kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+(AMEE::Epoch+1).xmlschema+'6kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+(AMEE::Epoch+5).xmlschema+'7kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ 'kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+(AMEE::Epoch+9).xmlschema+'11kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ '' MockResourceJSONTwo='{"dataItem":{"uid":"AD63A83B4D41"},"itemValues":['+ ' {"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+AMEE::Epoch.xmlschema+'","value":"1","uid":"127612FA4921","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ',{"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+(AMEE::Epoch+1).xmlschema+'","value":"6","uid":"127612FA4922","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ',{"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+(AMEE::Epoch+5).xmlschema+'","value":"7","uid":"127612FA4924","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ',{"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+(AMEE::Epoch+9).xmlschema+'","value":"11","uid":"127612FA4925","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ ']}' MockResourceXMLSingle='kgCO2PerPassengerJourneykgCO2 Per Passenger Journey'+AMEE::Epoch.xmlschema+'1kgCO2PerPassengerJourneykgCO2 Per Passenger JourneyfalsetruekgCO2PerJourneyDOUBLE'+ '' MockResourceJSONSingle='{"dataItem":{"uid":"AD63A83B4D41"},"itemValue":'+ ' {"item":{"uid":"AD63A83B4D41"},"modified":"2007-08-01 09:00:41.0","created":"2007-08-01 09:00:41.0","startDate":"'+AMEE::Epoch.xmlschema+'","value":"1","uid":"127612FA4921","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey","itemValueDefinition":{"valueDefinition":{"valueType":"DOUBLE","uid":"8CB8A1789CD6","name":"kgCO2PerJourney"},"uid":"653828811D42","path":"kgCO2PerPassengerJourney","name":"kgCO2 Per Passenger Journey"}}'+ '}' describe AMEE::Data::ItemValueHistory do before(:each) do @history = AMEE::Data::ItemValueHistory.new end it "should NOT have common AMEE object properties" do # we can't be an AMEE object, since we have a set of UIDs, not one UID @history.is_a?(AMEE::Data::Object).should be_false end it "should have an array of ItemValue objects" do @history.should respond_to(:values) end it "should be able to return a time series" do @history.should respond_to(:series) end it "should have a type" do @history.should respond_to(:type) end it "can be created with a type, and a time-value pairs array" do series=TestSeriesOne type = "TEXT" @history = AMEE::Data::ItemValueHistory.new(:series => series, :type => type) @history.series.should == series @history.type.should == type @history.values[0].is_a?(AMEE::Data::ItemValue).should be_true @history.values[0].value.should == 1 @history.values[0].start_date.should == AMEE::Epoch end it "can be created by pushing to the array of item values" do series=TestSeriesOne type = "TEXT" @history = AMEE::Data::ItemValueHistory.new(:type=>type) @history.series.should == [] @history.type.should == type @fstvalue = AMEE::Data::ItemValue.new(:type=>type,:value=>1,:start_date=>AMEE::Epoch) @sndvalue = AMEE::Data::ItemValue.new(:type=>type,:value=>2,:start_date=>AMEE::Epoch+1) @trdvalue = AMEE::Data::ItemValue.new(:type=>type,:value=>4,:start_date=>AMEE::Epoch+3) @history.values.push @fstvalue @history.values.push @sndvalue @history.values.push @trdvalue @history.values[0].is_a?(AMEE::Data::ItemValue).should be_true @history.values[0].value.should == 1 @history.values[0].start_date.should == AMEE::Epoch @history.series.should == series end it "should support DOUBLE data type" do @history = AMEE::Data::ItemValueHistory.new(:series => TestSeriesOne, :type => "DOUBLE") @history.series.should == TestSeriesOne end it "should support TEXT data type" do @history = AMEE::Data::ItemValueHistory.new(:series => TestSeriesOne, :type => "TEXT") @history.series.should == TestSeriesOne end it "allows value to be changed after creation" do series=TestSeriesOne type = "TEXT" @history = AMEE::Data::ItemValueHistory.new(:series => series, :type => type) @history.series.should == series series = TestSeriesTwo @history.series = series @history.series.should == series @history.values[1].value.should == 6 end it "allows item values to be found by time" do series=TestSeriesTwo type = "TEXT" @history = AMEE::Data::ItemValueHistory.new(:series => series, :type => type) @value=@history.value_at(AMEE::Epoch+5) @value.value.should == 7 lambda { @history.value_at(AMEE::Epoch+6) }.should raise_error @history.values_at([AMEE::Epoch+1,AMEE::Epoch+9]).length.should eql 2 end end describe AMEE::Data::ItemValueHistory, "when comparing to another history" do before(:each) do @historyone = AMEE::Data::ItemValueHistory.new(:series=>TestSeriesOne) @historytwo = AMEE::Data::ItemValueHistory.new(:series=>TestSeriesTwo) @comparison=@historytwo.compare(@historyone) end it "should be able to compare histories" do @comparison.should be_a Hash @comparison[:deletions].should be_a Array @comparison[:updates].should be_a Array @comparison[:insertions].should be_a Array end it "should return an array of items to update" do # note comparison list isnt order stable so sort here for test @changes=@comparison[:updates].sort{|x,y| x.start_date <=> y.start_date} @changes.length.should eql 2 @changes[1].value.should eql 6 @changes[1].start_date.should eql AMEE::Epoch+1 @changes[0].start_date.should eql AMEE::Epoch @changes[0].value.should eql 2 end it "should return an array of items to create" do @changes=@comparison[:insertions].sort{|x,y| x.start_date <=> y.start_date} @changes.length.should eql 2 @changes[0].start_date.should eql AMEE::Epoch+5 @changes[1].start_date.should eql AMEE::Epoch+9 @changes[0].value.should eql 7 @changes[1].value.should eql 11 end it "should return an array of items to delete" do @changes=@comparison[:deletions].sort{|x,y| x.start_date <=> y.start_date} @changes.length.should eql 1 @changes[0].start_date.should eql AMEE::Epoch+3 end end describe AMEE::Data::ItemValueHistory, "with an authenticated connection" do it "should parse XML correctly" do connection = flexmock "connection" connection.should_receive(:get).with(MockResourcePath,:valuesPerPage=>2). and_return(flexmock(:body => MockResourceXML)) @history = AMEE::Data::ItemValueHistory.get(connection, MockResourcePath) @history.series.should == TestSeriesOne @fstvalue=@history.values[0] @sndvalue=@history.values[1] @fstvalue.uid.should == "127612FA4921" @sndvalue.uid.should == "127612FA4922" @fstvalue.name.should == "kgCO2 Per Passenger Journey" @fstvalue.full_path.should == MockResourcePath @fstvalue.created.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.modified.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.type.should == "DOUBLE" @history.series.should == TestSeriesOne @history.type.should == "DOUBLE" end it "should parse JSON correctly" do connection = flexmock "connection" connection.should_receive(:get).with(MockResourcePath,:valuesPerPage=>2).and_return(flexmock(:body => MockResourceJSON)) @history = AMEE::Data::ItemValueHistory.get(connection, MockResourcePath) @fstvalue=@history.values[0] @sndvalue=@history.values[1] @fstvalue.uid.should == "127612FA4921" @sndvalue.uid.should == "127612FA4922" @fstvalue.name.should == "kgCO2 Per Passenger Journey" @fstvalue.full_path.should == MockResourcePath @fstvalue.created.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.modified.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.type.should == "DOUBLE" @history.series.should == TestSeriesOne @history.type.should == "DOUBLE" end it "should parse JSON correctly if theres only one point" do connection = flexmock "connection" connection.should_receive(:get).with(MockResourcePath,:valuesPerPage=>2).and_return(flexmock(:body => MockResourceJSONSingle)) @history = AMEE::Data::ItemValueHistory.get(connection, MockResourcePath) @fstvalue=@history.values[0] @fstvalue.uid.should == "127612FA4921" @fstvalue.name.should == "kgCO2 Per Passenger Journey" @fstvalue.full_path.should == MockResourcePath @fstvalue.created.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.modified.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.type.should == "DOUBLE" @history.series.should == [[AMEE::Epoch,1]] @history.type.should == "DOUBLE" end it "should parse XML correctly if theres only one point" do connection = flexmock "connection" connection.should_receive(:get).with(MockResourcePath,:valuesPerPage=>2).and_return(flexmock(:body => MockResourceXMLSingle)) @history = AMEE::Data::ItemValueHistory.get(connection, MockResourcePath) @fstvalue=@history.values[0] @fstvalue.uid.should == "127612FA4921" @fstvalue.name.should == "kgCO2 Per Passenger Journey" @fstvalue.full_path.should == MockResourcePath @fstvalue.created.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.modified.should == DateTime.new(2007,8,1,9,00,41) @fstvalue.type.should == "DOUBLE" @history.series.should == [[AMEE::Epoch,1]] @history.type.should == "DOUBLE" end it "should fail gracefully with incorrect XML data" do connection = flexmock "connection" xml = '' connection.should_receive(:get).with("/data",:valuesPerPage=>2).and_return(flexmock(:body => xml)) lambda{AMEE::Data::ItemValueHistory.get(connection, "/data")}.should raise_error(AMEE::BadData) end it "should fail gracefully with incorrect JSON data" do connection = flexmock "connection" json = '{}' connection.should_receive(:get).with("/data",:valuesPerPage=>2).and_return(flexmock(:body => json)) lambda{AMEE::Data::ItemValueHistory.get(connection, "/data")}.should raise_error(AMEE::BadData) end it "should fail gracefully on other errors" do connection = flexmock "connection" connection.should_receive(:get).with("/data",:valuesPerPage=>2).and_raise("unidentified error") lambda{AMEE::Data::ItemValueHistory.get(connection, "/data")}.should raise_error(AMEE::BadData) end it "should fail gracefully if create series without epoch" do @history = AMEE::Data::ItemValueHistory.new() lambda{@history.series=[[AMEE::Epoch+1,5]]}.should raise_error(AMEE::BadData) end end describe AMEE::Data::ItemValueHistory, "after loading" do before(:each) do @path = MockResourcePath @connection = flexmock "connection" @connection.should_receive(:get).with(@path,:valuesPerPage=>2).and_return(flexmock(:body => MockResourceJSON)) @val = AMEE::Data::ItemValueHistory.get(@connection, @path) end it "can have series changed and saved back to server" do @connection.should_receive(:put).with(MockDataItemPath+"/127612FA4921", :value => 2).once.and_return(flexmock(:body => '')) # note this one shouldn't include a start date as it is the epoch point @connection.should_receive(:put).with(MockDataItemPath+"/127612FA4922", :value => 6, :startDate => AMEE::Epoch+1).once.and_return(flexmock(:body => '')) @connection.should_receive(:delete).with(MockDataItemPath+"/127612FA4923").once.and_return(flexmock(:body => '')) @connection.should_receive(:post).with(MockDataItemPath, :kgCO2PerPassengerJourney => 7, :startDate => AMEE::Epoch+5).once.and_return({'Location'=>'http://foo.com/'}) @connection.should_receive(:post).with(MockDataItemPath, :kgCO2PerPassengerJourney => 11, :startDate => AMEE::Epoch+9).once.and_return({'Location'=>'http://foo.com/'}) lambda { @val.series = TestSeriesTwo @val.save! }.should_not raise_error end it "can have series changed and saved back to server (using SSL)" do @connection.should_receive(:put).with(MockDataItemPath+"/127612FA4921", :value => 2).once.and_return(flexmock(:body => '')) # note this one shouldn't include a start date as it is the epoch point @connection.should_receive(:put).with(MockDataItemPath+"/127612FA4922", :value => 6, :startDate => AMEE::Epoch+1).once.and_return(flexmock(:body => '')) @connection.should_receive(:delete).with(MockDataItemPath+"/127612FA4923").once.and_return(flexmock(:body => '')) @connection.should_receive(:post).with(MockDataItemPath, :kgCO2PerPassengerJourney => 7, :startDate => AMEE::Epoch+5).once.and_return({'Location'=>'https://foo.com/'}) @connection.should_receive(:post).with(MockDataItemPath, :kgCO2PerPassengerJourney => 11, :startDate => AMEE::Epoch+9).once.and_return({'Location'=>'https://foo.com/'}) lambda { @val.series = TestSeriesTwo @val.save! }.should_not raise_error end it "cannot create a new series (unsupported by platform)" do lambda { @val.series = TestSeriesTwo @val.create! }.should raise_error(AMEE::NotSupported,"Cannot create a Data Item Value History from scratch: at least one data point must exist when the DI is created") end it "cannot delete an entire series (unsupported by platform)" do lambda { @val.delete! }.should raise_error(AMEE::NotSupported,"Cannot delete all of history: at least one data point must always exist.") end end