| |
| ▲ | tossandthrow 3 days ago | parent | next [-] | | And this is definitely a problem. Had I had the opportunity to choose a language across the entire stack with mature wide adopted frameworks and libraries, I had done it. Had there been something line Rust, Go, Java, C#, etc. that would work end to end, that would have been amazing. In practice, even the weak safety typescript provides catches so many bugs before they hit production that it is indeed worth it - I have more than 140k LOCs of Typescript in production, and that would not be manageable without types. | | |
| ▲ | CharlieDigital 3 days ago | parent [-] | | > I have more than 140k LOCs of Typescript in production, and that would not be manageable without types
The Svelte team achieved it with JSDoc. Google's JS style guide also focuses on JSDoc for the same reasons[0].And to be just a tad pedantic: you have JS in production; your TS is only in dev. > Go, Java, C#, etc. that would work end to end, that would have been amazing.
It's not that you can't; it's that you choose not to (and yes, generally for good and valid reasons). There are end-to-end solutions for C# (e.g. Blazor), for example, that are perfectly fine depending on your use case (not great for all use cases). Fable is another example using F#[1]There are also libraries like Bootsharp[2] that are doing interesting things, IMO, and has some of the same energy as OP's project (moving the typing and logic into a runtime that supports runtime static types and interfacing minimally with JS) [0] https://google.github.io/styleguide/jsguide.html#jsdoc [1] https://fable.io/docs/ [2] https://sharp.elringus.com/ | | |
| ▲ | tossandthrow 3 days ago | parent | next [-] | | > you have JS in production; your TS is only in dev. As my sibling says, this is wrong. We indeed have TS in production. Even for the parts that are being compiled to JS: You wouldn't say: You can not have C++ in production, only binaries. The fact is that we don't write any JS as a part of our platform. > It's not that you can't; it's that you choose not to I think I made that quite clear in my comment. > a language across the entire stack with mature wide adopted frameworks and libraries | | |
| ▲ | CharlieDigital 3 days ago | parent [-] | | > Even for the parts that are being compiled to JS: You wouldn't say: You can not have C++ in production, only binaries.
I would say that because it can be decompiled; the type information is still present. Same with C#. I can decompile the binary and still see the type information.The process of going from TS to JS is lossy; you cannot get back the type information. I would absolutely say "I have C++ in production" or "I have C# in production" but not say "I have TypeScript in production". "We build our app with TypeScript" is accurate, but it is transpiled -- not compiled -- into JavaScript. Your Node.js server then interprets that JavaScript and executes C++. | | |
| ▲ | tossandthrow 3 days ago | parent | next [-] | | That is not right, When you compile to binaries, you do type erasure in CPP, som of the stuff can not be reconstructed or inferred, especially when using o flags. JS -> TS is easy, you just re-anotate with `: any` everywhere. Anyways, you are word juggling now. | | |
| ▲ | Nijikokun 3 days ago | parent [-] | | Type definitions are definitely compiled with CPP. Type erasure only happens with polymorphism types. You can actually view the types with ghidra. |
| |
| ▲ | unchar1 3 days ago | parent | prev [-] | | > Your Node.js server then interprets that JavaScript and executes C++. Umm...no? V8 specifically compiles it into machine code directly. There used to be a pseudo-translation layer in the CrankShaftScript days, but that hasn't been true in almost a decade. > I can decompile the binary and still see the type information. Also no. The de-compiler can _infer_ the types, much like how V8 tries to infer the type. But the actual type information is gone. Even in languages like Java where most of the type information is preserved, some things are still lost (e.g. generic methods with dynamicinvoke) |
|
| |
| ▲ | gcau 3 days ago | parent | prev | next [-] | | >And to be just a tad pedantic: you have JS in production; your TS is only in dev. This is not even pedantic, it's wrong. You have JS/TS in both dev and prod, with javascript being the actual runtime code in both, and typescript checking your code at build time in both. If you're running javascript compiled/checked by , and written in, typescript, it's not uncommon or unreasonable to call it typescript. | |
| ▲ | mubou 2 days ago | parent | prev [-] | | > Google's JS style guide also focuses on JSDoc for the same reasons Google uses TS internally. Any JS is strictly legacy code at this point. The link you posted even says this at the top: > Please note: This guide is no longer being updated. Google recommends migrating to TypeScript, and following the TypeScript guide. Also, Google's style guides are not really authoritative outside of Google. They are for some people, who choose to adopt them as their own, but you really shouldn't point to them as the end-all-be-all argument-stopper. Google has a lot of its own idiosyncrasies. Their C# style guide puts braces on the same line, for example, because Google primarily uses Java internally. |
|
| |
| ▲ | unchar1 3 days ago | parent | prev | next [-] | | > To be clear, they are not "using" TypeScript, it's more accurate to say they are providing TypeScript bindings. Interesting that you say it's more about providing "bindings", and not really "using". Much of the types in the svelte codebase are never exported outside of svelte, and they are only consumed internally. The problem they were having was with transpilation, since the browser doesn't run JS. From Rich Harris (months after svelte switched to JSDoc) [1]: > removing types from your own code is clownish, epically misguided behaviour, but whatever — to each their own I would suggest going through the issues and PRs in the codebase to see how invested the Svelte team is in typescript. > TypeScript is not "real" static typing in the same sense as Go, Rust, C#. That is true for Go, Rust, C#. But the same thing is also true for languages like C, and Generics in Java.
I'm sure both of those languages have weak type systems, but are definitely statically typed. I think the fact that type information is lost after being compiled isn't really a classifier for typed/non-typed. Ultimately it all comes down to machine code, and that certainly isn't typed either. [1]: https://x.com/Rich_Harris/status/1699490194565578882 | |
| ▲ | tshaddox 3 days ago | parent | prev | next [-] | | Does Go’s type system perform runtime type validations? My impression was that it does not. But Go probably supports reflection on the type information at runtime which could be used to implement a runtime type validator, right? | |
| ▲ | empw 3 days ago | parent | prev | next [-] | | You can write a function in go or rust, then write code in assembly to call it with any old nonsense. It is no different. The whole point of static typing is that it happens at compile time, not runtime. | |
| ▲ | MrJohz 3 days ago | parent | prev [-] | | > TypeScript is not "real" static typing in the same sense as Go, Rust, C#; the type information disappears the moment you build it. The type information in both Rust and Go disappears the moment you build it. The generated assembly is completely untyped: if you pass invalid objects to a compiled Rust function that you've dynamically loaded, you will cause problems, even memory safety issues, because Rust does not do any type checks at runtime. In the same way, if you call Typescript-defined functions from Javascript with the wrong types, you'll get problems. But if you write everything in Javascript, and don't ever use type assertions, you won't have issues. (In practice, I believe there are a couple of other issues you can run into, typically involving variance and mutability, but these are very rare. Also, some older APIs return `any` that should return `unknown`, but that can be fixed.) It's also worth keeping in mind that all languages have and require schema validation, it just might look different. Both Rust and Go have schema validation libraries - typically they parse the data directly into the correct format, but in Rust you could do something like `let x: HashMap<String, JsonValue> = string.parse()` and get roughly the same effect as Javascript's JSON.parse, it's just that JS/TS has a built-in data structure to represent arbitrary JSON data, whereas Rust does not. > To be clear, they are not "using" TypeScript, it's more accurate to say they are providing TypeScript bindings. To be clearer: they are absolutely using Typescript. The functions (and other parts of the code) are annotated using Typescript's extension of JSDoc types, which means Typescript can parse and typecheck this code as if it were "normal" Typescript, including generating type definition files. The .d.ts files in the source code are used to define additional types and interfaces that can be used in the .js files (you can see the first file you linked to import type definitions from ESTree, for example). The type checking step can be seen in the package.json file[0]. This is all explained in the comment from Rich Harris that you linked before. Using JSDoc rather than conventional Typescript syntax makes writing your source code more complicated, but in this case the Svelte team figured it would benefit them more in the long run to still be writing code that could be interpreted by a normal Javascript runtime. However, it really is just a different syntax for Typescript, and that's how they are using it. [0]: https://github.com/sveltejs/svelte/blob/80557bbc1c8a94c43a95... |
|
| |
| ▲ | recursive 3 days ago | parent | next [-] | | This is the real-est static typing that exists. "Static" refers to build time. By definition static types are checked at build time, not run time. If you want types to be checked at run time, that's called "dynamic" typing. | | |
| ▲ | mplanchard 3 days ago | parent [-] | | Sure, you're technically correct. But TS is compiled in to JS, which is dynamically typed at runtime. You're still ultimately in a dynamically typed language. In addition, in a typical statically typed, compiled language, your only place where you interact with data that isn't guaranteed to be type-conformant is at a foreign function interface, whereas in Typescript all your interaction with third-party libraries is via regular JS and may or may not be type conformant. | | |
| ▲ | throwaway894345 3 days ago | parent | next [-] | | Rust compiles into WASM, x86, etc, which are dynamically typed for all intents and purposes. They certainly don't understand or enforce Rust's type system invariants any more than JavaScript enforces TypeScript invariants. If you call a Rust function from WASM and pass it malformed data, the WASM runtime will happily execute it. (pretty sure WASM actually has some integer types, so I guess maybe it is technically "statically typed" but not in any interesting sense--we could similarly say that JavaScript is "statically typed" because every variable has a static "any" type). | |
| ▲ | recursive 3 days ago | parent | prev | next [-] | | Is there such a thing as a statically typed language? CPU opcodes don't type-check their parameters. | |
| ▲ | johnfn 3 days ago | parent | prev [-] | | And Go/C/Rust compiles to ASM, a dynamically typed language at runtime. |
|
| |
| ▲ | oynqr 3 days ago | parent | prev | next [-] | | > The types only matter during compilation. That's pretty much the definition of static typing. | |
| ▲ | throwaway894345 3 days ago | parent | prev [-] | | You're confusing "real static typing" with runtime type information. TypeScript has "real static typing" (if you disagree, ponder for a moment the meaning of "static"). A TypeScript program cannot natively query the _TypeScript type_ (not to be confused with the corresponding JavaScript type) of one of its variables in the way, for example, Go can. But neither can C, or C++, or Rust, all of which are unquestioningly statically typed. |
|