lloydrichards.dev

EFFECT

FUNCTIONAL-PROGRAMMING

CLEAN-ARCHITECTURE

April 02, 2025

Effect: Services (Part 4)

Applying what I learns at Effect Days 2025

While I've only been using Effect for around 2 months now, I'm already deeply impressed by the library/framework. So much so that a couple weeks ago I attended the Effect Days 2025 conference in Livorno, Italy. Was a wonderful event filled with inspiring talks and very impressive developers, many of whom's libraries I've been using my entire development career.

One particular part of the conference that I found very useful was the first day workshop with Maxwell Brown and Tim Smart. The workshop (repo) covered two important topics: using Effect.Service to structure Effect applications, and how to wrap existing API's in Effect. On the heels of this event I was inspired to do some more refactoring of my website to apply what I had learnt.

Effect.Service

I'm not going to go into too much detail about Effect.Service here, as I think Maxwell did an amazing job of explaining it in the workshop and would recommend anyone interested to check out the Effect docs on services or wait for the workshop to get on YouTube. However I will give a quick example so you know what I'm referring too:

// Define the Errors of the Service
class CacheMissError extends Data.TaggedError("CacheMissError")<{
  readonly message: string;
}> {}
 
// Define the Shape of the Service
interface CacheShape {
  readonly set: (data: any) => Effect.Effect<string, never>;
  readonly get: (key: string) => Effect.Effect<string, CacheMissError>;
}
 
// Define the Service
const Cache = Context.GenericTag<Cache, CacheShape>("app/Cache");
 
// Define the Service Implementation
const makeMockCache = Cache.of({
  set: (_data) => Effect.succeed("MockKey"),
  get: (_key) => Effect.fail(new CacheMissError({ message: "Mock missed" })),
});

This is a very simple example of a service that has two methods: set and get with a mocked implementation that could be used for testing. This patterns lets you define the interface for the repository in the domain of your application without needing to worry about implementation details that could come from your infrastructure layer.

But there is a bit of boilerplate to get this working, and I think the new Effect.Service makes it a bit cleaner if you start to need a dependency graph. Say you're making a Service that you know will require accessing another service, having to manually type the interface can be a bit verbose.

@/services/LabApi.ts
class Cache extends Effect.Service<Cache>()("app/Cache", {
  dependencies: [BunFileSystem.layer],
  effect: Effect.gen(function* () {
    const fs = yield* FileSystem.FileSystem;
    const random = yield* Random.Random;
    return {
      get: (key: string) => fs.readFile(key),
      set: (value: Uint8Array) =>
        random.next.pipe(
          Effect.map((key) => key.toString()),
          Effect.tap((key) => fs.writeFile(key, value)),
        ),
    };
  }),
}) {}

This is a lot cleaner and allows you to define the dependencies of the service as a default and then define other implementations as methods on the class.

Dependency Graphs

What really fascinated me about this pattern is how to define a graph of dependencies and then figure out the interfaces between them. This might be something like what database layer that shows up in multiple services, or an API client that is used in multiple services.

This makes the code much more modular and allows for parts to be swapped in and out of whole applications. Rather then building an application dependent on the infrastructure, its possible to plan out how everything will work together without writing a single line of implementation details.

For my own website this sort of looks like this:

Mocking Services

One of the hardest parts of testing is mocking modules and dependencies. But with this service structure it becomes incredible easy to mock out the dependencies and test very isolated parts of the application. For example say I want test the LabApi which depends on the FileSystem:

//      ┌─── Layer.Layer<LabApi, PlatformError | ImportError | ParseError, never>
const LabApiLayer = LabApi.Default;
 
//      ┌─── Layer.Layer<FileSystem.FileSystem, never, never>
const StubFileSystemLayer = Layer.succeed(
  FileSystem.FileSystem,
  FileSystem.makeNoop({
    readDirectory: () => Effect.succeed(["mockFile.ts"]),
  }),
);
 
//      ┌─── Layer.Layer<LabApi, PlatformError | ImportError | ParseError, never>
const MockLabApi = LabApi.DefaultWithoutDependencies.pipe(
  Layer.provide(MockFileSystemLayer),
);

I can now use the MockLabApi layer to test how the methods function with different .readDirectory implementations. This is a very powerful pattern that allows me to to write unit tests that are very isolated.

Conclusion

I think the new Effect.Service is a very powerful pattern that allows for incredible modularity and testability. Designing the graph of services and dependencies without needing to worry about the implementation details is very important for domain driven design. In my quest for cleaner architecture and more maintainable code, Effect has found a place in my toolkit.