AutomotiveDrivingModels is templated to efficiently run different types of simulations. Entities are parameterized by their:
Int, which uniquely identifies that entity.In addition to these types, the actions, environment and the driver models can also be parameterized.
This notebook demonstrates a 2D driving simulation where cars drive in a multi-lane stadium. The types are:
VehicleState, containing both the global and lane-relative position, and speedVehicleDef, containing length, width, and classIntWe use a Roadway as our environment. The Roadway type is based on the RNDF format.
In [1]:
using AutomotiveDrivingModels
using AutoViz
using Distributions
┌ Info: Precompiling AutomotiveDrivingModels [99497e54-f3d6-53d3-a3a9-fa9315a7f1ba]
└ @ Base loading.jl:1186
┌ Warning: The call to compilecache failed to create a usable precompiled cache file for AutomotiveDrivingModels [99497e54-f3d6-53d3-a3a9-fa9315a7f1ba]
│ exception = ArgumentError("Invalid header in cache file C:\\Users\\Maxime\\.julia\\compiled\\v1.1\\AutomotiveDrivingModels\\ZEfHM.ji.")
└ @ Base loading.jl:969
┌ Info: Precompiling AutoViz [82aa6e0c-a491-5edf-8d4b-c16b98e4ea17]
└ @ Base loading.jl:1186
┌ Warning: Module AutomotiveDrivingModels with build ID 614702750231424 is missing from the cache.
│ This may mean AutomotiveDrivingModels [99497e54-f3d6-53d3-a3a9-fa9315a7f1ba] does not support precompilation but is imported by a module that does.
└ @ Base loading.jl:947
┌ Info: Precompiling Cairo [159f3aea-2a34-519c-b102-8c37f9878175]
└ @ Base loading.jl:1186
We generate a 3-lane stadium roadway:
In [2]:
roadway = gen_stadium_roadway(3)
Out[2]:
Let's populate a scene.
In [3]:
scene = Scene()
push!(scene,Vehicle(VehicleState(VecSE2(10.0,-DEFAULT_LANE_WIDTH,0.0), roadway, 29.0), VehicleDef(), 1))
push!(scene,Vehicle(VehicleState(VecSE2(40.0,0.0,0.0), roadway, 22.0), VehicleDef(), 2))
push!(scene,Vehicle(VehicleState(VecSE2(70.0,-DEFAULT_LANE_WIDTH,0.0), roadway, 27.0), VehicleDef(), 3))
car_colors = get_pastel_car_colors(scene)
cam = FitToContentCamera()
render(scene, roadway, cam=cam, car_colors=car_colors)
Out[3]:
Let's assign driver models.
In [4]:
timestep = 0.1
models = Dict{Int, DriverModel}()
models[1] = LatLonSeparableDriver( # produces LatLonAccels
ProportionalLaneTracker(), # lateral model
IntelligentDriverModel(), # longitudinal model
)
models[2] = Tim2DDriver(timestep,
mlane = MOBIL(timestep),
)
models[3] = StaticDriver{AccelTurnrate, MvNormal}(MvNormal([0.0,0.0], [1.0,0.1]))
set_desired_speed!(models[1], 12.0)
set_desired_speed!(models[2], 10.0)
set_desired_speed!(models[3], 8.0)
nticks = 100
rec = SceneRecord(nticks+1, timestep)
simulate!(rec, scene, roadway, models, nticks)
render(rec[0], roadway, cam=cam, car_colors=car_colors)
Out[4]:
We can use interact to inspect the simulation record. Note that the static driver just drives off the road.
In [5]:
using Interact
@manipulate for frame_index in 1 : nframes(rec)
render(rec[frame_index-nframes(rec)], roadway, cam=cam, car_colors=car_colors)
end
┌ Info: Precompiling Interact [c601a237-2ae4-5e1e-952c-7a85b0c7eef1]
└ @ Base loading.jl:1186
┌ Warning: The call to compilecache failed to create a usable precompiled cache file for InteractBase [d3863d7c-f0c8-5437-a7b4-3ae773c01009]
│ exception = Required dependency Knockout [bcebb21b-c2e3-54f8-a781-646b90f6d2cc] failed to load from a cache file.
└ @ Base loading.jl:969
Out[5]:
WebIO.mount(this.previousSibling,{"props":{},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"className":"field interact-widget"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{},"nodeType":"Scope","type":"node","instanceArgs":{"imports":{"data":[{"name":"knockout","type":"js","url":"/assetserver/3eba94e42381e338b63445bb6165a6b74dfff7fe-knockout.js"},{"name":"knockout_punches","type":"js","url":"/assetserver/0b60b3bed0558f340d8f36b2e588fc19a5331e53-knockout_punches.js"},{"name":null,"type":"js","url":"/assetserver/e375b04ed295e371b269e6282dd0ece1d8f25f7c-all.js"},{"name":null,"type":"css","url":"/assetserver/74ab3fe28fdc96d2cdde1dc9103cd558bf15dd78-style.css"},{"name":null,"type":"css","url":"/assetserver/2b9bd1a7f1fd27241339301563046b45b31a5a73-bulma_confined.min.css"}],"type":"async_block"},"id":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","handlers":{"_promises":{"importsLoaded":[function (ko, koPunches) {
ko.punches.enableAll();
ko.bindingHandlers.numericValue = {
init : function(element, valueAccessor, allBindings, data, context) {
var stringified = ko.observable(ko.unwrap(valueAccessor()));
stringified.subscribe(function(value) {
var val = parseFloat(value);
if (!isNaN(val)) {
valueAccessor()(val);
}
})
valueAccessor().subscribe(function(value) {
var str = JSON.stringify(value);
if ((str == "0") && (["-0", "-0."].indexOf(stringified()) >= 0))
return;
if (["null", ""].indexOf(str) >= 0)
return;
stringified(str);
})
ko.applyBindingsToNode(element, { value: stringified, valueUpdate: allBindings.get('valueUpdate')}, context);
}
};
var json_data = {"formatted_vals":["1","2","3","4","5","6","7","8","9","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26","27","28","29","30","31","32","33","34","35","36","37","38","39","40","41","42","43","44","45","46","47","48","49","50","51","52","53","54","55","56","57","58","59","60","61","62","63","64","65","66","67","68","69","70","71","72","73","74","75","76","77","78","79","80","81","82","83","84","85","86","87","88","89","90","91","92","93","94","95","96","97","98","99","100","101"],"changes":WebIO.getval({"name":"changes","scope":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","id":"ob_02","type":"observable"}),"indexString":WebIO.getval({"name":"indexString","scope":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","id":"ob_03","type":"observable"}),"index":WebIO.getval({"name":"index","scope":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","id":"ob_01","type":"observable"})};
var self = this;
function AppViewModel() {
for (var key in json_data) {
var el = json_data[key];
this[key] = Array.isArray(el) ? ko.observableArray(el) : ko.observable(el);
}
[this["formatted_val"]=ko.computed( function(){
return this.formatted_vals()[parseInt(this.index())-1];
}
,this)]
[this["changes"].subscribe((function (val){!(this.valueFromJulia["changes"]) ? (WebIO.setval({"name":"changes","scope":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","id":"ob_02","type":"observable"},val)) : undefined; return this.valueFromJulia["changes"]=false}),self),this["indexString"].subscribe((function (val){!(this.valueFromJulia["indexString"]) ? (WebIO.setval({"name":"indexString","scope":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","id":"ob_03","type":"observable"},val)) : undefined; return this.valueFromJulia["indexString"]=false}),self),this["index"].subscribe((function (val){!(this.valueFromJulia["index"]) ? (WebIO.setval({"name":"index","scope":"knockout-component-3a4089d5-9c0b-4915-ad77-f71390a7746e","id":"ob_01","type":"observable"},val)) : undefined; return this.valueFromJulia["index"]=false}),self)]
var obs = this.index;
var obsString = this.indexString;
obsString.subscribe(function(value) {
var val = parseFloat(value);
if (!isNaN(val)) {
obs(val);
}
})
obs.subscribe(function(value) {
var str = JSON.stringify(value);
if ((str == "0") && (["-0", "-0."].indexOf(obsString()) >= 0))
return;
if (["null", ""].indexOf(str) >= 0)
return;
obsString(str);
})
}
self.model = new AppViewModel();
self.valueFromJulia = {};
for (var key in json_data) {
self.valueFromJulia[key] = false;
}
ko.applyBindings(self.model, self.dom);
}
]},"changes":[(function (val){return (val!=this.model["changes"]()) ? (this.valueFromJulia["changes"]=true, this.model["changes"](val)) : undefined})],"indexString":[(function (val){return (val!=this.model["indexString"]()) ? (this.valueFromJulia["indexString"]=true, this.model["indexString"](val)) : undefined})],"index":[(function (val){return (val!=this.model["index"]()) ? (this.valueFromJulia["index"]=true, this.model["index"](val)) : undefined})]},"systemjs_options":null,"observables":{"changes":{"sync":false,"id":"ob_02","value":0},"indexString":{"sync":false,"id":"ob_03","value":"51"},"index":{"sync":true,"id":"ob_01","value":51}}},"children":[{"props":{"className":"interact-flex-row interact-widget"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"className":"interact-flex-row-left"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"className":"interact ","style":{"padding":"5px 10px 0px 10px"}},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"label"},"children":["frame_index"]}]},{"props":{"className":"interact-flex-row-center"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"max":101,"min":1,"attributes":{"type":"range","data-bind":"value: indexString, valueUpdate: 'input', event: {change: function (){this.changes(this.changes()+1)}}","orient":"horizontal"},"step":1,"className":"slider slider is-fullwidth","style":{}},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"input"},"children":[]}]},{"props":{"className":"interact-flex-row-right"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"attributes":{"data-bind":"text: formatted_val"}},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"p"},"children":[]}]}]}]}]},{"props":{},"nodeType":"Scope","type":"node","instanceArgs":{"imports":{"data":[],"type":"async_block"},"id":"scope-2485f7e2-9882-4c10-a16c-b044de8d4a90","handlers":{"obs-output":[function (updated_htmlstr) {
var el = this.dom.querySelector("#out");
WebIO.propUtils.setInnerHtml(el, updated_htmlstr);
}]},"systemjs_options":null,"observables":{"obs-output":{"sync":false,"id":"ob_07","value":"<div class='display:none'></div><unsafe-script style='display:none'>\nWebIO.mount(this.previousSibling,{"props":{"className":"interact-flex-row interact-widget"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"setInnerHtml":"<img src=''></img>"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[]}]})</unsafe-script>"}}},"children":[{"props":{"id":"out","setInnerHtml":"<div class='display:none'></div><unsafe-script style='display:none'>\nWebIO.mount(this.previousSibling,{"props":{"className":"interact-flex-row interact-widget"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[{"props":{"setInnerHtml":"<img src=''></img>"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[]}]})</unsafe-script>"},"nodeType":"DOM","type":"node","instanceArgs":{"namespace":"html","tag":"div"},"children":[]}]}]})
We can save the run to a text file. We achieve this by first converting the the Trajdata type and then exporting that.
In [6]:
listrec = convert(Trajdata, rec)
open("2Dstadium_listrec.txt", "w") do io
write(io, MIME"text/plain"(), listrec)
end
The file can be loaded in a similar way.
In [7]:
listrec2 = open("2Dstadium_listrec.txt", "r") do io
read(io, MIME"text/plain"(), Trajdata)
end
Out[7]:
Trajdata(101 frames)
In [ ]:
Content source: tawheeler/CarEM.jl
Similar notebooks: