Я разрабатываю систему фактурирования, используя Java Spring Boot + Thymeleaf. Я получаю эту ошибку:
>Whitelabel Error Page
>This application has no explicit mapping for /error, so you are seeing this as a fallback.
>Wed Mar 04 10:58:26 ART 2020
>There was an unexpected error (type=Internal Server Error, status=500).
>No message available
>java.lang.StackOverflowError
Я использую два контроллера для обработки заполнения счета, который разделен на 3 таблицы: элементы Encabezado (заголовок) и класс P ie (фут) Encabezado:
@EqualsAndHashCode
@Entity
@Table(name="facturas_encabezado")
public class Encabezado implements Serializable {
private static final long serialVersionUID = -3479505724865821556L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@NotNull
private Timestamp fecha;
@NotNull
@Size(min = 1,max = 11)
private String numero;
@Nullable
private String letra;
@NotNull
@OneToOne
private Cliente cliente;
@OneToMany(mappedBy = "encabezado")
private List<Item> items;
@NotNull
private Boolean anulado=false;
Элемент класса
@Entity
@Table(name="facturas_items")
public class Item implements Serializable {
private static final long serialVersionUID = -3560761383569266746L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@OneToOne
@JoinColumn(name = "productos_id")
private Producto producto;
@ManyToOne
@JoinColumn(name = "facturas_encabezado_id")
private Encabezado encabezado;
private BigDecimal cantidad;
private BigDecimal precioUnitario;
private BigDecimal subTotal;
HeaderController
@Controller
@RequestMapping("/factura")
public class EncabezadoControl {
@Autowired
private ClienteRepositorio clienterepository;
@Autowired
private ProductoRepositorio productorepository;
@Autowired
private EncabezadoRepositorio encabezadorepository;
@GetMapping("/creacion_factura")
public String creacionFactura(Encabezado encabezado,Errors errors,Model model ) {
List<Cliente> clientes;
clientes=clienterepository.findByVisibilidad();
model.addAttribute("titulo", "Listado de Clientes");
model.addAttribute("data", clientes);
model.addAttribute("encabezado",new Encabezado());
return "facturas/creacion_factura";
}
@PostMapping(value = "/creacion_factura")
public String guardarFactura(Encabezado encabezado,Errors errors,Model model) {
Timestamp timestamp = new Timestamp(System.currentTimeMillis()); //Instanciamos fecha actual para el encabezado de factura.
encabezado.setFecha(timestamp);
model.addAttribute("facturaInfo",encabezado);
System.out.println(encabezado);
if(encabezado.getId()==null){
encabezadorepository.save(encabezado);
}
Long id=encabezado.getId();
return "redirect:/factura/establecer_productos/"+id;
}
}
ItemController
@Controller
@RequestMapping("/factura")
public class ItemControl {
@Autowired
private ItemRepositorio itemrepository;
@Autowired
private ProductoRepositorio productorepository;
@Autowired
private EncabezadoRepositorio encabezadorepository;
@GetMapping(value = "/establecer_productos/{id}")
public String indicarProductos(Model model, @PathVariable Long id){
List<Producto> listaproductos=null;
listaproductos=productorepository.findByEstado();
model.addAttribute("titulo", "Selección de Productos");
model.addAttribute("productos",listaproductos);
model.addAttribute("item", new Item());
return "facturas/seleccion_productos";
}
@PostMapping(value = "/establecer_productos/{ided}")
public String procesarProductos(@PathVariable Long ided, Item item,Model model){
if(item.getId()==null) {
BigDecimal cantidad = item.getCantidad();
BigDecimal precio = item.getPrecioUnitario();
item.setSubTotal(cantidad.multiply(precio));
item.setEncabezado(encabezadorepository.getOne(ided));
itemrepository.save(item);
}
System.out.println("ITEM\n\n\n"+item+"\n\n\n");
return "redirect:/factura/establecer_productos/"+ided;
}
}
Представление первого шага: создание заголовка
<div class="container">
<form method="post" th:action="@{/factura/creacion_factura}" th:object="${encabezado}">
<br/>
<h2>Introduzca los datos de la Factura</h2>
<hr/>
<div class="row">
<div class="col-md-3">
<h3>Cliente</h3>
</div>
<div class="col-md-6">
<select name="cliente" class="form-control form-control-lg" th:field="*{cliente}">
<option th:each="cliente : ${data}" th:value="${cliente.id}" th:text="${cliente.nombre}"></option>
</select>
</div>
<div class="col-md-3"></div>
</div>
<div class="row" th:class="${#fields.hasErrors('numero')} ? 'form-group row has-danger' : (*{numero == null} ? 'form-group row' : 'form-group row has-success')">
<div class="col-md-3">
<h3>Numero Factura</h3>
</div>
<div class="col-md-6">
<div class="input-group">
<input th:field="*{numero}" type="text" th:class="${#fields.hasErrors('numero')} ? 'form-control form-control-danger' : (*{numero == null} ? 'form-control' : 'form-control form-control-success')" placeholder="Ingrese el numero de factura" required="required"/>
</div>
<div th:if="${#fields.hasErrors('numero')}" class="form-control-feedback">El número debe tener entre 1 y 11 caracteres</div>
</div>
<div class="col-md-3"></div>
</div>
<div class="row" th:class="${#fields.hasErrors('letra')} ? 'form-group row has-danger' : (*{letra == null} ? 'form-group row' : 'form-group row has-success')">
<div class="col-md-3">
<h3>Categoria Factura</h3>
</div>
<div class="col-md-6">
<select class="form-control form-control-lg" th:field="*{letra}" >
<option th:value="'A'"th:text="A"></option>
<option th:value="'B'"th:text="B"></option>
<option th:value="'C'"th:text="C"></option>
</select>
</div>
<div class="col-md-3"></div>
</div>
<div class="form-group row">
<div class="col-sm-2"></div>
<div class="col-sm-6">
<button type="submit" class="btn btn-primary btn-md btn-block">Siguiente Paso</button>
</div>
</div>
</form>
</div>
Второй шаг: Добавить элементы зрения
<div class="container">
<form method="post" th:action="@{/factura/establecer_productos/{id}(id=${id})}" th:object="${item}">
<br/>
<h2>Seleccione los Productos de la Factura</h2>
<hr/>
<div class="row">
<div class="col-md-3">
<h3>Productos</h3>
</div>
<div class="col-md-6">
<select class="form-control form-control-lg" name="producto" th:field="*{producto}">
<option th:each="producto : ${productos}" th:value="${producto.id}" th:text="${producto.nombre}"></option>
z</select>
</div>
<div class="col-md-3"></div>
</div>
<div class="row">
<div class="col-md-3">
<h4>Cantidad</h4>
</div>
<div class="col-md-6">
<div class="input-group">
<input class="form-control text-center" th:field="*{cantidad}" type="text" th:class="${#fields.hasErrors('cantidad')} ? 'form-control form-control-danger' : (*{cantidad == null} ? 'form-control' : 'form-control form-control-success')" placeholder="introduzca cantidad a comprar" required="required"/>
</div>
<div th:if="${#fields.hasErrors('cantidad')}" class="form-control-feedback">La cantidad es obligatoria</div>
</div>
<div class="col-md-3"></div>
</div>
<div class="row">
<div class="col-md-3">
<h4>Precio Unitario</h4>
</div>
<div class="col-md-6">
<div class="input-group">
<input class="form-control text-center" th:field="*{precioUnitario}" type="text" th:class="${#fields.hasErrors('precioUnitario')} ? 'form-control form-control-danger' : (*{precioUnitario == null} ? 'form-control' : 'form-control form-control-success')" placeholder="introduzca precio unitario" required="required"/>
</div>
<div th:if="${#fields.hasErrors('precioUnitario')}" class="form-control-feedback">El precio es obligatoria</div>
</div>
<div class="col-md-3"></div>
</div>
<div class="form-group row">
<div class="col-sm-2"></div>
<div class="col-sm-6">
<button type="submit" class="btn btn-primary btn-md btn block">Siguiente</button>
</div>
</div>
</form>
</div>
Компиляция выполнения идет хорошо. Заголовок счета правильно сохраняется в БД, но проблема возникает, когда мы переходим ко второму шагу: элемент сохраняется после полного выполнения. Любые решения?