-
Notifications
You must be signed in to change notification settings - Fork 1.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Do we want a new Juggler? #537
Comments
The current version of
When we discuss what should be or should not be supported out of box in LB next, these different perspectives should be separated to avoid confusions. We have planned for refactoring the juggler into separate modules. The |
Let me start by admitting my mixed love/hate relationship with juggler. On the brighter side Having the standard ORM in LoopBack allowed us to implement great tools simplifying building of LoopBack apps:
On the darker side: There is a reason why the SQL language cannot be used for NoSQL databases: each NoSQL database has a different approach to address the CAP theorem and therefore requires a different programming model and mindset. For example, MongoDB prefers partial updates using operators like
My takeway In LB Next/4.0, we are decoupling REST API from the ORM API. This will take away benefits like single definition for both REST API and ORM, while bringing other advantages like giving developers tighter control of there public REST API. In that light, I think we are pretty much ready to abandon juggler, if we can find a way how to preserve the following features:
While we are discussing alternative ORMs for SQL backends, I'd like to bring the following projects into attention:
|
ORMIMO the biggest challenge to build an ORM would be we have to either build a "perfect" one or don't do it. By "perfect" I mean:
Compare with other existing ORMs from community, it's not hard for us to come up with a better design&implementation in a specific area, but given the resources we have, my concern is how much time do we need to build an OVERALL better ORM... And if we turn to be more determined on closing features that are not reasonable for us to support, would that benefit users more than telling them from the beginning to spend some time on investigating the most appropriate libs/modules they need in product? Sugar functionsIIRC we have a story discussing simplifying functions provided in Remote method hookActually it's now implemented in loopback core, I love the hook system and I assume loopback-next already implements it. ScopePeople may still want to have a set of apis organized under a certain name or say tag, and also easy to reuse when extending model. Inclusion, Getter and Setter, 2nd level CacheInspired by this article and the "updateOnly" PR recently merged into juggler, I think what limited by our current resources are those things lead us to build SQL/NoSQL queries, but we still need a module serve as a middleware between the modelDef and a db's driver functions.
|
To echo @jannyHou's comments, I propose that we first build a list of features/responsibilities for the current Having a big-bang/wholesale yes/no debate is NOT going to be very productive, IMO. |
I don't think it was ever about all of the juggler; many of the parts of juggler v3 have already been spoken for as separate modules within loopback-next, like authentication. I'm mostly using the term juggler for the persistence and relations since they don't have their own names. I do agree that we should make that list anyway, though figuring out exactly what is affected by the greater whole is difficult to talk about, and easier to demonstrate, which is why we're working on a "real" app to start testing out these use cases: https://github.com/strongloop/chit-chat |
Sorry if this out of scope for this issue but I just want to ask whether auto-discovery of models and relations is still part of the planned feature set of the new Juggler (which it seems to be heading towards)? Discovering models and relations based on existing tables is a major part of our current workflow with Loopback as we have hundreds of "old" tables that need CRUD APIs. Hand writing each model and method would make this framework almost unusable here. I saw someone else asking this in a referenced issue but he didn't get an answer (#419 (comment)) |
@ExTheSea This is one of those design decisions that would be influenced by the way we choose to implement and support the persistence layer. If we decide to continue providing our own ORM, it would mean that we would also be responsible for the discovery and migration stories that are a part of loopback@3. My current proposal is to use mixins for popular ORMs, as well as templates to help auto-generate code for users based on their chosen protocol (REST, gRPC, MQTT, etc) and chosen mixin. We're currently hashing that out as a team and any feedback for either approach would be welcome. If you have any questions about what my proposal would entail, just ask. :) |
So, as a team, we came to a decision yesterday regarding our approach here and this is what we've come up with Roadmap for JugglerWe will be keeping the Juggler as a part of LoopBack, but we will be constraining its scope for the next major release. Planned Changes
Other ORMsWe will provide some tutorial materials on how to create your own mixins to make use of your own ORMs, though we will not provide templating support or other materials to ease in the use of those ORMs. |
Additional Questionscc @strongloop/loopback-devs
|
Another question: Will we constrain the number of relationships to something simpler than before? |
|
|
Thank you @kjdelisle for writing down the proposal, and @virkt25 and @jannyHou for your comments. I'd like to add few more thoughts to consider. First of all, I think we should make juggler a first-class package that can be used outside of LoopBack too. We have interesting features that are not available in other ORMs - see e.g. #776 (comment) and the feature comparison between TypeORM and Juggler that @raymondfeng wrote but which I am not able to find now :( (@raymondfeng - could you please post link to your table here?) We should be promoting our ORM more too, so that when people learn that LoopBack uses Juggler as the default ORM, they won't think "why are they using this ORM I never heard of instead of ", but instead they will understand Juggler is a well-known fully-featured ORM and we have to pick one anyways.
I want us to work on the "new" juggler incrementally. I really want to avoid the situation we have here in loopback-next, where we spent 12 months building a new version from scratch and there is still nothing that our users could use in production. Instead, I am proposing the following approach:
I personally see a lot of value in having a monorepo that contains Juggler and all connectors we are maintaining. In my past experience, it was cumbersome to add new features to Juggler, because a PR to juggler would have to be accompanied by 10+ pull requests to our connectors to implement support for that new feature. Sharing the test suite between juggler and the connectors had it problems too, how often we could not land a pull request in one repository because the tests were failing until another pull request was landed somewhere else? Having the ability to test all connectors together with any change made in juggler will simplify our life too, as we won't have to rely on cis-jenkins dependency-based-triggers anymore. (cis-jenkins has two issues: a) it can be slow to start downstream jobs b) test results are not visible to community (non-IBM) contributors ). The downside is that running all connector tests will add significant time overhead to What I think is a more important question is whether Juggler and connectors should live in loopback's main monorepo, or whether they should have their own monorepo? If we want to promote Juggler as standalone ORM, then it may make more sense to let it have its own monorepo, own issue tracker, etc. (Another benefit of a different monorepo is that we can defer implementation of the CI tooling I mentioned above for a while, because Last but not least, I think we should find a new name for our ORM, perhaps one that's not so coupled with LoopBack. How about "Juggler ORM"? (I am already imagining a cheerful logo of a circus artist juggling with balls 🤹♀️🤹♂️, where each ball can be a logo of a different SQL/NoSQL database.) Few more alternatives that come to my mind: "Strong ORM" to keep StrongLoop's theme of prefixing modules with "Strong", "LoopBack ORM" to keep the association with LoopBack, or perhaps
+1 for simplifying things. I think there will be many more opportunities to simplify things. For example, embedded relations have always had a lot of shortcomings, they may be a good candidate for removal too. |
We discussed the next steps for this issue with @kjdelisle and come up with the following plan:
We will create follow-up issues later. There should be a special Epic to group these issues together. |
We (@kjdelisle and me) have created follow-up issues with the exception of the step 7, we will handle publishing as part of our regular work. |
I've been thinking about the persistence story in LoopBack over the last few months and have been wondering if the juggler is the right approach for the framework.
Warning: Opinions Ahead -- Please don your 3D glasses now.
Background
In the current iteration of loopback and loopback-datasource-juggler, we define relations between Models to provide much of the interoperability that users have come to rely on for making their REST APIs. These relations are supported by a Domain-Specific Language (DSL) that leverages that metadata to generate queries against datasources.
The Everything ORM
In many cases, our mapping of the DSL has required us to make an ORM-like set of facilities that generate query strings (in the case of SQL) or translate our DSL into one understood by the target database/database drivers (MongoDB, as an example). Support for this idea falls short of what many in the development community would expect for each of the individual use cases; SQL queries are often limited to primitive, inefficient operations and some NoSQL query objects do not accurately represent their original intention when translated accordingly. In many cases, users must fall back on basic methods that do not take advantage of the metadata our relations and DSL represent.
In each of these cases, whether or not it was our intent, we have implied to the community that we would shoulder the burden of providing a reasonably-complete and efficient set of query-generation tools to work hand-in-hand with our DSL to provide fine-grained control to various datasources.
In my opinion, the very idea of having our own DSL gives this impression; what purpose does it serve otherwise?
The Extensibility Problem (Routing around the Damage)
In the case of loopback-next, our extreme extensibility is a massive advantage, but it comes with the a significant "downside"; members of the community will simply avoid the Juggler if it doesn't meet their needs, and those that find it sufficient will keep using it.
"Great! Why's that a problem?"
Our Choice Informs Our Design
The decision of which approach we consider to be the best practice will have a direct effect on the design of the other core components of the framework.
As an example, one of the other things that loopback 3.x gives you with its relations is automatic generation of REST APIs based on your models. "Foo hasMany Bar"? Great! We'll generate all of the
/foos/{id}/bars
routes for you!If we decide that there will be no Juggler for 4.x, then that probably means no relations, too. That doesn't mean we won't have a story for generating those API definitions in some other way, but it does give you two very different approaches.
"We can keep relations without using the DSL or the juggler for persistence!", you might say. But would you want to create all of the relationships between your Models in a declarative format just to generate the shape of your REST APIs? Typing
/foos/{id}/bars
into a Swagger editor isn't any more difficult than making those relations. The combined value of getting that for free as well as being able to leverage that metadata to generate queries against datasources is what makes the sales pitch for the idea of having those relations to begin with.DSL? I Prefer Cable, Thanks
Convincing community developers to invest great effort into creating these middleware layers that translate our DSL into commands for various database drivers is a tough sell when ready-made ORMs can be liberally sprinkled into your code in loopback-next.
Many of them come with their own Model and relationship engines (sequelize being the example everyone is tired of me bringing up whenever we discuss this), and since they tend to specialize in their chosen domain, they're often very efficient, well-written ORMs that do one (or a handful) of things extremely well.
Competing with this would require us to once again take on the burden of building all of these connectors ourselves, only this time, we'd have to make damn sure that we're making solid-quality SQL statements, and properly translating our MongoDB queries because developers will simply swap to using the underlying drivers if they're not already in too deep. Combine this with the fact that we just don't have the resources to build all of this in a timely fashion and the value-for-money of this proposition suddenly seems thin at best.
What Other Circus Acts Are We Good At?
The juggler is definitely something that distinguishes us from other frameworks that are value-adds on Express.js, Koa.js and so on. I find myself wondering what other killer features we could provide that existing frameworks don't. It might be that what we need to differentiate ourselves in a meaningful way doesn't even have to be a particular feature; if our framework and tooling do a better, faster job of getting people from idea to API, and gives you easy flexibility to add all sorts of awesome components, wouldn't that be our killer feature?
Trade Chainsaws for Bowling Pins
It might be that someone who's much more creative/intelligent than me knows of a way we can have our cake and eat it too; if a design for the new juggler provides a low-effort API to implement that bridges the gap between driver and DSL, then I'm definitely on board with the idea.
** You may now remove your 3D glasses. **
I definitely want to hear everyone's thoughts on this, both the validity/invalidity of my concerns, as well as approaches we could take to solve this problem. Thanks for reading!
The text was updated successfully, but these errors were encountered: