Spring MVC with Hibernate - OneToMany mapping in form

bontade picture bontade · Mar 29, 2012 · Viewed 8.6k times · Source

I have Car and Rental models conntected OneToMany relation:

@Entity
public class Rental {

private Long id;
private Car car;
private User user;

@Id
@GeneratedValue
@Column(name = "RENTAL_ID")
public Long getId() {
    return id;
}

public void setId(Long id) {
    this.id = id;
}

@ManyToOne(cascade = CascadeType.ALL)
@JoinColumn(name = "CAR_ID")
public Car getCar() {
    return car;
}

public void setCar(Car car) {
    this.car = car;
}

@ManyToOne(cascade = CascadeType.ALL)
@JoinColumn(name = "USER_ID")
public User getUser() {
    return user;
}

public void setUser(User user) {
    this.user = user;
}
}





@Entity
public class Car {

private Long id;
private String name;
private String description;

@Id
@GeneratedValue
@Column(name = "CAR_ID")
public Long getId() {
    return id;
}

public void setId(Long id) {
    this.id = id;
}
...
}

And I want to set car during rental creation:

<form:form method="POST" commandName="rental">
<form:errors path="*" cssClass="errorblock" element="div" />
<table cellspacing="10">
    <tr>
        <td>Car</td>
        <td>
        <form:select path="car" multiple="false" items="${cars}" itemLabel="name" itemValue="id"/></td>
    </tr>
    <tr>
        <td colspan="2"><input type="submit" value="Create"></td>
    </tr>
</table>

Form:select tag displays correctly available cars, but after submit Car instance variable of Rental is null.

@Controller
@RequestMapping("/make-rental.htm")
public class MakeRentalController {

private CarDAO carDAO;
private UserDAO userDAO;
private RentalDAO rentalDAO;

@Autowired
public void setCarDAO(CarDAO carDAO) {
    this.carDAO = carDAO;
}

@Autowired
public void setUserDAO(UserDAO userDAO) {
    this.userDAO = userDAO;
}

@Autowired
public void setRentalDAO(RentalDAO rentalDAO) {
    this.rentalDAO = rentalDAO;
}

@RequestMapping(method = RequestMethod.GET)
public String initForm(ModelMap mapModel) {
    Rental rental = new Rental();

    // command object
    mapModel.addAttribute("cars", carDAO.getAll());
    mapModel.addAttribute("rental", rental);

    // return form view
    return "makeRentalFormView";
}

@RequestMapping(method = RequestMethod.POST)
public String processSubmit(@ModelAttribute("rental") Rental rental,
        BindingResult result, SessionStatus status, HttpSession session) {
    Object userId = session.getAttribute("userId");
    if (userId != null) {
        rental.setUser(userDAO.getById((Long) userId));
        rentalDAO.save(rental);

        // clear the command object from the session
        status.setComplete();
    }

    // form success
    return "redirect:/index.htm";
}
}

What's wrong?

Answer

Pau Kiat Wee picture Pau Kiat Wee · Mar 30, 2012

You should implement a Service Layer into you application

For example:

public interface RentalService {
    void addRental(Rental rental, Long userId, Long carId);
}

Implementation:

@Service
@Transactional
class DefaultRentalService implement RentalService {

    @Autowired
    private CarDAO carDAO;
    @Autowired
    private UserDAO userDAO;
    @Autowired
    private RentalDAO rentalDAO;

    @Override
    void addRental(Rental rental, Long userId, Long carId) {
        User user = userDAO.getById(userId);
        Car car = carDAO.getById(carId);
        rental.setUser(user);
        rental.setCar(car);
        rentalDao.save(rental);
    }
}

HTML:

<form:form method="POST" commandName="rental">
<form:errors path="*" cssClass="errorblock" element="div" />
<table cellspacing="10">
    <tr>
        <td>Car</td>
        <td>
        <select name="carId" >
            <c:forEach items="${cars}" var="car">
               <option value="${car.id}">${car.name}
            </c:forEach>
        </select>
     </td>
    </tr>
    <tr>
        <td colspan="2"><input type="submit" value="Create"></td>
    </tr>
</table>

Controller:

@AutoWired
private RentalService rentalService;

// ..
// ..

@RequestMapping(method = RequestMethod.POST)
public String processSubmit(@ModelAttribute("rental") Rental rental, @RequestParam Long carId,
        BindingResult result, SessionStatus status, HttpSession session) {
    Object userId = session.getAttribute("userId");
    if (userId != null) {
        rentalService.add(rental, (Long)userId, carId);
        status.setComplete();
    }

    // form success
    return "redirect:/index.htm";
}