Didier, I think your synopsis is right on the mark. I shifted from the canonical class-based data model for entities (primarily for discrete event simulation, with applications to games) toward a functional ECS approach a while back - even before I found clojure in 2010. The biggest utilities I find are a) flexibility in data definition(s) - add/remove components as needed, b) unified storage medium for entity information (like an in-memory database, with the ability to easily define aggregate queries and other relational stuff), and c) the ability to operate on domain-specific slices of entities (via selected components). There's a similarity between shifting to an ECS model and using clojure maps for generic data storage; lots of flexbility and unintentional re-use.
My (work in progress, but useful) implementation of ECS (from a melange of libraries called spork <https://github.com/joinr/spork/>): spork.entitysytem.store. <https://github.com/joinr/spork/blob/master/src/spork/entitysystem/store.clj> This is useful, albeit research-quality software that I'm still adapting to my needs and documenting. If nothing else, there may be pedagogical value for other folks. <https://github.com/joinr/spork/blob/master/src/spork/entitysystem/store.clj> I took the approach of pushing the entitystore behind some fundamental protocols, and evolved functionality as I ran into problems. Like relational databases, you can take a column or row-based approach to implementing an entitystore (where component data is defined). Depending on your needs and use-cases, either may be appropriate. I stuck with a column store, with IEntityStore protocols implemented on a record that stores information about entity component membership (has-a?) and the actual component domain data. So, we end up with a structure like {:entities {:id1 #{:name :domain2 ...}} :domains {:name {:id1 "SomeName"}}} as our "store". From there, we can define operations on how to add, drop, update, modify entries in the store. Convenience functions, like implementing sql-like queries are easy to implement too. You can probably stick datalog or other unification-based queries on the store since the structure is pretty easy to pick apart. I opted for clojurey names for operating on entity components, ala "assoce, updatee, mergee, assoc-ine, ..." which provide a familiar layer (for me) over the like-named clojure functions. When you stick with the intended use-case [designing "systems", i.e. functions that operate on one or more domain of component data, to compute new entity stores and the like] the scheme works really nicely. Specifically, systems that are tied to one component, or entity stores with a relatively small number of components that are cheap to join. About 1/2 way into fleshing out the design and putting it through its paces, I realized that i) I often don't care about all the components of an entity.....but I may not know that information ahead of time; it'd be nice to not have to compute joins for every entity every time on certain systems ii) updating entities by-component can be very inefficient, so batching updates would be nice.... This cried out for either a row-based implementation, or a pseudo-row based representation of an entity that avoided paying costs for computing joins across components, while allowing for the column-based properties of the backing store. I ended up implementing a lazy entity record, which provides a map-like reference to the entity's components but performs joins lazily and on-demand, and kept track of changed or dirty components smartly. This alleviated most of the performance concerns I ran into, and enabled me to define and additional set of operations on entities as if they were row-based, i.e. clojure maps. Regarding inheritance and some of the class-based things you brought up, I baked up a little DSL for defining entity constructors (really just functions making maps) that's deriving partially from the CLOS way of defining structs (with inherited fields and the like). I thought it was pretty slick when I first cooked it up (after reading Land of Lisp), but I don't really use it all that much. In practice, you can just create a map and shove it into the entitystore, which implies dissecting the key-vals of the map and associng like component entries for the entity (assuming a :name key exists). Or, you can create complex "class-like" hierarchies and use that to alleviate some of the burden. Or you can make your own composition of functions - operating on data - to achieve the same result. Currently, I have the entitystore paired with a behavior tree system that I use for entity behaviors / "ai". I'm still feeling it out, but it works and provides a way to compose sophisticated behaviors that gets away from the FSM spaghetti I ran into originally. In this case, the entitystore provides another advantage, since you can use it as a blackboard for communicating information between multiple concurrent entities. Rather than "firing" events and side-effecting, you can use the presence/absence of data via components to communicate. You can still implement messaging via component data as well. Communication (or simulating communication between entities) was one of the harder problems that seemed to be left as an "exercise for the reader" in much of the ECS literature/presentations... For me, the combination of an entitystore and behavior trees has ended up fairly robust in practice. It's even surprisingly performant for some real-time 60fps visualizations I've done (tied to a simulation), where I thought I'd have framerate problems. I'm still optimizing (for faster simulation runs), looking into varying implementations for the store - mutable, async, row-store. On Tuesday, August 15, 2017 at 7:52:38 PM UTC-5, Didier wrote: > > I recently stumbled upon the entity-component-system design pattern which > is popular in game engine design: > https://en.wikipedia.org/wiki/Entity%E2%80%93component%E2%80%93system, > and really liked what I saw, thought it could be a good fit for Clojure. > > Basically, it has three concepts: > > 1) Components are pure data grouped together as per a given domain. In a > game engine that would be for example the 3D positional data related to > positioning of objects in the 3D scene. So one component would be > PositionComponent and it would have :X, :Y. > > 2) Entities are collections of Components with a unique ID. > > 3) Systems are processing functions that take an entity, transforming > their components' data, or performing side effects from them. > > Generally, in games, they inverse the entities, so instead of having > entities contain components, they have components stored in an array with > the index being the entity ID, and another array which contains the set of > components for the entity at that index. All of this is kept track of by a > world container. > > (def world > {:entities [] > :comp1 [] > :comp2 [] > ...}) > > > So say you want to create an entity which is composed of comp1 and comp2, > you would just add to the world :entities at index 0 a set #{:comp1 > :comp2}, and to the world :comp1 and :comp2 vectors at index 0 an initial > component1 and component2 data structure. In games, for performance, they > use a bitmask instead of a set for the entry of :entities. > > > I'm not sure this structure is necessary if trying to use the pattern not > for game, but it doesn't hurt either I think. > > What I like about this, is I'm thinking its possible to use it to do > data-driven functional object modeling in Clojure. A problem I face, and I > feel other faces in Clojure, is how do you model entities without OOP? I > find this creates a kind of OO that is functional and data driven. > > You would spec a bunch of component, they're pure data. Then you'd define > systems (aka functions) which take an entity, and operate on the entity's > components (aka its data). At first glance, this appears to just be OOP, > but there's no inheritance here, and functions that operate or related data > are decoupled from the data. Systems are implicitly mapped to components, > based on what they work on. So you can extend all entities with more > functionality easily. You can also create entities from components on the > fly. > > On second glance, I wonder what's different about this from just functions > operating over data. I think its just a more rigid means to do so when you > need the concept of entities. In a way, entities act as a class, in that > they're a template of data. A system works over that template. > > Has anyone experimented with this in Clojure? > -- You received this message because you are subscribed to the Google Groups "Clojure" group. To post to this group, send email to clojure@googlegroups.com Note that posts from new members are moderated - please be patient with your first post. To unsubscribe from this group, send email to clojure+unsubscr...@googlegroups.com For more options, visit this group at http://groups.google.com/group/clojure?hl=en --- You received this message because you are subscribed to the Google Groups "Clojure" group. To unsubscribe from this group and stop receiving emails from it, send an email to clojure+unsubscr...@googlegroups.com. For more options, visit https://groups.google.com/d/optout.