-
Notifications
You must be signed in to change notification settings - Fork 1.3k
Enable type-checker support in js-ipfs and js-libp2p #2320
Comments
Typing would be useful. If this gains traction, I'd +1 for typescript, since flow is mostly just pushed by FB. |
There is multiple ways one could go about adding type-checker support and each has different value proposition. I would personally advocate to go full on and just add flow to a toolchain as it would give you all the benefits at the cost of contributir workflow. I have experience with starting to use comments initially before going full on and from my experience no developer involved wanted to go back. I understand that may not be ready to do that. So alternative could be to just add type annotations so that users of libraries will reap all the benefits and hopefully implementation would actually work as typed. I already submitted some pull requests to that end. Main issue with this approach is that type definitions and implementation start to get out off sync mostly because core contributors off the library don’t actually use type annotatins and type check can’t report any mismatches either. |
I’m also happy to lead this effort if there is a will to accept this. |
@RangerMauve It is true, but isn’t ts also just pushed by Microsoft ? I would prefer flow as type inference is far more superior and toolchain does not require full buy-inn.
Sadly there is no such tool as far as I’m aware but if one of the typecheckers is used writing just tyoe signatures for the other should be viable and less prone to be out of sync |
I am also happy to provide specific explanation of why flow type checker type inference is superior and what does that actually provides as consequence if that is a conversation worth having. |
TS is used by more than just Microsoft, for example, Google's Angular team found it appealing enough to focus Angular 2+ on it, and I, personally, have seen more projects using TS than using flow. |
@RangerMauve Well if you're talking about users there are ton of users of flow as well at Mozilla we use it quite a bit and most of the projects around React ecosystem tend to use flow for the reasons you can guess. That being said you're probably right and more projects use TS but I wonder how much of that is due to TS being just an older project. In terms of contributors to ts and flow projects they seemed to be mostly ms and facebook. |
This is why I would suggest supporting both, because otherwise there will be a constant stream of people arguing over which type system should be integrated with or not. |
Anyway here is a small list of objective reasons why I think flow is a better choice:
|
To be absolutely clear I'm happy to do the work necessary for flow integration. As of TS support, I think it would be great and someone should do it, it's just I can not contribute to that effort as it does not help me with my project. I hope you understand. |
Here is a list of PRs I'have submitted yesterday, before @victorbjelkholm asked me to open this issue: |
Copying conversation from that happend over at libp2p/js-peer-info#62 (comment) with @dryajov as discussing it here makes more sense:
|
So here is the list of options in which flow types could be added to the ipfs libraries: 1. Go full on with flow type checker.Pros
Cons
2. Comment TypesSame list of pros / cons as above, except you'd trade need for transpilation with a lengthy comments blocks inline (usually makes it column width 80 a huge challenge). 3. Write file.js.flow along the side of implementation.Pros
Cons
4. Add ipfs to flow-typed
Pros
Cons
|
UpdateAfter doing some more exploration I realized that without type information for pull-streams utility of having type annotations in ipfs libs will be fairly limited. So I started doing flow-typed annotations for pull-streams which as it turns out has a very very very generic API since it takes 1 or 0 But in order this to work type checker needs support for existential types and type variance as described in the prior comment typescript has neither and there for it would not be able to handle type mismatches like this: Note that Typescript here would treated both I just wanted to put this out here to make it clear why I'm pushing for flow over ts. Please also note that pull-stream has quite exotic API but this issue occurs in not so exotic cases, essentially any time you return generic function who's type parameters are refined by it's usage later. Edit: Changed second example to make my explanation somewhat more clear. |
Woa, tons of work and exploration here, thank you so much @Gozala 🙌🏽
💯 and also compliance tests to ensure that behavior is correct. We have had in the past experimented with transpilation steps to support old ES5 browsers and after a while no one could bare to debug transpiled code all the way. The situation was especially bad since we have the code super modular and sometimes making a change requires you to to I saw your suggestion to add Isn't there a way to check the type annotations on CI so that only well annotated code gets merged? This way the contributors would still be able to tinker with the code in plain JS and learn flow or get support to update the annotations just during the PR process. This would also avoid all the transpilation steps. Adding the flow type annotations would also make doc generation way simpler. 10x💯 for that 👌🏽 |
Here is even more not 💯 there yet but almost. Also I could use some feedback on whether my interpretation of APIs are correct. I did add bunch of
It's unfortunate you had a bad experience with that. It could be that things have improved since, at least I don't struggle given that source-maps essentially allow me deal with code I've written.
I'm not sure what setup did you use, but you can link as many modules as you like and compile only module you're making changes to. You don't need to build complete bundle. You could also have a watcher running that does compiling on changes. Here is my usual setup: https://github.com/Gozala/router.flow/blob/master/package.json#L40
Sure, I usually run type checker on CI to catch any errors in the contrib pulls that I might overlook. But then again flow will type check annotated files, there is no way to tell it that
I think you're trying to achieve two different goals, and I'll provide my input on each one separately: 1. Defer type annotations until PR is submitted.I mean it's entirely possible to defer dealing with type-checker it's matter of just not running it locally. In fact I have being getting contributions from people who did not even installed one & once CI reported errors they went I fixed them without installing type-checker. Please note that type-checking code is somewhat independent of transpiling (at least with flow). Now that being said, I don't think deferred use of type-checker is a good idea though. In fact that can make a terrible experience, it's a lot easier to deal types as you work because it forces you to stay checked, small mismatches have better error messages etc... Amount of errors one would get otherwise can be quite an overwhelming. Another thing is that type-checker also guides you towards better interface design mostly because issues become apparent even before you get to implement. In fact many (including myself) tend to write out interfaces before doing any implementation work, you can sketch out API even try it out before you get to implement it. 2. Avoid compiler toolchainI think only option where one could take full advantage of type-checker and avoid adding compilers into toolchain would be through flow comments as described above. In fact some editors (vscode does and I think atom as well) even syntax highlight and treats those comment blocks as if they were just code. I also noticed that some of the ipfs packages were using this already. You could possibly take it step further and define interfaces in separate files those will have only types so won't even load at runtime. I think over time you'll probably be inclined to just switch to non commented options with transipiling, but this might be a least intrusive way to start. I would still recommend trying transpiler setup in one or two packages and see how it works out. One thing I'm unsure with this route is how module imports going to work out, flow use native JS module syntax. I think there is some form of |
Given that js-ipfs and js-libp2p are designed to work both in Node.js and in the Browser, I take that you are suggesting that we always use node-inspector with all of our debugging so that we can get Source Maps for Node.js. This is not necessarily a bad thing, just one that we have to handle with docs on the README on how to debug one of these modules. It happened in the past, when we did the transpilation madness, that contributors kept getting lost while trying to fix issues.
Simple
👍🏽 for better tooling such a watcher. We are getting a full working group on Team Dev Enablement, @victorbjelkholm has been structuring the goals for this quarter and might have some ideas on how to get these tools out the door faster :)
"I think over time you'll probably be inclined to just switch to non commented options with transipiling, but this might be a least intrusive way to start." I do feel that is going to happen too. I just want to make sure that the upgrade path is smooth and that we get to level up our way to dev without adding major obstacles that will disincourage contribution and participation on code reviews.
John Dalton created std/esm to add support to ES Modules to Node.js, we explored it also here #951. It is not necessarily fully done but good enough to enable us to move forward, I think. |
I didn’t imply that. There is node source maps support package (can’t remember name, on the 📱right now) |
So workflow could be simpler even without wathcher. Just link once whatever you need to change (I assume that’s already a case) after changes run build in that package. Note that you don’t need to relink or rebuild package it was linked into (unless exposed API interface changed in incomplete way - major version bump in semver). |
Apologies for the delay in following up on this one. I believe we have different experiences using these tools and there are clear learnings that have to happen from both parts. In order to avoid investing to much time in painting this bikeshed, I have a more pragmatic proposal. Let's use follow for
I expect that we it we will also learn how to dev faster (debug faster) and have always up to date docs. The repos to upgrade are:
@Gozala I'll assume that you are still willing to take the lead on this endeavour. Pinging @vmx, @hacdias, @victorbjelkholm, @lidel, @olizilla and @alanshaw as reviewers. I believe everyone is interested on this :) |
@Gozala I'd like to see |
I'm not a flow or typescript user. I've never used flow but I have worked on typescript projects in the past and got a bit bogged down with annotations and I felt like it hindered my productivity. I found that annotations can get really tricky unless you only use them for the simple cases. Good type annotations should handle the tricky bits, but in my experience this means quite a significant investment in time to get right. I could put that down to my newness to typescript but that aside, I personally don't find joy in authoring the tricky bits - I wouldn't, for example, relish the idea of authoring type annotations for pull streams. I don't dispute that having type annotations would help some developers who use IPFS JS libs. I'm raising a concern that if we add annotations to the projects we're enforcing the overhead of learning and using flow type annotations on all current and future IPFS devs and contributors. It's something we'll have to maintain and use forever and it might deter contributions from the community. I'm sad to hear that |
@alanshaw thanks for commenting on this issue, I think it's important tea hear arguments form both sides of the argument. I am also sorry to hear you had bad experience with typescript projects in the past, while I do realize no words could counter that experience, if you let me and keep an open mind I'd try to pursue to give it another chance. Here some of my comments inline below:
You are absolutely correct that type annotations can be of arbitrary complexity starting with obvious and simple and picking levels of absurd, where understanding code is actually easier than understanding those type annotations. But I would argue it's not inherent to usage of flow or ts, like any tool they could be misused and abused. More on that below
That is a perfect example actually, as I mentioned earlier type system tends to guide you towards a simpler API design, what I meant by that is if you end up designing an API like pull-streams you'll quickly realize that having same function entry with many different input combinations and returning many different kind of results is painful (for reasons you mentioned), so usually you face a choice of either making dead simple API choice (have a separate function for separate input / output combo), make your type annotations complex, or just bail from type safety (both ts and flow allow that but I'll cover that a bit later). But to be fare it's not really type annotations there to blame but rather an API choice than needlessly mixes different procedures into single one, in practice it does 4 different things:
I mean no offence to the pull-streams library but having four distinct functions would make reading code using pull-streams a lot easier. And if that was the case type signatures would have being pretty trivial as well. Now on the relaxing type guarantees bit. Both ts and flow allow you to choose how much safety you'd like to get from the type system & make your choices accordingly. For instance for the pull-streams I really wanted to get no runtime error guarantee from flow which is why end up encoding all the combination of arguments it can take (well up to 9 arguments). So that if I try to say provide two sinks instead of sink & source I get type checker tell me that. I also wanted to make sure that if I have through from
That is entirely a valid point that it would require some degree of type system understanding for non trivial changes (by which I mean extending public API), but I don't necessarily agree in regards to changes that keep public API intact, given that there type annotations are purely optional and fully inferred, I also had contributors that knew nothing about flow or cared to install it successfully submitting changes, (sometimes changes required reword as flow in CI caught subtle issues, but I think that's a good thing). Where I do disagree is though whether it is a good thing or not. I think you're viewing it purely in making usage of ipfs easier, while I view it as it would make it so much easier for people to contribute and making changes without fear. I am hesitant to state it as a fact but from my personal experience and experience of others I worked with static types do help make large projects more maintainable as you don't have to have a clear picture of the whole system, types provide you necessary context.
If the goal is to just make ipfs more accessible to it's users, I actually agree. Forcing everyone to use tool that they wouldn't otherwise use just to make ipfs more usable for few using does not seem like a right trade-off and flow-typed (however good or bad it might be) would make more sense. That being said I'd argue that ipfs contributors could take a huge advantage by adopting a type-checker given the size of the project and layers involved. It is also worth considering that bunch of libraries I think what is being proposed by @diasdavid is to take small subset of libraries and try type-checker on those, that way we could actually learn all about pros & cons it provides and draw informed conclusions from there. Without this it's just speculation. Unless of course there is general agreement that there adding types is just for sake of ipfs users rather than contributors. |
👍 I can't agree more! As mentioned earlier there is no way to draw informed conclusion without trying and learning from that.
I am not sure how far I'll be able to get this (due to time constraints) but I'm definitely eager to lead this effort till conclusions can be drawn on it's merits. |
While it's entirely possible to start from there, in my experience it's more productive to start from the leaf dependencies, so maybe it would make sense to start from one that has no or least dependencies ? |
any plans for typescript support as well? |
@gozes Currently this effort is focusing exclusively on Flow. There are no plans for TypeScript support at the moment. |
Closing as this was done via JSDoc types. |
Hi Folks,
Lately I have being trying to use js-ipfs in a project and got somewhat overwhelmed by amount of libraries at play, don’t get me wrong, I like the seperation of concerns this layered setup provides, it’s just incredibly hard to pin down what are the things being passed around and I’ve being following a rabbit hole for multiple days now. I also noticed the pattern across the stack where there are defined interfaces like transports and ipdl-resolvers, etc... and multiple implementations of them.
This got me thinking that it would be exteremly helpful if those interfaces were not just written out in markdown but were typed as flow (or typescript) interfaces than implementation could in fact declare that class / object implements it and type checker will make sure that is the case.
As I have being struggling to understand shapes of things being passed around (argument types, return types) I really started wishing those were tyoe annotated as that would have saved me so much time jumping across libraries to look things up. In fact I noticed that I had to look same things multiple times because as I kept going amount of info overflowed my mental capacity. That is yet another time where I wished type signatures were available as that would allowed my editor to just tell me what the interfaces is without following dependency chain and looking things up. In fact it got so difficult that I end up writing type signatures myself to save time redoing same lookups over and over.
Another controversial but still argument worth making is that there have type checker help make API’s clearer and lmby consequence easier for users. For instance I encountered a function that either takes protocol and buffer and returns string or takes protocol and string and returns buffer, it’s cool but consequently margin for error is higher + has runtime costs. Once you start using type system it navigates you towards less ambiguous API as concrete return types save you from type refinements down the line.
I also noticed multiple cases of polymorphic APIs where funtion needs to match over the type of argument and those checks are somewhat optimistic (as in no guarantee that argument is actually implements expected interface) that is another area where type checker like flow would greatly help. It provides a specific way to refine types and provides guarantees that with in the refined block value implements interface of that type it also ensures that no variant is being overlooked (meaning that each variant of argument type us handled).
The text was updated successfully, but these errors were encountered: