29 Aug 2021
I’m reimplementing git clone in Scala based of this reimplementation in Haskell. My goals in order are:
SWE practices I’ll follow:
I’ll break the project up into phases. Phase 1 is to complete the following:
Half way through, the post became overwhelming. I decided to skim for a high level overview for the rest, and focus on a deep understanding of git transport and pack wire protocal so that I can start implementing that part of it.
This will complete step 1 - 3 of the git clone process detailed in the original blog post
Done. I should probably write more tests for better coverage, but my goal is to get a working program quickly, not one that will be put in production / care about resiliency.
I moved all tests to use the zio-test framework. I’ll still figure out what the test for this should look like.
I’m reading about TCP / Akka actors on stackoverflow and from the official docs to understand how I can send a TCP message in Scala. I think the ZIO environments I’ll need are System and maybe Blocking. I haven’t worked with either of these ZIO environments, so that’ll be fun to learn about!
I’m skipping the ls-remote test for now. I’m not quite sure yet how to write tests that depend on external services (like a local tcp / git server), let alone an network-dependent service. I was able to get a prototyped local version of ref discovery working with akka. AKA I can see the correct git info when I put up a local git server, and run my tcp client code. It’s not type-safe / functional / ZIO’d yet. It looks like there’s some ZIO socket info out there. I’m not sure if I’ll be able to use this, or be able to wrap my akka setup in ZIO. It seems like this is the bulk of ls-remote though.
Before, I only had the actor ‘fire-and-forget’, so I wouldn’t get a response back. Now, lsRemote returns a Future, which is the return type of an ask. I’m still not super clear on the setup here. I’m not even sure how many actors I have, tbh, but something is working. I’ll have to dig deeper to really understand what’s going on here, and I’ll probably be able to prune the process from what it is now.
The HFT bot mentioned above is on hold, so I’m continuing with this project!
The akka actors I was using to handle the TCP messages to and from the git server weren’t quite working how I needed them to. I was following this example. I was sending an ‘ask’ to the TCP client actor from outside the client system. This is supposed to create a Future. The TCP client actor was receiving a response and the ‘info’ messages containted the correct response from the git server, but then that wasn’t being passed to the Future outside of the actor system. When I tried to send a message to the ‘sender()’ (what I thought would be the outside-the-actor-system caller), I was getting a ‘deadLetters’ actor issue. So I moved on for now.
This official TCP client example, needs a ‘replies’ / ‘listener’ of type ActorRef passed to it. My hangup was that I didn’t know what to pass here, and how to use it. There’s not many great docs / examples / blog posts on Akka actors, and if I go the actor route and figure this out, I might make some content.
When you send an ‘ask’ to an actor, you get back a Future. Because I’d like to use ZIO, the plan was to wrap the Future in a ‘ZIO.fromFuture’. Well, I was having trouble with the actor model, and I found this ZIO socket channel. I’m not sure if this will do what I want it to, namely connecting to the server, writing the request in packet line format to the socket, then reading the response from the socket. The source code has methods called ‘write’ and ‘read’, so I’m hoping I’m on the right track. So far, I’ve gotten it to compile, but the ‘write’ method I’m calling on AsynchronousSocketChannel doesn’t match the method call in the source code. Very suspicious!
I also made my first pull request to ZIO! It’s just a minor documentation fix! Here it is.
When testing the akka TCP actor example, it seemed like overkill. I also didn’t spend the time to get it to work fully. I pivoted away from Akka for two reasons:
The Akka actor model seems like the canonical way in Scala to send and recieve TCP messages, but also complicated and overkill. While I want to gain more familiarity with the actor abstraction, how to use it, and understand what problems it solves, I want to move quicker through building this app. For now, I’m just going to use Java sockets from this example to build from.