Introduction to Controllers in MVC Architecture

Welcome to the fourth lesson of our course on building an image generation service in Java! So far, we've built several key components of our application: the PromptManager for formatting user inputs, the ImageManager for storing and processing images, and the ImageGeneratorService that connects to Google's Gemini API to generate images.

Now, we're ready to add another important layer to our application architecture: the controller. In Java web applications, controllers play a crucial role in the Model-View-Controller (MVC) pattern. They act as intermediaries between the service layer (which contains our business logic) and the presentation layer (which handles user interactions, such as HTTP requests and responses).

The controller's primary responsibilities include:

  • Receiving and validating input from the user interface (such as HTTP requests)
  • Calling the appropriate service methods with validated inputs
  • Handling errors that might occur during processing
  • Formatting responses in a consistent way before sending them back to the user

In our image generation application, the controller will receive text prompts and an aspectRatio from users, validate them, pass them to our ImageGeneratorService, and then format the responses (either successful image data or error messages) before sending them back.

By adding this controller layer, we're further improving the separation of concerns in our application. The service layer can focus purely on business logic (generating images), while the controller handles the specifics of HTTP requests and responses. This makes our code more maintainable, testable, and easier to extend in the future.

Setting Up the ImageGeneratorController

Let's start by creating our controller class. We'll create a new file called ImageGeneratorController.java in the controllers package. This controller will depend on our ImageGeneratorService to perform the actual image generation.

Here's how we'll set up the basic structure of our controller:

In this code, we're importing our ImageGeneratorService class that we created in the previous lesson.

The controller's constructor initializes an instance of the ImageGeneratorService. This establishes the dependency between our controller and service layers.

Our controller will be responsible for two main operations:

  1. Generating a new image based on user input and aspect ratio
  2. Retrieving all previously generated images

For each of these operations, we'll create a dedicated method in our controller class. These methods will handle input validation, error handling, and response formatting, ensuring that our provides a consistent interface to clients.

Implementing the Image Generation Endpoint

Now, let's implement the method for generating images. This method will receive user input and aspect ratio, validate them, call the service to generate an image, and format the response.

We'll define a simple Response class to encapsulate the response data and status code:

Now, let's implement the generateImage method in the controller:

Let's break down what's happening in this method:

  • First, we check if the userInput is empty or null. If it is, we return an error response with a status code (), indicating that the client provided invalid input.
Implementing the Image Retrieval Endpoint

Next, let's implement the method for retrieving all previously generated images:

This method is simpler than the previous one because it doesn't require input validation. We simply call the getAllImages method of our service, which returns a List of all stored images.

The response is a Map with an "images" key containing the list of image objects. Each image object includes an ID, the prompt used to generate it, and the base64-encoded image data.

With these two methods, our controller provides a complete interface for clients to generate images and retrieve previously generated ones. The controller abstracts away the details of how images are generated and stored, presenting a clean, consistent API to clients.

Full Controller Implementation
Manual Validation of the Controller

Now that we've implemented our controller, let's validate it to make sure it works correctly. We'll create a simple test class called Main.java to use the new controller:

In this test script, we're creating an instance of our ImageGeneratorController and using it to perform two operations:

  • First, we call the generateImage method with a sample prompt and aspect ratio. We check if the response contains an "error" key, which would indicate that something went wrong. If there's no error, we print the base64-encoded image data.
  • Then, we call the method to retrieve all stored images. We print the response , which includes all previously generated images.
Verifying Error Handling Behavior

Let's also test the error handling by trying to generate an image with an empty prompt:

This would produce output like:

This confirms that our controller correctly validates input and returns appropriate error responses with status codes.

Summary and Practice Preview

In this lesson, we've built the ImageGeneratorController, which serves as an intermediary between our service layer and future web endpoints. This controller handles input validation, aspect ratio defaults, error management, and response formatting, providing a clean, consistent interface for clients to interact with our image generation service.

Let's review what we've learned:

  1. We understood the role of controllers in the MVC architecture and how they fit into our application.
  2. We created a controller class that depends on our ImageGeneratorService.
  3. We implemented methods for generating images and retrieving previously generated ones.
  4. We added input validation and error handling to ensure robust operation.
  5. We passed the selected aspectRatio through to the service layer.
  6. We tested our controller to verify that it works correctly.

The ImageGeneratorController is a crucial component of our application architecture. It abstracts away the details of how images are generated and stored, presenting a clean, consistent to clients. This makes our code more maintainable, testable, and easier to extend.

Sign up
Join the 1M+ learners on CodeSignal
Be a part of our community of 1M+ users who develop and demonstrate their skills on CodeSignal