Blorp Language

(blorp-lang.org)

71 points | by croottree 1 day ago

15 comments

  • xigoi 22 hours ago
    Why is `pure` a keyword that needs to be added, with impure being the default? This discourages programmers from marking functions as pure. I like how Nim does it, with `func` declaring a function (pure) and `proc` declaring a procedure (impure).
    • rtfeldman 20 hours ago
      Roc defaults functions to being pure, and functions that can run side effects are inferred to have a different type by the compiler based on usage. By convention, their names should also end in `!` (e.g. `transform` for the name of a pure function and `transform!` if it does side effects), and the compiler warns you if you don't follow that convention.

      https://github.com/roc-lang/roc/blob/b2503210da6b58a4ce1254d...

    • adamddev1 21 hours ago
      I also really like the distinction between a function and procedure. The function is a pure mathematical function. The procedure is a series of instructions.
    • onlyrealcuzzo 22 hours ago
      I would also recommend this default.

      We want languages that encourage good design.

      If your goal is - like Crystal - to be as pain free of a migration from Python to Blorp, this shouldn't really impact it, since the compiler can and should be able to auto-fix this.

    • miroljub 20 hours ago
      > Why is `pure` a keyword that needs to be added, with impure being the default?

      Marketing.

      Instead of reading the code littered with "impure" keywords, you look at the beautiful code marked as "pure".

      • pepa65 6 hours ago
        An exclamation mark instead for impure would solve that.
      • adastra22 17 hours ago
        The LLM is just going to leave out the pure by default.
  • mapcars 21 hours ago
    Readable syntax with mandatory indentation is a very questionable idea. For me its easier to understand that something ends with a specific designation, not with a lack of it. Indentation should be solved by formatter and not the language.

    And I don't quite understand the memory model, is it something similar to Rust?

    • MarkusQ 21 hours ago
      That's more of a compiler limitation that became cultural for a while. Most languages (both natural and artificial) use delimited structures sparingly and rely more on other cues. It sometimes appears spontaneously (e.g. "∫ dx f(x)" is logically fine, but feels wrong) but in general it's rare.

      The move away from indentation in programing came as a rebellion against the too-constraining fixed column languages, in the interval between punched cards and python, with a brief resurgence in the early blink tag and font potpourri web era. These days, it's perfectly reasonable.

      • mapcars 19 hours ago
        In my experience there are many problems with significant whitespaces, things like copying pieces of code require much more work, when indentation actually changes the logic you can not ask your tool to do it automatically - because there is no single right way to do it. Tabs vs spaces can also be a problem.
        • xigoi 6 hours ago
          > copying pieces of code require much more work

          IME, it requires less work. You just grab the piece of code you want, whereas with braces, you need to count which closing brace is the correct one.

      • 3836293648 19 hours ago
        No, significant whitespace today is still terrible, coming from Haskell, Python and Go (newlines that break expressions unreasonably)
      • MarkusQ 18 hours ago
        Apparently the truth hurts.

        I can't believe that the claim "python-style syntax is perfectly reasonable" deserves to be down-voted into the basement.

  • mauvehaus 21 hours ago
    "Blorp" is the notional noise of kimchi or sauerkraut fermenting as the carbon dioxide escapes the airlock. Vigorous fermentation can be described as "the kimchi is really blorping along today". It's almost onomatopoetic, but not quite.

    We ferment wine or beer in a different vessel with different airlock, so it does not blorp. We don't have a word for that yet.

    The crock we used that birthed this word is this one:

    https://www.lehmans.com/product/striped-european-style-ferme...

    • thaumasiotes 21 hours ago
      > "Blorp" is the notional noise of kimchi or sauerkraut fermenting as the carbon dioxide escapes the airlock. Vigorous fermentation can be described as "the kimchi is really blorping along today".

      I find it difficult to believe that a noise assigned to the preparation of kimchi would be so flagrantly incompatible with the Korean language.

      • mauvehaus 21 hours ago
        I'm a basically monolingual white guy from the Midwest USA, transplanted to Vermont. I discovered kimchi in a restaurant and learned to make it from the internet[0].

        I'm sure people who come by their kimchi-making through their family or culture natively probably have words that work better for them that I would stumble over and mangle is truly epic fashion :-)

        [0] https://www.maangchi.com/recipe/tongbaechu-kimchi

  • bobajeff 22 hours ago
    I know there are people that are used to the indention based scope but that has a real problem when it comes to copy/pasting code. I think a alternative that still looks pretty clean is to do like Ruby and Julia and have the function/class imply begin and have a literal 'end'.
    • nchammas 20 hours ago
      I don't understand this concern. How exactly are you copy/pasting code such that significant indentation causes "real problems"?

      I remember the creators of Go explained [1] that they chose explicit block delimiters because of problems they saw when embedding snippets of Python in other languages. But this seems like a very niche kind of problem.

      [1]: https://go.dev/talks/2012/splash.article#:~:text=we%20have%2...

      • bobajeff 16 hours ago
        Scope being determined by indention makes it easier to paste things in the wrong scope and not notice.
    • mohragk 21 hours ago
      Fun fact, in Python, the indentation is checked per block. So, in the outer block, indentation can be 2 spaces, while in the inner block, the indentation is 3 spaces. The only prerequisite is that the indentation in the block is the same.

      This, to circumvent copy/paste issues.

    • xigoi 22 hours ago
      If your editor messes up indentation when copy-pasting, you need a better editor.
      • Narishma 21 hours ago
        You paste code in more places than just your editor.
  • cupofjoakim 1 day ago
    Interesting. there are some parts i like a lot here, but two things that I really dislike syntax wise. One is the lean towards a chainable syntax - this has proven to a big footgun for many devs in both java streams and typescript, making it very easy to go from O(n) to O(2n). The other part i really dislike is the first argument principle noted. If i myself define `string_and_reverse` and I can call it both through `string_and_reverse(42)` and `42.string_and_reverse()` i could definitely see this leading to some very funky looking chaining.

    Perhaps it's just one point from me - not liking chaining :D

    • KolmogorovComp 1 day ago
      > making it very easy to go from O(n) to O(2n)

      Strictly speaking I assume everyone knows O(n) = O(2n) =O(kn) for k in R.

      But I see your point. I assume any decent compiler would merge the loops though

      • cupofjoakim 22 hours ago
        Fair! That'd depend on the operations right? For example, AFAIK typescript can't do much about multiple chained `map` calls, and i've seen quite a few `.filter(...).map(...).filter(Boolean).map(...)` :/
        • c0balt 22 hours ago
          To be fair this likely should be handled by the interpreter/compiler for the compiled JS. V8 probably can merge this into one loop or another similar based on runtime types
      • frwrfwrfeefwf 21 hours ago
        what if k = n
        • MarkusQ 21 hours ago
          Then either n is a constant and it's really O(1), or k isn't a constant and its naming was in violation of the International Conventions on Naming Things to Avoid Silly Arguments, section 3, paragraph IV.7 & ff.
    • xigoi 21 hours ago
      > i could definitely see this leading to some very funky looking chaining.

      At least for me,

        thing
          .doThis()
          .thenDoThat()
          .andFinallyThis()
      
      is much more readable than

        andFinallyThis(
          thenDoThat(
            doThis(thing)
          )
        )
      • cupofjoakim 21 hours ago
        Well, nesting is not the only option.

        ``` thing.doThis() thing.thenDoThat() thing.andFinallyThis()

        // or

        doThis(thing) thenDoThat(thing) andFinallyThis(thing) ```

        • xigoi 20 hours ago
          That’s not equivalent.
  • kgeist 21 hours ago
    I applaud the effort, but every time there's a new hobbyist programming language on HN, almost always it's something I've already seen in countless other hobbyist languages, just a slight variation of it based on the author's personal tastes. It doesn't tell me why I should adopt it over language X. What I'd like to see is exploration of novel practical ideas that would make certain types of projects much faster to write/read compared to most other languages.

    For example, a typical web service I work on:

        - uses JSON APIs
        - it's fully stateless (uses external DBs/caches for persistence)
        - has the concepts of value objects, entities, architectural layers (app, domain, infra), ports/adapters etc.
        - only entities are proper rich objects, while most of the code is stateless services that operate on requests + entities + value objects
        - stateless services are composed (via interfaces) into a dependency tree (stored in the dependency container)
    
    Currently I'm playing around with an idea for a language that makes writing things like that fast and compact to read. Something like:

        module my_service
    
        layer app {
            service Adder {   // stateless service
                uses base int // a value-based dependency, injected in the container below
    
                method add(x int) int {
                    return base + x
                }
            }
    
            service Doubler {
                uses a Adder  // delegates to another service
    
                method double(x int) int {
                    return a.add(x) + a.add(x)
                }
            }
        }
    
        container {       // dependency container construction with injections
            A = Adder { base: 10 }
            D = Doubler { a: A }
        }
    
        // automatically generates a web server that exposes a JSON API with method "double" and accepts the "n" argument
        endpoint double(n int) int {
            return D.double(n)
        }
    
    This is a synthetic example, but you get the idea (entitites and value objecst omitted here)

    What do you think? Does it make sense? It basically moves something usually implemented by a framework into the language, but that's the entire point: a language optimized for writing compact, architecturally safe stateless services in a few lines of code. For example, since we know a request's memory is bound to that request (no global state), we can have very optimized memory management without a full GC => improved latency. Or for example, we can have compile-time checks for things like dependency direction validation (i.e. the domain layer cannot reference the infrastructure layer) to keep the architecture clean, etc.

    • jjice 21 hours ago
      I like these hobby languages just because they help expose and experiment with interesting higher level language constructs. Because of that, I don't really care if they try to sell me on the language or not.

      As for your concept, I think this is super interesting. A language catered towards higher level abstractions that we use for web services these days is very appealing. The service and container constructs are particularly enticing.

    • onlyrealcuzzo 19 hours ago
      I would recommend building a macro system and/or library for an existing language - that most closely aligns with your goals.

      It seems like your goal is to make things more declarative / readable.

      Creating a language is a pretty large undertaking, and unless you need to do it to achieve your goals, I wouldn't recommend it - unless you really just want to see what it's all about and make one.

  • semilin 20 hours ago
    "${name}: ${features.join(", ")}"

    This is in the very first example you see on the site. If it's a mistake, that's not encouraging. If this is actually how the language works, that's even less encouraging -- the syntax highlighter doesn't even get it right!

    • ModernMech 20 hours ago
      The whole thing is 400k lines vibecoded over 3 weeks, I don't think there's been a lot of effort put into it yet.
  • voidUpdate 1 day ago
    Is it just me that doesnt like automatically returning the last statement in functions? It makes it hard to see where a function returns, and I dont see how you would do a guard clause at the start of a function without having the entire rest of the function in an else block
    • zdragnar 1 day ago
      I suspect the idea would be to use `match` instead of an imperative `if`. There's an example here:

      https://github.com/kablorp/blorp/blob/main/benchmarks/blorp/...

      Then again, there's really not too many examples of early return guards, but I did manage to find one where the body is stuffed in an `else`:

      https://github.com/kablorp/blorp/blob/main/benchmarks/blorp/...

      It does make me think that the usual types of guards might typically happen higher up (handled by the caller) or hidden with safe / monadic type operators that simply pass through rather than bailing out, so to speak.

    • rtpg 1 day ago
      I remember really bumping up against this learning OCaml in college after having experienced oodles of imperative programming.

      I understand the sort of philosophy and ergonomics of not having an early return, but it really does hurt certain kinds of code that otherwise would be more readable

      • orthoxerox 21 hours ago
        > ergonomics of not having an early return

        I wonder who came up with this idea first. I find obvious early returns incredibly ergonomic.

        • voidUpdate 21 hours ago
          Wikipedia says that "guard clause" was a term invented by Kent Beck, but that the actual practice was used since at least the early 60s
          • orthoxerox 21 hours ago
            No, I meant the idea that guard clauses are antipatterns and your subroutine should have a single implicit return.
            • voidUpdate 21 hours ago
              Sorry, I misread. Need more coffee
    • bjoli 1 day ago
      I think it is much more obvious than being able to return from anywhere in a function. If the last expression is a match, I know every match body must return the same type. if the last is a (cond ...) I know ever cond branch must return a value. I vastly prefer that.
    • troupo 23 hours ago
      If it's inconsistently applied, yes.

      In most functional languages however you can view the end of any statement/expression as a return/assign which makes it very easy and trivial to assign anything to variables, or split anything into function calls.

  • anentropic 21 hours ago
    It looks nice. The general syntax isn't explained AFAICT?

    Looks somewhat Python-like but modernised (great!) - is it indentation sensitive?

  • lekevicius 22 hours ago
    Feels like CoffeeScript for C, in the best way
  • ramon156 22 hours ago
    I like it. Reminds me of ruby. maybe a more verbose/explicit go? cool stuff!
    • onlyrealcuzzo 22 hours ago
      You might like this language I've been working on: https://GitHub.com/Cuzzo/clear

      It's not as true to Ruby as Crystal is, because I aim to make it far safer. It's closer to Elixir, if anything.

      But I love Ruby to death, and it is definitely the desire to make it as close to Ruby spiritually as possible.

  • deterministic 8 hours ago
    It would be nice to see a "new" language that actually does something truly new and valuable.

    Almost all "new" languages presented on HN are basically slightly different flavours of languages that have been around for a very long time. But without the libraries/documentation/tools etc. needed to make it useful.

  • fithisux 18 hours ago
    Is it vibe coded?
    • queenmab 17 hours ago
      Looking at the OCaml code, I can guarantee you the whole thing is vibe coded.
  • NuclearPM 23 hours ago
    [dead]
  • flintenmuschi 21 hours ago
    [flagged]