• 7 Posts
  • 123 Comments
Joined 1 year ago
cake
Cake day: June 19th, 2023

help-circle
  • In Maybe monadic, its monadic bind will automatically resolves any failed computation, and don’t need explicit checking.

    for example, the code in Haskell looks something like the following:

    fib: Int -> Int -> Maybe Int
    fib max_depth idx =
      do
         guard (0 <= max_depth)
         n1 <- fib (max_depth - 1) (idx - 1)
         n2 <- fib (max_depth - 1) (idx - 2)
         return (n1 + n2)
    

    Haskell type class system automatically figures out this is a maybe monad, and check for error accordingly.

    Notice, unlike the C code the author provide, this haskell code will exit immediately when n1 failed and never compute n2, similar to the behavior of the exception code. Thus I believe his point about performance is at least unjustified, if not wrong.

    Another interesting fact about this code is that there is nothing that is built into the compiler/interpretor (except the do expression, which is just a minor syntactical sugar). For this code, the compiler designers don’t need to design special semantics for raise and catch. Everything here, guard, return, and the Maybe monad (which is in charge of propagating errors) is defined by the user, using normal functions, no metaprogramming involved.

    Wouldn’t effect systems still be considered exceptions, but handled differently?

    Yes, unlike monad, the error in algebraic effect is propagated by the compiler/interpretor, instead of user defined. But unlike implicit effect, explicit effect (algebraic effect, throwable, etc.) makes it clear how the code can go wrong.

    Although explicit error through monad or algebraic effect is more clear in general, there are special cases where explicit effect is undesirable. One such example is effect pollution: low-level effects that are unlikely to cause impure behaviors are unnecessarily propagated through the call stack. This problem can make the code more verbose and difficult to handle.


  • The more I read about these kind of article the more I am amazed that our digital future is at hand in utterly incompetent people.

    This person clearly have no understanding of monadic error (AKA Maybe/option monad or slightly more advanced Either monad), which is the first monad we teach at a class targeting second year undergrad.

    The performance comparison is just plain factual error. The functional error code will continue to compute n2 when computation of n1 failed; the same do not happen in the exception version. If you compare codes with completely different traces, of course they will have different performance…

    A properly implemented monadic error will return as soon as compute for n1 failed, and never execute the rest of the code. This is the default and idiomatic behavior in Haskell, OCaml, F#, and rust. This performance problem doesn’t even happen in LINQ-style handling like in C# and Kotlin (maybe also Typescript?).

    The point of monadic error is that its control flow is local, whereas exception is non-local. Specifically, the exception can be handled and occur anywhere in the code base, with no indication on the type level. So programmers will be constantly worrying about whether the exception in a function call is properly handled.

    Even worse, when you try to catch a certain error, there is always the risk to accidentally catch similar exceptions in a library call or someone else’s code. Writing good code with try-catch requires a lot of principle and style guides. But unlike monads, these principle and rules cannot be enforced by the type system, adding extra burden to programmers.

    In fact, we have known for a long time that non-local control flows (goto, break, contiune, exception, long jump) are the breeding ground for spaghetti code. As an evidence, many non-local control flows (goto, long jump) are baned in most languages.

    That being said, there are certainly cases, with proper documentation, that exception style is easy to write and understand. But I think they are very specific scenarios, which have to be justified on a case-by-case basis.



  • Honestly, it is much more code to use loop with non-local control like break, continue etc. (variable initialization, append, variable mutation in loops…) than just calling a collect function (which I assume just means to_list). In the above example, in most programming language I know, you don’t even need to collect the result into a list.

    Not to mention, large loops with non-local control is a breeding ground for spegatti code. Because you no longer have a consistent exit point to the loop, thus making the semantics hard o reason about.

    In many languages, there are type class / trait / interfaces (whatever you want to call them) that allows lazy structures to share the same API as strict ones.









  • Two of the same MAC address can’t exist in the same IP space, else the router can’t route packets to them.

    Yes, this seems to be my problem, both the host and the vm got the same IP, and I think I cannot send any traffic to either my host or vm. So my router is probably confused, as you suggested.

    Is there an issue with using Docker for this?

    I forgot to mention this, docker indeed work. However, ha requires a privileged docker running as root, which means ha essentially runs as root on the host.

    This is fine on dedicated hardware, but as my server have other infrastructure on there, running ha as root can be a security risk.


  • Hi, thank you for your reply.

    Did you make sure that eno2 is enslaved by br0? When br0 is created, it indeed have a unique MAC, but once it enslaves the hardware, it inherits the hardware address.

    I have not tried to get the bridge going with virsh, but I was unsuccessful with the virt-manager ui. And I assume they use the same system?

    It is possible I have accidentally disabled some network virtualization kernel component during setup, as I have implemented some mods from secureblue. I will try to reset everything to default, and try again.

    Do you have a tutorial for making bridge via virsh that you can share.







  • Although it is true that no organization is perfect, there are certainly better companies v.s. worse companies. For example, I believe Linux foundation is more ethical than Nestle, Mozilla is more ethical than Facebook, world bank is more ethical than American military.

    If all company is equally evil, then you probably won’t boycott nestle, because all of them are the same.

    Hence, to me, there are certainly more ethical job than other jobs, depends what you are contributing to. I would argue a gardener in charge of planting and preserving local plants, is more ethical than CEO of nestle. Marketing at EFF is probably a more ethical job than marketing for nestle.

    Of course, if you would starve when you don’t work for nestle, then the society has failed you, it is not your fault. This is why I said that I wouldn’t judge a person solely because they work at nestle, it really depends on their alternative.

    I understand different person realize their ethical standards to different extent, I personally won’t accept a job from nestle, no matter how much they are paying me, if I have the opportunity to earn a living wage as a barista, waiter, janitor, cook, or any other job.

    Finally, I believe “working is not unethical” seems misguided, by this logic CEO of nestle is ethical, he claim water shouldn’t be a human right, which indeed helps their company sell more product and make more money (like everyone else at nestle). By your reasoning, his claim should be completely ethical, because making money for nestle is in his job description.

    Like you said, real life is complicated, hence I believe blanketed statements like “working is not unethical” probably won’t hold true.