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 asHTTPrequests) - Calling the appropriate
service methodswith 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.
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:
- Generating a new image based on user input and aspect ratio
- 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.
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
userInputis empty ornull. If it is, we return an error response with a status code (), indicating that the client provided invalid input.
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.
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
generateImagemethod with a samplepromptand aspect ratio. We check if theresponsecontains an"error"key, which would indicate that something went wrong. If there's no error, we print thebase64-encodedimage data. - Then, we call the method to retrieve all stored images. We print the response , which includes all previously generated images.
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.
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:
- We understood the role of
controllersin theMVCarchitecture and how they fit into our application. - We created a
controllerclass that depends on ourImageGeneratorService. - We implemented methods for
generating imagesandretrieving previously generated ones. - We added
input validationanderror handlingto ensure robust operation. - We passed the selected
aspectRatiothrough to the service layer. - We tested our
controllerto 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.
