The project is a multi-module Maven project.
At the bottom of the module tree there are the back end and front end modules, which contain the application source code.
The standalone module is an assembly module that combines the back end and front end into a single executable JAR file.
The distribution module represents the final assembly step. It takes the standalone application and the documentation and wraps them in an archive that is easy to distribute.
The back end and front end are separate projects that can be built and deployed separately. In fact, they are written in completely different languages and built with different tools. Both projects have tools that provide a modern developer experience with fast turn-around between code changes and the running application.
In the next sections you will learn how to run both back end and front end projects in development mode.
The back end module contains a server-side application that uses OptaPlanner to optimize vehicle routes. Optimization is a CPU-intensive computation that must avoid any I/O operations in order to perform to its full potential. Because one of the chief objectives is to minimize the travel cost, either time or distance, we need to keep the travel cost information in RAM memory. While solving, OptaPlanner needs to know the travel cost between every pair of locations entered by the user. This information is stored in a structure called the distance matrix.
When a new location is entered, we calculate the travel cost between the new location and every other location that has been entered so far, and store the travel cost in the distance matrix. The travel cost calculation is performed by a routing engine called GraphHopper.
Finally, the back end module implements additional supporting functionality, such as:
-
persistence,
-
WebSocket connection for the front end,
-
data set loading, export, and import.
In the next sections you will learn how to configure and run the back end in development mode. To learn more about the back end code architecture, see appendix-backend-architecture.adoc.
-
Java 11 or higher is installed.
-
The data directory is set up.
-
An OSM file is downloaded.
You can manually set up the data directory and download the OSM file or you can use the run script to complete these tasks.
To run the back end in development mode, enter the following command:
mvn spring-boot:run
Automatic restart is provided by Spring Boot DevTools and only works when the back end is running using Spring Boot Maven Plugin. It scans files on the classpath, so you only need to recompile your changes to trigger application restart. No IDE configuration is needed.
If your IDE has a compile-on-save feature (for example Eclipse or NetBeans), you just need to save the files that have changed since the last compilation.
IntelliJ IDEA saves changes automatically and you need to select either menu:Build[Recompile], which recompiles the file in the active tab, or menu:Build[Build Project] which recompiles all changes. See Compile and build applications with IntelliJ IDEA.
-
Run
org.optaweb.vehiclerouting.OptaWebVehicleRoutingApplication
. This will create a run configuration that you will edit in the next step.-
Open the
OptaWebVehicleRoutingApplication
class in the editor. -
Click the green symbol in the editor window gutter and select Run 'OptaWebVehicleRoutingApplication'.
NoteThe run fails because the working directory is set to the root of the project, whereas the back end module directory is expected. You are going to change the working directory in the next step. NoteSee Run applications to learn more about running applications in IntelliJ IDEA.
-
-
Select menu:Run[Edit Configurations…] and then select menu:Spring Boot[OptaWebVehicleRoutingApplication].
-
Set Program arguments to
--spring.profiles.active=local
to activate the Spring profile calledlocal
. This will make the application use configuration fromapplication-local.properties
. -
Change Working directory to the back end module (
optaweb-vehicle-routing-backend
). -
Optionally, set On Update action to Hot swap classes and update trigger file if failed. This will allow you to use the Update action to quickly restart the application.
See Spring and Spring Boot in IntelliJ IDEA 2018.1 for more details.
There are many ways that you can set configuration properties. If you are running locally, you will probably want to use one of these:
-
Set configuration properties in the
application.properties
file, under/src/main/resources/
. -
Use a command line argument when running the packaged application (for example
java -jar optaweb-vehicle-routing-backend.jar --app.my-property=value1
). -
Use an environment variable when running the application with
spring-boot:run
.For example:
app_my_property=value1 ./mvnw spring-boot:run
if the property name isapp.my-property
(this requires relaxed binding which only works if the property is defined using@ConfigurationProperties
).
Note
|
It is not possible to set properties by specifying -D when running the application using the Spring Boot Maven plugin (./mvnw spring-boot:run -Dmy-property ).
Any system properties that should be set by the plugin to the forked Java process in which the application runs need to be specified using systemPropertiesVariables
plugin configuration.
|
You can learn more about configuring a Spring Boot application on the Spring Boot Externalized Configuration page.
Tip
|
Use src/main/resources/application-local.properties to store your personal configuration without affecting the Git working tree.
|
See the complete list of appendix-backend-config.adoc.
See also the complete list of common application properties available in Spring Boot.
OptaWeb uses the SLF4J API and Logback as the logging framework.
The Spring environment enables you to configure most logging aspects including levels, patterns, and log files in the same way as any other Configuration (most often using application.properties
or arguments --property=value
).
See the
Spring Boot Logging
documentation for more information.
Following are examples of properties you can use to control logging level of some parts of the application:
-
Use
logging.level.org.optaweb.vehiclerouting=debug
to enable debug level for the back end code. -
Use
logging.level.org.optaplanner.core=warn
to reduce OptaPlanner logging. -
Use
logging.level.org.springframework.web.socket=trace
to access more details when investigating problems with WebSocket connection.
The front end project was bootstrapped with Create React App. Create React App provides a number of scripts and dependencies that help with development and with building the application for production.
-
On Fedora, run the following command to install npm:
sudo dnf install npm
See Downloading and installing Node.js and npm for more information about installing npm.
Unlike Maven, the npm package manager installs dependencies in node_modules
under the project directory and does that only when requested by running npm install
.
Whenever the dependencies listed in package.json
change (for example when you pull changes to the master branch) you must run npm install
before you run the development server.
-
Change directory to the front end module:
cd optaweb-vehicle-routing-frontend
-
Install dependencies:
npm install
-
npm is installed.
-
npm dependencies are installed.
-
Run the development server:
npm start
-
Open http://localhost:3000/ in a web browser. By default, the
npm start
command attempts to open this URL in your default browser.
Tip
|
Prevent
npm start from launching your default browserIf you don’t want You can use echo BROWSER=none >> .env.local |
The browser refreshes the page whenever you make changes in the front end source code. The development server process running in the terminal picks up the changes as well and prints compilation and lint errors to the console.
Use an environment variable called REACT_APP_BACKEND_URL
to change the backend URL when running npm start
or npm run build
.
For example:
REACT_APP_BACKEND_URL=http://10.0.0.123:8081
Note that environment variables will be "baked" inside the JavaScript bundle during the npm build, so you need to know the back end location before you build and deploy the front end.
Learn more about the React environment variables in Adding Custom Environment Variables.