diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000000..5583ee6e7d --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,4 @@ +# Default ignored files +/shelf/ +/workspace.xml +.idea \ No newline at end of file diff --git a/.idea/compiler.xml b/.idea/compiler.xml new file mode 100644 index 0000000000..784b413f8f --- /dev/null +++ b/.idea/compiler.xml @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/encodings.xml b/.idea/encodings.xml new file mode 100644 index 0000000000..7581ecbe6d --- /dev/null +++ b/.idea/encodings.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/jarRepositories.xml b/.idea/jarRepositories.xml new file mode 100644 index 0000000000..712ab9d985 --- /dev/null +++ b/.idea/jarRepositories.xml @@ -0,0 +1,20 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000000..46acaf5e70 --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,14 @@ + + + + + + + + + + \ No newline at end of file diff --git a/.idea/modules.xml b/.idea/modules.xml new file mode 100644 index 0000000000..5d057fbd79 --- /dev/null +++ b/.idea/modules.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/.idea/ouicha-hammou-yassine-poseidon-trading-app.iml b/.idea/ouicha-hammou-yassine-poseidon-trading-app.iml new file mode 100644 index 0000000000..d6ebd48059 --- /dev/null +++ b/.idea/ouicha-hammou-yassine-poseidon-trading-app.iml @@ -0,0 +1,9 @@ + + + + + + + + + \ No newline at end of file diff --git a/.idea/vcs.xml b/.idea/vcs.xml new file mode 100644 index 0000000000..35eb1ddfbb --- /dev/null +++ b/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Poseiden-skeleton/.idea/.gitignore b/Poseiden-skeleton/.idea/.gitignore new file mode 100644 index 0000000000..95b027519c --- /dev/null +++ b/Poseiden-skeleton/.idea/.gitignore @@ -0,0 +1,4 @@ +# Default ignored files +/shelf/ +/workspace.xml +.idea diff --git a/Poseiden-skeleton/.idea/compiler.xml b/Poseiden-skeleton/.idea/compiler.xml new file mode 100644 index 0000000000..784b413f8f --- /dev/null +++ b/Poseiden-skeleton/.idea/compiler.xml @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/Poseiden-skeleton/.idea/encodings.xml b/Poseiden-skeleton/.idea/encodings.xml new file mode 100644 index 0000000000..63e9001932 --- /dev/null +++ b/Poseiden-skeleton/.idea/encodings.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Poseiden-skeleton/.idea/jarRepositories.xml b/Poseiden-skeleton/.idea/jarRepositories.xml new file mode 100644 index 0000000000..712ab9d985 --- /dev/null +++ b/Poseiden-skeleton/.idea/jarRepositories.xml @@ -0,0 +1,20 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/Poseiden-skeleton/.idea/misc.xml b/Poseiden-skeleton/.idea/misc.xml new file mode 100644 index 0000000000..67e1e6113b --- /dev/null +++ b/Poseiden-skeleton/.idea/misc.xml @@ -0,0 +1,12 @@ + + + + + + + + \ No newline at end of file diff --git a/Poseiden-skeleton/.idea/vcs.xml b/Poseiden-skeleton/.idea/vcs.xml new file mode 100644 index 0000000000..6c0b863585 --- /dev/null +++ b/Poseiden-skeleton/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Poseiden-skeleton/doc/data.sql b/Poseiden-skeleton/doc/data.sql index 569bdeb432..955ff637a0 100644 --- a/Poseiden-skeleton/doc/data.sql +++ b/Poseiden-skeleton/doc/data.sql @@ -1,99 +1,138 @@ +SHOW DATABASES; +CREATE DATABASE IF NOT EXISTS demo; +USE demo; -CREATE TABLE BidList ( - BidListId tinyint(4) NOT NULL AUTO_INCREMENT, +-- Table bidlist +DROP TABLE IF EXISTS bidlist; +CREATE TABLE bidlist ( + bid_list_id int NOT NULL AUTO_INCREMENT, account VARCHAR(30) NOT NULL, type VARCHAR(30) NOT NULL, - bidQuantity DOUBLE, - askQuantity DOUBLE, - bid DOUBLE , + bid_quantity DOUBLE, + ask_quantity DOUBLE, + bid DOUBLE, ask DOUBLE, benchmark VARCHAR(125), - bidListDate TIMESTAMP, + bid_list_date TIMESTAMP NULL, commentary VARCHAR(125), security VARCHAR(125), status VARCHAR(10), trader VARCHAR(125), book VARCHAR(125), - creationName VARCHAR(125), - creationDate TIMESTAMP , - revisionName VARCHAR(125), - revisionDate TIMESTAMP , - dealName VARCHAR(125), - dealType VARCHAR(125), - sourceListId VARCHAR(125), + creation_name VARCHAR(125), + creation_date TIMESTAMP NULL, + revision_name VARCHAR(125), + revision_date TIMESTAMP NULL, + deal_name VARCHAR(125), + deal_type VARCHAR(125), + source_list_id VARCHAR(125), side VARCHAR(125), + PRIMARY KEY (bid_list_id) +); - PRIMARY KEY (BidListId) -) - -CREATE TABLE Trade ( - TradeId tinyint(4) NOT NULL AUTO_INCREMENT, +-- Table trade +DROP TABLE IF EXISTS trade; +CREATE TABLE trade ( + trade_id int NOT NULL AUTO_INCREMENT, account VARCHAR(30) NOT NULL, type VARCHAR(30) NOT NULL, - buyQuantity DOUBLE, - sellQuantity DOUBLE, - buyPrice DOUBLE , - sellPrice DOUBLE, - tradeDate TIMESTAMP, + buy_quantity DOUBLE, + sell_quantity DOUBLE, + buy_price DOUBLE, + sell_price DOUBLE, + trade_date TIMESTAMP NULL, security VARCHAR(125), status VARCHAR(10), trader VARCHAR(125), benchmark VARCHAR(125), book VARCHAR(125), - creationName VARCHAR(125), - creationDate TIMESTAMP , - revisionName VARCHAR(125), - revisionDate TIMESTAMP , - dealName VARCHAR(125), - dealType VARCHAR(125), - sourceListId VARCHAR(125), + creation_name VARCHAR(125), + creation_date TIMESTAMP NULL, + revision_name VARCHAR(125), + revision_date TIMESTAMP NULL, + deal_name VARCHAR(125), + deal_type VARCHAR(125), + source_list_id VARCHAR(125), side VARCHAR(125), + PRIMARY KEY (trade_id) +); - PRIMARY KEY (TradeId) -) - -CREATE TABLE CurvePoint ( - Id tinyint(4) NOT NULL AUTO_INCREMENT, - CurveId tinyint, - asOfDate TIMESTAMP, - term DOUBLE , - value DOUBLE , - creationDate TIMESTAMP , - - PRIMARY KEY (Id) -) - -CREATE TABLE Rating ( - Id tinyint(4) NOT NULL AUTO_INCREMENT, - moodysRating VARCHAR(125), - sandPRating VARCHAR(125), - fitchRating VARCHAR(125), - orderNumber tinyint, +-- Table curvepoint +DROP TABLE IF EXISTS curvepoint; +CREATE TABLE curvepoint ( + id int NOT NULL AUTO_INCREMENT, + curve_id int, + as_of_date TIMESTAMP NULL, + term DOUBLE, + value DOUBLE, + creation_date TIMESTAMP NULL, + PRIMARY KEY (id) +); - PRIMARY KEY (Id) -) +-- Table rating +DROP TABLE IF EXISTS rating; +CREATE TABLE rating ( + id int NOT NULL AUTO_INCREMENT, + moodys_rating VARCHAR(125), + sand_p_rating VARCHAR(125), + fitch_rating VARCHAR(125), + order_number int, + PRIMARY KEY (id) +); -CREATE TABLE RuleName ( - Id tinyint(4) NOT NULL AUTO_INCREMENT, +-- Table rulename +DROP TABLE IF EXISTS rulename; +CREATE TABLE rulename ( + id int NOT NULL AUTO_INCREMENT, name VARCHAR(125), description VARCHAR(125), json VARCHAR(125), template VARCHAR(512), - sqlStr VARCHAR(125), - sqlPart VARCHAR(125), + sql_str VARCHAR(125), + sql_part VARCHAR(125), + PRIMARY KEY (id) +); + +-- Table users +DROP TABLE IF EXISTS users; +CREATE TABLE users ( + id int NOT NULL AUTO_INCREMENT, + username VARCHAR(125) NOT NULL, + password VARCHAR(125) NOT NULL, + fullname VARCHAR(125) NOT NULL, + role VARCHAR(125) NOT NULL, + PRIMARY KEY (id) +); + +-- Initial data +INSERT INTO users (fullname, username, password, role) VALUES +("Administrator", "admin", "$2a$10$pBV8ILO/s/nao4wVnGLrh.sa/rnr5pDpbeC4E.KNzQWoy8obFZdaa", "ADMIN"), +("User", "user", "$2a$10$pBV8ILO/s/nao4wVnGLrh.sa/rnr5pDpbeC4E.KNzQWoy8obFZdaa", "USER"); + +INSERT INTO bidlist (account, type, bid_quantity, ask_quantity, bid, ask, benchmark, bid_list_date, commentary, security, status, trader, book, creation_name, creation_date, revision_name, revision_date, deal_name, deal_type, source_list_id, side) VALUES +('Account1', 'TypeA', 100.0, 150.0, 50.5, 51.0, 'BenchmarkA', NOW(), 'CommentA', 'SecurityA', 'Active', 'TraderA', 'BookA', 'Admin', NOW(), 'Admin', NOW(), 'DealA', 'Type1', 'SourceA', 'SideA'), +('Account2', 'TypeB', 200.0, 250.0, 60.5, 61.0, 'BenchmarkB', NOW(), 'CommentB', 'SecurityB', 'Inactive', 'TraderB', 'BookB', 'Admin', NOW(), 'Admin', NOW(), 'DealB', 'Type2', 'SourceB', 'SideB'); + +INSERT INTO trade (account, type, buy_quantity, sell_quantity, buy_price, sell_price, trade_date, security, status, trader, benchmark, book, creation_name, creation_date, revision_name, revision_date, deal_name, deal_type, source_list_id, side) VALUES +('Account1', 'TypeA', 50.0, 40.0, 100.5, 102.0, NOW(), 'SecurityA', 'Completed', 'TraderA', 'BenchmarkA', 'BookA', 'Admin', NOW(), 'Admin', NOW(), 'DealA', 'Type1', 'SourceA', 'SideA'), +('Account2', 'TypeB', 60.0, 55.0, 120.5, 122.0, NOW(), 'SecurityB', 'Pending', 'TraderB', 'BenchmarkB', 'BookB', 'Admin', NOW(), 'Admin', NOW(), 'DealB', 'Type2', 'SourceB', 'SideB'); - PRIMARY KEY (Id) -) +INSERT INTO curvepoint (curve_id, as_of_date, term, value, creation_date) VALUES +(1, NOW(), 5.0, 105.5, NOW()), +(2, NOW(), 10.0, 110.0, NOW()); -CREATE TABLE Users ( - Id tinyint(4) NOT NULL AUTO_INCREMENT, - username VARCHAR(125), - password VARCHAR(125), - fullname VARCHAR(125), - role VARCHAR(125), +INSERT INTO rating (moodys_rating, sand_p_rating, fitch_rating, order_number) VALUES +('A1', 'AA', 'AAA', 1), +('Baa2', 'BBB', 'BB+', 2); - PRIMARY KEY (Id) -) +INSERT INTO rulename (name, description, json, template, sql_str, sql_part) VALUES +('Rule1', 'Description1', '{"rule":"test1"}', 'Template1', 'SELECT * FROM trade', 'SQLPart1'), +('Rule2', 'Description2', '{"rule":"test2"}', 'Template2', 'SELECT * FROM bidlist', 'SQLPart2'); -insert into Users(fullname, username, password, role) values("Administrator", "admin", "$2a$10$pBV8ILO/s/nao4wVnGLrh.sa/rnr5pDpbeC4E.KNzQWoy8obFZdaa", "ADMIN") -insert into Users(fullname, username, password, role) values("User", "user", "$2a$10$pBV8ILO/s/nao4wVnGLrh.sa/rnr5pDpbeC4E.KNzQWoy8obFZdaa", "USER") \ No newline at end of file +-- Verifications +SELECT * FROM users; +SELECT * FROM rulename; +SELECT * FROM rating; +SELECT * FROM curvepoint; +SELECT * FROM trade; +SELECT * FROM bidlist; \ No newline at end of file diff --git a/Poseiden-skeleton/doc/guideline.txt b/Poseiden-skeleton/doc/guideline.txt index 38ae324254..4279260991 100644 --- a/Poseiden-skeleton/doc/guideline.txt +++ b/Poseiden-skeleton/doc/guideline.txt @@ -23,6 +23,7 @@ 3. Create controller class and place in package com.nnk.springboot.controllers 4. Create view files and place in src/main/resource/templates + ## Write unit test 1. Create unit test and place in package com.nnk.springboot in folder test > java diff --git a/Poseiden-skeleton/pom.xml b/Poseiden-skeleton/pom.xml index 255c20c118..dcfc51e9bf 100644 --- a/Poseiden-skeleton/pom.xml +++ b/Poseiden-skeleton/pom.xml @@ -1,7 +1,7 @@ + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 net.guides.springboothelloworld @@ -16,7 +16,7 @@ org.springframework.boot spring-boot-starter-parent 3.1.0 - + @@ -26,53 +26,146 @@ + org.springframework.boot - spring-boot-starter-test - 3.1.0 - test + spring-boot-starter + + - org.springframework.boot - spring-boot-starter-data-jpa - 3.1.0 + org.springframework.boot + spring-boot-starter-web + + - org.springframework.boot - spring-boot-starter-web - 3.1.0 + org.springframework.boot + spring-boot-starter-data-jpa + + org.springframework.boot spring-boot-starter-thymeleaf + + org.springframework.boot spring-boot-starter-security - 3.1.0 + + org.springframework.boot spring-boot-devtools true - 3.1.0 + + mysql mysql-connector-java + 8.0.33 + + com.h2database h2 + 2.1.214 + + + + + org.springframework.boot + spring-boot-starter-test + test + + + + + jakarta.validation + jakarta.validation-api + + + + + junit + junit + test + + + + + org.projectlombok + lombok + 1.18.30 + provided + + + + + org.springframework.boot + spring-boot-starter-validation + org.springframework.boot spring-boot-maven-plugin + + + + org.apache.maven.plugins + maven-surefire-plugin + 3.0.0-M7 + + ${jacoco.argLine} + 1 + true + false + + **/*Test.java + + + + + + + + org.jacoco + jacoco-maven-plugin + 0.8.9 + + + prepare-agent + + prepare-agent + + + jacoco.argLine + ${project.build.directory}/jacoco.exec + + + + report + verify + + report + + + ${project.build.directory}/jacoco.exec + ${project.build.directory}/jacoco + + + + - \ No newline at end of file + diff --git a/Poseiden-skeleton/spring-boot-skeleton.iml b/Poseiden-skeleton/spring-boot-skeleton.iml deleted file mode 100644 index bac7aedc30..0000000000 --- a/Poseiden-skeleton/spring-boot-skeleton.iml +++ /dev/null @@ -1,102 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/Application.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/Application.java index f53aeb0fba..b3092240a2 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/Application.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/Application.java @@ -1,12 +1,14 @@ -package com.nnk.springboot; - -import org.springframework.boot.SpringApplication; -import org.springframework.boot.autoconfigure.SpringBootApplication; - -@SpringBootApplication -public class Application { - - public static void main(String[] args) { - SpringApplication.run(Application.class, args); - } -} +package com.nnk.springboot; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; + +@SpringBootApplication +public class Application { + + public static void main(String[] args) { + SpringApplication.run(Application.class, args); + } +} + + diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/config/SecurityConfig.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/config/SecurityConfig.java new file mode 100644 index 0000000000..42a9c595b3 --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/config/SecurityConfig.java @@ -0,0 +1,112 @@ +package com.nnk.springboot.config; + +import com.nnk.springboot.repositories.UserRepository; +import com.nnk.springboot.services.CustomUserDetailsService; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import org.springframework.security.crypto.password.PasswordEncoder; +import org.springframework.security.web.SecurityFilterChain; +import org.springframework.security.web.csrf.CookieCsrfTokenRepository; +import org.springframework.security.web.header.writers.XXssProtectionHeaderWriter; +import org.springframework.security.web.authentication.AuthenticationSuccessHandler; + +/** + * Security configuration for the application. + */ +@Configuration +public class SecurityConfig { + + private final CustomUserDetailsService userDetailsService; + + /** + * Constructor for SecurityConfig that initializes the user service. + * + * @param userRepository the user repository + */ + public SecurityConfig(UserRepository userRepository) { + this.userDetailsService = new CustomUserDetailsService(userRepository); + } + + /** + * Defines the password encoder used to secure user passwords. + * + * @return a BCrypt password encoder + */ + @Bean + public BCryptPasswordEncoder passwordEncoder() { + return new BCryptPasswordEncoder(); + } + + /** + * Defines the HTTP security configuration for the application. + * + * @param http the HttpSecurity object used to configure security + * @return the configured SecurityFilterChain object + * @throws Exception if an error occurs during configuration + */ + @Bean + public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception { + http + .userDetailsService(userDetailsService) + .authorizeHttpRequests(auth -> auth + .requestMatchers("/css/**", "/js/**", "/images/**").permitAll() + .requestMatchers("/", "/login", "/error").permitAll() + .requestMatchers("/user/**").hasRole("ADMIN") + .anyRequest().authenticated() + ) + .formLogin(form -> form + .loginPage("/login") + .successHandler(customAuthenticationSuccessHandler()) + .failureUrl("/login?error=true") + .permitAll() + ) + .logout(logout -> logout + .logoutUrl("/logout") + .logoutSuccessUrl("/logout-success") + .invalidateHttpSession(true) + .deleteCookies("JSESSIONID") + .permitAll() + ) + .sessionManagement(session -> session + .maximumSessions(1) + .expiredUrl("/login?expired") + ) + .exceptionHandling(ex -> ex + .accessDeniedPage("/403") + ) + .csrf(csrf -> csrf + .csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse()) + ) + .headers(headers -> headers + .xssProtection(xss -> xss + .headerValue(XXssProtectionHeaderWriter.HeaderValue.ENABLED_MODE_BLOCK) + ) + .contentSecurityPolicy(csp -> csp + .policyDirectives("frame-ancestors 'self'") + ) + ); + return http.build(); + } + + /** + * Defines a custom authentication success handler that redirects users based on their role. + * + * @return an authentication success handler + */ + @Bean + public AuthenticationSuccessHandler customAuthenticationSuccessHandler() { + return (request, response, authentication) -> { + String role = authentication.getAuthorities().iterator().next().getAuthority(); + System.out.println("Authority received in handler: " + role); + if (role.equals("ROLE_ADMIN")) { + System.out.println("Redirecting to admin home"); + response.sendRedirect("/admin/home"); + } else { + System.out.println("Redirecting to bid list"); + response.sendRedirect("/bidList/list"); + } + }; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/BidListController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/BidListController.java index a31b9b53ca..db51ad4338 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/BidListController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/BidListController.java @@ -1,55 +1,133 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.domain.BidList; -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.validation.BindingResult; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; - -import javax.validation.Valid; - - -@Controller -public class BidListController { - // TODO: Inject Bid service - - @RequestMapping("/bidList/list") - public String home(Model model) - { - // TODO: call service find all bids to show to the view - return "bidList/list"; - } - - @GetMapping("/bidList/add") - public String addBidForm(BidList bid) { - return "bidList/add"; - } - - @PostMapping("/bidList/validate") - public String validate(@Valid BidList bid, BindingResult result, Model model) { - // TODO: check data valid and save to db, after saving return bid list - return "bidList/add"; - } - - @GetMapping("/bidList/update/{id}") - public String showUpdateForm(@PathVariable("id") Integer id, Model model) { - // TODO: get Bid by Id and to model then show to the form - return "bidList/update"; - } - - @PostMapping("/bidList/update/{id}") - public String updateBid(@PathVariable("id") Integer id, @Valid BidList bidList, - BindingResult result, Model model) { - // TODO: check required fields, if valid call service to update Bid and return list Bid - return "redirect:/bidList/list"; - } - - @GetMapping("/bidList/delete/{id}") - public String deleteBid(@PathVariable("id") Integer id, Model model) { - // TODO: Find Bid by Id and delete the bid, return to Bid list - return "redirect:/bidList/list"; - } -} +package com.nnk.springboot.controllers; + +import com.nnk.springboot.domain.BidList; +import com.nnk.springboot.services.BidListService; +import jakarta.validation.Valid; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; + +import java.util.List; + +/** + * Controller for handling bid list operations. + */ +@Controller +@RequestMapping("/bidList") +public class BidListController { + + + private final BidListService bidListService; + + /** + * Constructor for BidListController. + * + * @param bidListService the service for bid list operations + */ + public BidListController(BidListService bidListService) { + this.bidListService = bidListService; + } + + /** + * Displays the list of bids. + * + * @param model the model to hold bid list data + * @return the bid list view + */ + @RequestMapping("/list") + public String home(Model model) { + List bids = bidListService.getAllBidList(); + model.addAttribute("bidLists", bids); + + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "bidList/list"; + } + + /** + * Displays the form to add a new bid. + * + * @param bid the bid object to be added + * @return the add bid form view + */ + @GetMapping("/add") + public String addBidForm(BidList bid) { + return "bidList/add"; + } + + /** + * Validates and saves a new bid. + * + * @param bid the bid to be validated and saved + * @param result the result of validation + * @param model the model to hold bid list data + * @return redirect to the bid list if successful, otherwise return to the add form + */ + @PostMapping("/validate") + public String validate(@Valid BidList bid, BindingResult result, Model model) { + if (result.hasErrors()) { + return "bidList/add"; + } + bidListService.saveBidList(bid); + model.addAttribute("bidLists", bidListService.getAllBidList()); + return "redirect:/bidList/list"; + } + + /** + * Displays the form to update an existing bid. + * + * @param id the id of the bid to update + * @param model the model to hold bid data + * @return the update bid form view + */ + @GetMapping("/update/{id}") + public String showUpdateForm(@PathVariable("id") Integer id, Model model) { + BidList bidList = bidListService.getBidListById(id).orElseThrow(() -> new IllegalArgumentException("Invalid bidList Id:" + id)); + model.addAttribute("bidList", bidList); + return "bidList/update"; + } + + /** + * Updates an existing bid. + * + * @param id the id of the bid to update + * @param bidList the updated bid data + * @param result the result of validation + * @param model the model to hold bid list data + * @return redirect to the bid list if successful, otherwise return to the update form + */ + @PostMapping("/update/{id}") + public String updateBid(@PathVariable("id") Integer id, @Valid BidList bidList, + BindingResult result, Model model) { + if (result.hasErrors()) { + return "bidList/update"; + } + bidList.setBidListId(id); + bidListService.saveBidList(bidList); + model.addAttribute("bidLists", bidListService.getAllBidList()); + return "redirect:/bidList/list"; + } + + /** + * Deletes a bid. + * + * @param id the id of the bid to delete + * @param model the model to hold bid list data + * @return redirect to the bid list view + */ + @GetMapping("/delete/{id}") + public String deleteBid(@PathVariable("id") Integer id, Model model) { + BidList bidList = bidListService.getBidListById(id).orElseThrow(() -> new IllegalArgumentException("Invalid bidList Id:" + id)); + bidListService.deleteBidListById(id); + model.addAttribute("bidLists", bidListService.getAllBidList()); + return "redirect:/bidList/list"; + } +} \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/CurveController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/CurveController.java deleted file mode 100644 index db69caf549..0000000000 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/CurveController.java +++ /dev/null @@ -1,54 +0,0 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.domain.CurvePoint; -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.validation.BindingResult; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; - -import javax.validation.Valid; - -@Controller -public class CurveController { - // TODO: Inject Curve Point service - - @RequestMapping("/curvePoint/list") - public String home(Model model) - { - // TODO: find all Curve Point, add to model - return "curvePoint/list"; - } - - @GetMapping("/curvePoint/add") - public String addBidForm(CurvePoint bid) { - return "curvePoint/add"; - } - - @PostMapping("/curvePoint/validate") - public String validate(@Valid CurvePoint curvePoint, BindingResult result, Model model) { - // TODO: check data valid and save to db, after saving return Curve list - return "curvePoint/add"; - } - - @GetMapping("/curvePoint/update/{id}") - public String showUpdateForm(@PathVariable("id") Integer id, Model model) { - // TODO: get CurvePoint by Id and to model then show to the form - return "curvePoint/update"; - } - - @PostMapping("/curvePoint/update/{id}") - public String updateBid(@PathVariable("id") Integer id, @Valid CurvePoint curvePoint, - BindingResult result, Model model) { - // TODO: check required fields, if valid call service to update Curve and return Curve list - return "redirect:/curvePoint/list"; - } - - @GetMapping("/curvePoint/delete/{id}") - public String deleteBid(@PathVariable("id") Integer id, Model model) { - // TODO: Find Curve by Id and delete the Curve, return to Curve list - return "redirect:/curvePoint/list"; - } -} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/CurvePointController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/CurvePointController.java new file mode 100644 index 0000000000..f778ba2dba --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/CurvePointController.java @@ -0,0 +1,131 @@ +package com.nnk.springboot.controllers; + +import com.nnk.springboot.domain.CurvePoint; +import com.nnk.springboot.services.CurvePointService; +import jakarta.validation.Valid; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; + +/** + * Controller for managing CurvePoint operations. + */ +@Controller +@RequestMapping("/curvePoint") +public class CurvePointController { + + private final CurvePointService curvePointService; + + /** + * Constructor for CurvePointController. + * + * @param curvePointService the service handling curve point operations + */ + public CurvePointController(CurvePointService curvePointService) { + this.curvePointService = curvePointService; + } + + /** + * Displays the list of all curve points. + * + * @param model the model to hold curve point data + * @return the curve point list view + */ + @RequestMapping("/list") + public String home(Model model) { + model.addAttribute("curvePoints", curvePointService.getAllCurvePoint()); + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "curvePoint/list"; + } + + /** + * Displays the form to add a new curve point. + * + * @param curvePoint the curve point object to be added + * @return the add curve point form view + */ + @GetMapping("/add") + public String addCurvePointForm(CurvePoint curvePoint) { + return "curvePoint/add"; + } + + /** + * Validates and saves a new curve point. + * + * @param curvePoint the curve point to be validated and saved + * @param result the result of validation + * @param model the model to hold curve point data + * @return redirect to the curve point list if successful, otherwise return to the add form + */ + @PostMapping("/validate") + public String validate(@Valid CurvePoint curvePoint, BindingResult result, Model model) { + if (result.hasErrors()){ + return "curvePoint/add"; + } + curvePointService.saveCurvePoint(curvePoint); + model.addAttribute("curvePoints", curvePointService.getAllCurvePoint()); + return "redirect:/curvePoint/list"; + } + + /** + * Displays the form to update an existing curve point. + * + * @param id the id of the curve point to update + * @param model the model to hold curve point data + * @return the update curve point form view + */ + @GetMapping("/update/{id}") + public String showUpdateForm(@PathVariable("id") Integer id, Model model) { + CurvePoint curvePoint = curvePointService.getCurvePointById(id) + .orElseThrow(()-> new IllegalArgumentException("Invalid CurvePoint Id:" + id)); + model.addAttribute("curvePoint", curvePoint); + return "curvePoint/update"; + } + + /** + * Updates an existing curve point. + * + * @param id the id of the curve point to update + * @param curvePoint the updated curve point data + * @param result the result of validation + * @param model the model to hold curve point data + * @return redirect to the curve point list if successful, otherwise return to the update form + */ + @PostMapping("/update/{id}") + public String updateCurve(@PathVariable("id") Integer id, @Valid CurvePoint curvePoint, + BindingResult result, Model model) { + if (result.hasErrors()){ + return "curvePoint/update"; + } + curvePoint.setId(id); + curvePointService.saveCurvePoint(curvePoint); + model.addAttribute("curvePoints", curvePointService.getAllCurvePoint()); + return "redirect:/curvePoint/list"; + } + + /** + * Deletes a curve point. + * + * @param id the id of the curve point to delete + * @param model the model to hold curve point data + * @return redirect to the curve point list view + */ + @GetMapping("/delete/{id}") + public String deleteCurve(@PathVariable("id") Integer id, Model model) { + CurvePoint curvePoint = curvePointService.getCurvePointById(id) + .orElseThrow(()-> new IllegalArgumentException("Invalid CurvePoint Id:" + id)); + curvePointService.deleteCurvePointById(id); + model.addAttribute("curvePoints", curvePointService.getAllCurvePoint()); + return "redirect:/curvePoint/list"; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/HomeController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/HomeController.java index 50685b213f..b710a97d88 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/HomeController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/HomeController.java @@ -1,23 +1,44 @@ -package com.nnk.springboot.controllers; - -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.web.bind.annotation.RequestMapping; - -@Controller -public class HomeController -{ - @RequestMapping("/") - public String home(Model model) - { - return "home"; - } - - @RequestMapping("/admin/home") - public String adminHome(Model model) - { - return "redirect:/bidList/list"; - } - - -} +package com.nnk.springboot.controllers; + +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.web.bind.annotation.GetMapping; + +/** + * Controller to handle home and admin home page requests. + */ +@Controller +public class HomeController { + + /** + * Displays the home page with the authenticated user's username. + * + * @param model the model to hold user information + * @return the home view + */ + @GetMapping("/") + public String home(Model model) { + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "home"; + } + + /** + * Redirects admins to the bid list page after authentication. + * + * @param model the model to hold user information + * @return redirect to the bid list view + */ + @GetMapping("/admin/home") + public String adminHome(Model model) { + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "redirect:/bidList/list"; + } +} + diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LoginController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LoginController.java index ef0c657c72..76db457441 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LoginController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LoginController.java @@ -1,40 +1,72 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.repositories.UserRepository; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Controller; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.servlet.ModelAndView; - -@Controller -@RequestMapping("app") -public class LoginController { - - @Autowired - private UserRepository userRepository; - - @GetMapping("login") - public ModelAndView login() { - ModelAndView mav = new ModelAndView(); - mav.setViewName("login"); - return mav; - } - - @GetMapping("secure/article-details") - public ModelAndView getAllUserArticles() { - ModelAndView mav = new ModelAndView(); - mav.addObject("users", userRepository.findAll()); - mav.setViewName("user/list"); - return mav; - } - - @GetMapping("error") - public ModelAndView error() { - ModelAndView mav = new ModelAndView(); - String errorMessage= "You are not authorized for the requested data."; - mav.addObject("errorMsg", errorMessage); - mav.setViewName("403"); - return mav; - } -} +package com.nnk.springboot.controllers; + +import com.nnk.springboot.repositories.UserRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.servlet.ModelAndView; + +/** + * Controller for handling login and access-related requests. + */ +@Controller +public class LoginController { + + private final UserRepository userRepository; + + /** + * Constructor for LoginController. + * + * @param userRepository the user repository + */ + public LoginController(UserRepository userRepository) { + this.userRepository = userRepository; + } + + /** + * Displays the login page. + * + * @return the login view + */ + @GetMapping("/login") + public ModelAndView login() { + ModelAndView mav = new ModelAndView(); + mav.setViewName("login"); + return mav; + } + + /** + * Retrieves and displays a list of users for authorized users (role = admin). + * + * @return the user list view + */ + @GetMapping("/secure/article-details") + public ModelAndView getAllUserArticles() { + ModelAndView mav = new ModelAndView(); + mav.addObject("users", userRepository.findAll()); + mav.setViewName("user/list"); + return mav; + } + + /** + * Displays the access denied (403) error page. + * + * @return the 403 error view + */ + @GetMapping("/403") + public ModelAndView error() { + ModelAndView mav = new ModelAndView(); + String errorMessage= "You are not authorized for the requested data."; + mav.addObject("errorMsg", errorMessage); + mav.setViewName("403"); + + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = (UserDetails) authentication.getPrincipal(); + mav.addObject("username", userDetails.getUsername()); + + return mav; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LogoutController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LogoutController.java new file mode 100644 index 0000000000..7edecfcc9e --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/LogoutController.java @@ -0,0 +1,24 @@ +package com.nnk.springboot.controllers; + +import org.springframework.stereotype.Controller; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.servlet.ModelAndView; + +/** + * Controller for handling user logout. + */ +@Controller +public class LogoutController { + + /** + * Displays the logout success page after the user logs out. + * + * @return the logout view + */ + @GetMapping("/logout-success") + public ModelAndView logoutSuccess() { + ModelAndView mav = new ModelAndView(); + mav.setViewName("logout"); + return mav; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RatingController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RatingController.java index 5e15e68fbc..76d335e731 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RatingController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RatingController.java @@ -1,54 +1,132 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.domain.Rating; -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.validation.BindingResult; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; - -import javax.validation.Valid; - -@Controller -public class RatingController { - // TODO: Inject Rating service - - @RequestMapping("/rating/list") - public String home(Model model) - { - // TODO: find all Rating, add to model - return "rating/list"; - } - - @GetMapping("/rating/add") - public String addRatingForm(Rating rating) { - return "rating/add"; - } - - @PostMapping("/rating/validate") - public String validate(@Valid Rating rating, BindingResult result, Model model) { - // TODO: check data valid and save to db, after saving return Rating list - return "rating/add"; - } - - @GetMapping("/rating/update/{id}") - public String showUpdateForm(@PathVariable("id") Integer id, Model model) { - // TODO: get Rating by Id and to model then show to the form - return "rating/update"; - } - - @PostMapping("/rating/update/{id}") - public String updateRating(@PathVariable("id") Integer id, @Valid Rating rating, - BindingResult result, Model model) { - // TODO: check required fields, if valid call service to update Rating and return Rating list - return "redirect:/rating/list"; - } - - @GetMapping("/rating/delete/{id}") - public String deleteRating(@PathVariable("id") Integer id, Model model) { - // TODO: Find Rating by Id and delete the Rating, return to Rating list - return "redirect:/rating/list"; - } -} +package com.nnk.springboot.controllers; + +import com.nnk.springboot.domain.Rating; +import com.nnk.springboot.services.RatingService; +import jakarta.validation.Valid; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; + +import java.util.List; + +/** + * Controller to manage Rating entities. + */ +@Controller +@RequestMapping("/rating") +public class RatingController { + + + private final RatingService ratingService; + + /** + * Constructor injection for RatingService. + * + * @param ratingService the rating service + */ + public RatingController(RatingService ratingService) { + this.ratingService = ratingService; + } + + /** + * Displays the list of ratings. + * + * @param model the model + * @return the rating list view + */ + @RequestMapping("/list") + public String home(Model model) { + model.addAttribute("ratings", ratingService.getAllRating()); + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "rating/list"; + } + + /** + * Displays the form to add a new rating. + * + * @param rating a new rating object + * @return the add rating view + */ + @GetMapping("/add") + public String addRatingForm(Rating rating) { + return "rating/add"; + } + + /** + * Validates and saves a new rating. + * + * @param rating the rating object + * @param result validation result + * @param model the model + * @return the redirect path + */ + @PostMapping("/validate") + public String validate(@Valid Rating rating, BindingResult result, Model model) { + if (result.hasErrors()){ + return "rating/add"; + } + ratingService.saveRating(rating); + model.addAttribute("ratings", ratingService.getAllRating()); + return "redirect:/rating/list"; + } + + /** + * Displays the form to update an existing rating. + * + * @param id the rating id + * @param model the model + * @return the update rating view + */ + @GetMapping("/update/{id}") + public String showUpdateForm(@PathVariable("id") Integer id, Model model) { + Rating rating = ratingService.getRatingById(id).orElseThrow(()-> new IllegalArgumentException("Invalid Rating Id:" + id)); + model.addAttribute("rating", rating); + return "rating/update"; + } + + /** + * Updates an existing rating. + * + * @param id the rating id + * @param rating the updated rating object + * @param result validation result + * @param model the model + * @return the redirect path + */ + @PostMapping("/update/{id}") + public String updateRating(@PathVariable("id") Integer id, @Valid Rating rating, + BindingResult result, Model model) { + if (result.hasErrors()){ + return "rating/update"; + } + rating.setId(id); + ratingService.saveRating(rating); + model.addAttribute("ratings", ratingService.getAllRating()); + return "redirect:/rating/list"; + } + + /** + * Deletes a rating. + * + * @param id the rating id + * @param model the model + * @return the redirect path + */ + @GetMapping("/delete/{id}") + public String deleteRating(@PathVariable("id") Integer id, Model model) { + Rating rating = ratingService.getRatingById(id).orElseThrow(()-> new IllegalArgumentException("Invalid Rating Id:" + id)); + ratingService.deleteRatingById(id); + model.addAttribute("ratings", ratingService.getAllRating()); + return "redirect:/rating/list"; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RuleNameController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RuleNameController.java index b9e72b1ba6..7813f1f2f0 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RuleNameController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/RuleNameController.java @@ -1,54 +1,127 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.domain.RuleName; -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.validation.BindingResult; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; - -import javax.validation.Valid; - -@Controller -public class RuleNameController { - // TODO: Inject RuleName service - - @RequestMapping("/ruleName/list") - public String home(Model model) - { - // TODO: find all RuleName, add to model - return "ruleName/list"; - } - - @GetMapping("/ruleName/add") - public String addRuleForm(RuleName bid) { - return "ruleName/add"; - } - - @PostMapping("/ruleName/validate") - public String validate(@Valid RuleName ruleName, BindingResult result, Model model) { - // TODO: check data valid and save to db, after saving return RuleName list - return "ruleName/add"; - } - - @GetMapping("/ruleName/update/{id}") - public String showUpdateForm(@PathVariable("id") Integer id, Model model) { - // TODO: get RuleName by Id and to model then show to the form - return "ruleName/update"; - } - - @PostMapping("/ruleName/update/{id}") - public String updateRuleName(@PathVariable("id") Integer id, @Valid RuleName ruleName, - BindingResult result, Model model) { - // TODO: check required fields, if valid call service to update RuleName and return RuleName list - return "redirect:/ruleName/list"; - } - - @GetMapping("/ruleName/delete/{id}") - public String deleteRuleName(@PathVariable("id") Integer id, Model model) { - // TODO: Find RuleName by Id and delete the RuleName, return to Rule list - return "redirect:/ruleName/list"; - } -} +package com.nnk.springboot.controllers; + +import com.nnk.springboot.domain.RuleName; +import com.nnk.springboot.services.RuleNameService; +import jakarta.validation.Valid; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; + +import java.util.List; + +/** + * Controller to manage RuleName entities. + */ +@Controller +@RequestMapping("/ruleName") +public class RuleNameController { + + private final RuleNameService ruleNameService; + + /** + * Constructor-based dependency injection. + * + * @param ruleNameService the rule name service + */ + public RuleNameController(RuleNameService ruleNameService) { + this.ruleNameService = ruleNameService; + } + + /** + * Displays the list of rule names. + * + * @param model the model + * @return the rule name list view + */ + @RequestMapping("/list") + public String home(Model model) { + model.addAttribute("ruleNames", ruleNameService.getAllRuleNames()); + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "ruleName/list"; + } + + /** + * Displays the form to add a new rule name. + * + * @return the add rule name view + */ + @GetMapping("/add") + public String addRuleNameForm(RuleName bid) { + return "ruleName/add"; + } + + /** + * Validates and saves a new rule name. + * + * @param ruleName the rule name object + * @param result validation result + * @return the redirect path + */ + @PostMapping("/validate") + public String validate(@Valid RuleName ruleName, BindingResult result, Model model) { + if (result.hasErrors()){ + return "ruleName/add"; + } + ruleNameService.saveRuleName(ruleName); + model.addAttribute("ruleNames", ruleNameService.getAllRuleNames()); + return "redirect:/ruleName/list"; + } + + /** + * Displays the form to update an existing rule name. + * + * @param id the rule name id + * @param model the model + * @return the update rule name view + */ + @GetMapping("/update/{id}") + public String showUpdateForm(@PathVariable("id") Integer id, Model model) { + RuleName ruleName = ruleNameService.getRuleNameById(id).orElseThrow(()-> new IllegalArgumentException("Invalid RuleName Id:" + id)); + model.addAttribute("ruleName", ruleName); + return "ruleName/update"; + } + + /** + * Updates an existing rule name. + * + * @param id the rule name id + * @param ruleName the updated rule name object + * @param result validation result + * @return the redirect path + */ + @PostMapping("/update/{id}") + public String updateRuleName(@PathVariable("id") Integer id, @Valid RuleName ruleName, + BindingResult result, Model model) { + if(result.hasErrors()){ + return "ruleName/update"; + } + ruleName.setId(id); + ruleNameService.saveRuleName(ruleName); + model.addAttribute("ruleNames", ruleNameService.getAllRuleNames()); + return "redirect:/ruleName/list"; + } + + /** + * Deletes a rule name. + * + * @param id the rule name id + * @return the redirect path + */ + @GetMapping("/delete/{id}") + public String deleteRuleName(@PathVariable("id") Integer id, Model model) { + RuleName ruleName = ruleNameService.getRuleNameById(id).orElseThrow(()-> new IllegalArgumentException("Invalid RuleName Id:" + id)); + ruleNameService.deleteRuleNameById(id); + model.addAttribute("ruleNames", ruleNameService.getAllRuleNames()); + return "redirect:/ruleName/list"; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/TradeController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/TradeController.java index 4e667eec22..47b05a775b 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/TradeController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/TradeController.java @@ -1,54 +1,133 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.domain.Trade; -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.validation.BindingResult; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; - -import javax.validation.Valid; - -@Controller -public class TradeController { - // TODO: Inject Trade service - - @RequestMapping("/trade/list") - public String home(Model model) - { - // TODO: find all Trade, add to model - return "trade/list"; - } - - @GetMapping("/trade/add") - public String addUser(Trade bid) { - return "trade/add"; - } - - @PostMapping("/trade/validate") - public String validate(@Valid Trade trade, BindingResult result, Model model) { - // TODO: check data valid and save to db, after saving return Trade list - return "trade/add"; - } - - @GetMapping("/trade/update/{id}") - public String showUpdateForm(@PathVariable("id") Integer id, Model model) { - // TODO: get Trade by Id and to model then show to the form - return "trade/update"; - } - - @PostMapping("/trade/update/{id}") - public String updateTrade(@PathVariable("id") Integer id, @Valid Trade trade, - BindingResult result, Model model) { - // TODO: check required fields, if valid call service to update Trade and return Trade list - return "redirect:/trade/list"; - } - - @GetMapping("/trade/delete/{id}") - public String deleteTrade(@PathVariable("id") Integer id, Model model) { - // TODO: Find Trade by Id and delete the Trade, return to Trade list - return "redirect:/trade/list"; - } -} +package com.nnk.springboot.controllers; + +import com.nnk.springboot.domain.Trade; +import com.nnk.springboot.services.TradeService; +import jakarta.validation.Valid; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; + +import java.util.List; + +/** + * Controller to manage Trade entities. + */ +@Controller +@RequestMapping("/trade") +public class TradeController { + + + private final TradeService tradeService; + + /** + * Constructor-based dependency injection. + * + * @param tradeService the service to manage trade entities + */ + public TradeController(TradeService tradeService) { + this.tradeService = tradeService; + } + + /** + * Displays the list of all trades. + * + * @param model the model to pass attributes to the view + * @return the view name for displaying the list of trades + */ + @RequestMapping("/list") + public String home(Model model) { + model.addAttribute("trades", tradeService.getAllTrades()); + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = ( UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername() ); + return "trade/list"; + } + + /** + * Displays the form to add a new trade. + * + * @return the add trade view + */ + @GetMapping("/add") + public String addTradeForm(Trade trade) { + return "trade/add"; + } + + /** + * Validates and saves a new trade. + * If there are validation errors, the user will be shown the add form again. + * + * @param trade the trade object populated with form data + * @param result the binding result that contains validation errors, if any + * @param model the model to pass attributes to the view + * @return the redirect path to the trade list if the trade is saved, or the add form if there are errors + */ + @PostMapping("/validate") + public String validate(@Valid Trade trade, BindingResult result, Model model) { + if (result.hasErrors()){ + return "trade/add"; + } + tradeService.saveTrade(trade); + model.addAttribute("trades", tradeService.getAllTrades()); + return "redirect:/trade/list"; + } + + /** + * Displays the form to update an existing trade. + * + * @param id the id of the trade to be updated + * @param model the model to pass the trade data to the view + * @return the update trade view + */ + @GetMapping("/update/{id}") + public String showUpdateForm(@PathVariable("id") Integer id, Model model) { + Trade trade = tradeService.getTradeById(id).orElseThrow(()-> new IllegalArgumentException("Invalid Trade Id: " + id)); + model.addAttribute("trade", trade); + return "trade/update"; + } + + /** + * Validates and updates an existing trade. + * If there are validation errors, the user will be shown the update form again. + * + * @param id the id of the trade to be updated + * @param trade the updated trade object + * @param result the binding result that contains validation errors, if any + * @param model the model to pass attributes to the view + * @return the redirect path to the trade list if the trade is updated, or the update form if there are errors + */ + @PostMapping("/update/{id}") + public String updateTrade(@PathVariable("id") Integer id, @Valid Trade trade, + BindingResult result, Model model) { + if (result.hasErrors()){ + return "trade/update"; + } + trade.setTradeId(id); + tradeService.saveTrade(trade); + model.addAttribute("trades", tradeService.getAllTrades()); + return "redirect:/trade/list"; + } + + /** + * Deletes a trade by its id. + * + * @param id the id of the trade to be deleted + * @param model the model to pass attributes to the view + * @return the redirect path to the trade list after deletion + */ + @GetMapping("/delete/{id}") + public String deleteTrade(@PathVariable("id") Integer id, Model model) { + Trade trade = tradeService.getTradeById(id).orElseThrow(()-> new IllegalArgumentException("Invalid Trade Id: " + id)); + tradeService.deleteTradeById(id); + model.addAttribute("trades", tradeService.getAllTrades()); + return "redirect:/trade/list"; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/UserController.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/UserController.java index 29e30be3d6..5e4038a524 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/UserController.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/controllers/UserController.java @@ -1,76 +1,130 @@ -package com.nnk.springboot.controllers; - -import com.nnk.springboot.domain.User; -import com.nnk.springboot.repositories.UserRepository; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; -import org.springframework.stereotype.Controller; -import org.springframework.ui.Model; -import org.springframework.validation.BindingResult; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestMapping; - -import javax.validation.Valid; - -@Controller -public class UserController { - @Autowired - private UserRepository userRepository; - - @RequestMapping("/user/list") - public String home(Model model) - { - model.addAttribute("users", userRepository.findAll()); - return "user/list"; - } - - @GetMapping("/user/add") - public String addUser(User bid) { - return "user/add"; - } - - @PostMapping("/user/validate") - public String validate(@Valid User user, BindingResult result, Model model) { - if (!result.hasErrors()) { - BCryptPasswordEncoder encoder = new BCryptPasswordEncoder(); - user.setPassword(encoder.encode(user.getPassword())); - userRepository.save(user); - model.addAttribute("users", userRepository.findAll()); - return "redirect:/user/list"; - } - return "user/add"; - } - - @GetMapping("/user/update/{id}") - public String showUpdateForm(@PathVariable("id") Integer id, Model model) { - User user = userRepository.findById(id).orElseThrow(() -> new IllegalArgumentException("Invalid user Id:" + id)); - user.setPassword(""); - model.addAttribute("user", user); - return "user/update"; - } - - @PostMapping("/user/update/{id}") - public String updateUser(@PathVariable("id") Integer id, @Valid User user, - BindingResult result, Model model) { - if (result.hasErrors()) { - return "user/update"; - } - - BCryptPasswordEncoder encoder = new BCryptPasswordEncoder(); - user.setPassword(encoder.encode(user.getPassword())); - user.setId(id); - userRepository.save(user); - model.addAttribute("users", userRepository.findAll()); - return "redirect:/user/list"; - } - - @GetMapping("/user/delete/{id}") - public String deleteUser(@PathVariable("id") Integer id, Model model) { - User user = userRepository.findById(id).orElseThrow(() -> new IllegalArgumentException("Invalid user Id:" + id)); - userRepository.delete(user); - model.addAttribute("users", userRepository.findAll()); - return "redirect:/user/list"; - } -} +package com.nnk.springboot.controllers; + +import com.nnk.springboot.domain.User; +import com.nnk.springboot.services.UserService; +import jakarta.validation.Valid; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.stereotype.Controller; +import org.springframework.ui.Model; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; + +/** + * Controller to manage User entities. + */ +@Controller +@RequestMapping("/user") +public class UserController { + + private final UserService userService; + + /** + * Constructor-based dependency injection. + * + * @param userService the user service to handle user-related operations + */ + public UserController(UserService userService) { + this.userService = userService; + } + + /** + * Displays the list of all users. + * + * @param model the model to pass attributes to the view + * @return the view name for displaying the list of users + */ + @RequestMapping("/list") + public String home(Model model) { + model.addAttribute("users", userService.getAllUsers()); + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + UserDetails userDetails = (UserDetails) authentication.getPrincipal(); + model.addAttribute("username", userDetails.getUsername()); + return "user/list"; + } + + /** + * Displays the form to add a new user. + * + * @return the add user view + */ + @GetMapping("/add") + public String addUserForm(User user) { + return "user/add"; + } + + /** + * Validates and saves a new user. + * If there are validation errors, the user will be shown the add form again. + * + * @param user the user object populated with form data + * @param result the binding result that contains validation errors, if any + * @param model the model to pass attributes to the view + * @return the redirect path to the user list if the user is saved, or the add form if there are errors + */ + @PostMapping("/validate") + public String validate(@Valid User user, BindingResult result, Model model) { + if (!result.hasErrors()) { + userService.saveUser(user); + model.addAttribute("users", userService.getAllUsers()); + return "redirect:/user/list"; + } + return "user/add"; + } + + /** + * Displays the form to update an existing user. + * + * @param id the id of the user to be updated + * @param model the model to pass the user data to the view + * @return the update user view + */ + @GetMapping("/update/{id}") + public String showUpdateForm(@PathVariable("id") Integer id, Model model) { + User user = userService.getUserById(id).orElseThrow(() -> new IllegalArgumentException("Invalid user Id:" + id)); + user.setPassword(""); // Clear password to not show the current password + model.addAttribute("user", user); + return "user/update"; + } + + /** + * Validates and updates an existing user. + * If there are validation errors, the user will be shown the update form again. + * + * @param id the id of the user to be updated + * @param user the updated user object + * @param result the binding result that contains validation errors, if any + * @param model the model to pass attributes to the view + * @return the redirect path to the user list if the user is updated, or the update form if there are errors + */ + @PostMapping("/update/{id}") + public String updateUser(@PathVariable("id") Integer id, @Valid User user, + BindingResult result, Model model) { + if (result.hasErrors()) { + return "user/update"; + } + + user.setId(id); + userService.saveUser(user); + model.addAttribute("users", userService.getAllUsers()); + return "redirect:/user/list"; + } + + /** + * Deletes a user by its id. + * + * @param id the id of the user to be deleted + * @param model the model to pass attributes to the view + * @return the redirect path to the user list after deletion + */ + @GetMapping("/delete/{id}") + public String deleteUser(@PathVariable("id") Integer id, Model model) { + userService.deleteUserById(id); + model.addAttribute("users", userService.getAllUsers()); + return "redirect:/user/list"; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/BidList.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/BidList.java index 3a0e27efc8..3a2ebdcce9 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/BidList.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/BidList.java @@ -1,15 +1,83 @@ -package com.nnk.springboot.domain; - -import org.springframework.beans.factory.annotation.Required; - -import javax.persistence.*; -import javax.validation.constraints.Digits; -import javax.validation.constraints.NotBlank; -import java.sql.Date; -import java.sql.Timestamp; - -@Entity -@Table(name = "bidlist") -public class BidList { - // TODO: Map columns in data table BIDLIST with corresponding java fields -} +package com.nnk.springboot.domain; + +import jakarta.persistence.*; +import jakarta.validation.constraints.NotBlank; +import jakarta.validation.constraints.NotNull; +import lombok.Data; +import java.sql.Timestamp; + +@Entity +@Data +@Table(name = "bidlist") +public class BidList { + + @Id + @GeneratedValue(strategy = GenerationType.AUTO) + @Column(name = "bid_list_id") + private int bidListId; + + @NotBlank(message = "Account is mandatory") + @Column(name = "account") + private String account; + + @NotBlank(message = "Type is mandatory") + @Column(name = "type") + private String type; + + @Column(name = "bid_quantity") + private double bidQuantity; + + @Column(name = "ask_quantity") + private double askQuantity; + + @Column(name = "bid") + private double bid; + + @Column(name = "ask") + private double ask; + + @Column(name = "benchmark") + private String benchmark; + + @Column(name = "bid_list_date") + private Timestamp bidListDate; + + @Column(name = "commentary") + private String commentary; + + @Column(name = "security") + private String security; + + @Column(name = "status") + private String status; + + @Column(name = "trader") + private String trader; + + @Column(name = "book") + private String book; + + @Column(name = "creation_name") + private String creationName; + + @Column(name = "creation_date") + private Timestamp creationDate; + + @Column(name = "revision_name") + private String revisionName; + + @Column(name = "revision_date") + private Timestamp revisionDate; + + @Column(name = "deal_name") + private String dealName; + + @Column(name = "deal_type") + private String dealType; + + @Column(name = "source_list_id") + private String sourceListId; + + @Column(name = "side") + private String side; +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/CurvePoint.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/CurvePoint.java index 151f80d02f..3ec6fbbdcf 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/CurvePoint.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/CurvePoint.java @@ -1,15 +1,44 @@ -package com.nnk.springboot.domain; - -import org.hibernate.validator.constraints.Length; - -import javax.persistence.*; -import javax.validation.constraints.NotBlank; -import javax.validation.constraints.NotNull; -import java.sql.Timestamp; - - -@Entity -@Table(name = "curvepoint") -public class CurvePoint { - // TODO: Map columns in data table CURVEPOINT with corresponding java fields -} +package com.nnk.springboot.domain; + +import jakarta.persistence.*; +import jakarta.validation.constraints.NotNull; +import lombok.Data; +import java.sql.Timestamp; + +@Data +@Entity +@Table(name = "curvepoint") +public class CurvePoint { + + @Id + @GeneratedValue(strategy = GenerationType.AUTO) + @Column(name = "id") + private int id; + + @NotNull(message = "Curve Id must not be null") + @Column(name = "curve_id") + private Integer curveId; + + @Column(name = "as_of_date") + private Timestamp asOfDate; + + @NotNull(message = "Term is mandatory") + @Column(name = "term") + private Double term; + + @NotNull(message = "Term is mandatory") + @Column(name = "value") + private Double value; + + @Column(name = "creation_date") + private Timestamp creationDate; + + + public Integer getCurveId() { + return curveId; + } + + public void setCurveId(Integer curveId) { + this.curveId = curveId; + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Rating.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Rating.java index 12d1be58c0..b8c32a9808 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Rating.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Rating.java @@ -1,12 +1,33 @@ -package com.nnk.springboot.domain; - -import javax.persistence.*; -import javax.validation.constraints.NotBlank; -import javax.validation.constraints.NotNull; -import java.sql.Timestamp; - -@Entity -@Table(name = "rating") -public class Rating { - // TODO: Map columns in data table RATING with corresponding java fields -} +package com.nnk.springboot.domain; + +import jakarta.persistence.*; +import jakarta.validation.constraints.NotBlank; +import jakarta.validation.constraints.NotNull; +import lombok.Data; + +@Data +@Entity +@Table(name = "rating") +public class Rating { + + @Id + @GeneratedValue(strategy = GenerationType.AUTO) + @Column(name = "id") + private int id; + + @NotBlank(message = "Moodys Rating is mandatory") + @Column(name = "moodys_rating") + private String moodysRating; + + @NotBlank(message = "Sand P Rating is mandatory") + @Column(name = "sand_p_rating") + private String sandPRating; + + @NotBlank(message = "Fitch Rating is mandatory") + @Column(name = "fitch_rating") + private String fitchRating; + + @NotNull(message = "Order Number is mandatory") + @Column(name = "order_number") + private Integer orderNumber; +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/RuleName.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/RuleName.java index b8ac970edf..4548f20a4c 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/RuleName.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/RuleName.java @@ -1,11 +1,40 @@ -package com.nnk.springboot.domain; - -import javax.persistence.*; -import javax.validation.constraints.NotBlank; -import java.sql.Timestamp; - -@Entity -@Table(name = "rulename") -public class RuleName { - // TODO: Map columns in data table RULENAME with corresponding java fields -} +package com.nnk.springboot.domain; + +import jakarta.persistence.*; +import jakarta.validation.constraints.NotBlank; +import lombok.Data; + +@Data +@Entity +@Table(name = "rulename") +public class RuleName { + + @Id + @GeneratedValue(strategy = GenerationType.AUTO) + @Column(name = "id") + private int id; + + @NotBlank(message = "Name is mandatory") + @Column(name = "name") + private String name; + + @NotBlank(message = "Description is mandatory") + @Column(name = "description") + private String description; + + @NotBlank(message = "Json is mandatory") + @Column(name = "json") + private String json; + + @NotBlank(message = "Template is mandatory") + @Column(name = "template") + private String template; + + @NotBlank(message = "SQL Str is mandatory") + @Column(name = "sql_str") + private String sqlStr; + + @NotBlank(message = "Sql Part is mandatory") + @Column(name = "sql_part") + private String sqlPart; +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Trade.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Trade.java index b6db7c13b7..0eb28aa385 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Trade.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/Trade.java @@ -1,12 +1,79 @@ -package com.nnk.springboot.domain; - -import javax.persistence.*; -import javax.validation.constraints.NotBlank; -import java.sql.Timestamp; - - -@Entity -@Table(name = "trade") -public class Trade { - // TODO: Map columns in data table TRADE with corresponding java fields -} +package com.nnk.springboot.domain; + +import jakarta.persistence.*; +import jakarta.validation.constraints.NotBlank; +import lombok.Data; +import java.sql.Timestamp; + +@Data +@Entity +@Table(name = "trade") +public class Trade { + + @Id + @GeneratedValue(strategy = GenerationType.AUTO) + @Column(name = "trade_id") + private int tradeId; + + @NotBlank(message = "Account is mandatory") + @Column(name = "account") + private String account; + + @NotBlank(message = "Type is mandatory") + @Column(name = "type") + private String type; + + @Column(name = "buy_quantity") + private Double buyQuantity; + + @Column(name = "sell_quantity") + private Double sellQuantity; + + @Column(name = "buy_price") + private Double buyPrice; + + @Column(name = "sell_price") + private Double sellPrice; + + @Column(name = "trade_date") + private Timestamp tradeDate; + + @Column(name = "security") + private String security; + + @Column(name = "status") + private String status; + + @Column(name = "trader") + private String trader; + + @Column(name = "benchmark") + private String benchmark; + + @Column(name = "book") + private String book; + + @Column(name = "creation_name") + private String creationName; + + @Column(name = "creation_date") + private Timestamp creationDate; + + @Column(name = "revision_name") + private String revisionName; + + @Column(name = "revision_date") + private Timestamp revisionDate; + + @Column(name = "deal_name") + private String dealName; + + @Column(name = "deal_type") + private String dealType; + + @Column(name = "source_list_id") + private String sourceListId; + + @Column(name = "side") + private String side; +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/User.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/User.java index 2be0b8c4ab..d2cef1cb5f 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/User.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/domain/User.java @@ -1,60 +1,35 @@ -package com.nnk.springboot.domain; - -import javax.persistence.*; -import javax.validation.constraints.NotBlank; - -@Entity -@Table(name = "users") -public class User { - @Id - @GeneratedValue(strategy= GenerationType.AUTO) - private Integer id; - @NotBlank(message = "Username is mandatory") - private String username; - @NotBlank(message = "Password is mandatory") - private String password; - @NotBlank(message = "FullName is mandatory") - private String fullname; - @NotBlank(message = "Role is mandatory") - private String role; - - public Integer getId() { - return id; - } - - public void setId(Integer id) { - this.id = id; - } - - public String getUsername() { - return username; - } - - public void setUsername(String username) { - this.username = username; - } - - public String getPassword() { - return password; - } - - public void setPassword(String password) { - this.password = password; - } - - public String getFullname() { - return fullname; - } - - public void setFullname(String fullname) { - this.fullname = fullname; - } - - public String getRole() { - return role; - } - - public void setRole(String role) { - this.role = role; - } -} +package com.nnk.springboot.domain; + + +import jakarta.persistence.*; +import jakarta.validation.constraints.NotBlank; +import jakarta.validation.constraints.Pattern; +import lombok.Data; + +@Data +@Entity +@Table(name = "users") +public class User { + @Id + @GeneratedValue(strategy= GenerationType.AUTO) + @Column(name = "id") + private Integer id; + + @NotBlank(message = "Username is mandatory") + @Column(name = "username", unique = true) + private String username; + + @Pattern(regexp = "^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[@#$%^&+=])(?=\\S+$).{8,}$", message = "The password must contain at least 8 characters, a capital letter, a number and a symbol") + @NotBlank(message = "Password is mandatory") + @Column(name = "password") + private String password; + + @NotBlank(message = "FullName is mandatory") + @Column(name = "fullname") + private String fullname; + + @NotBlank(message = "Role is mandatory") + @Column(name = "role") + @Pattern(regexp = "ROLE_ADMIN|ROLE_USER", message = "Role must be ROLE_ADMIN or ROLE_USER") + private String role; +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/BidListRepository.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/BidListRepository.java index f74b94e51d..139e7dcb4a 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/BidListRepository.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/BidListRepository.java @@ -1,9 +1,10 @@ -package com.nnk.springboot.repositories; - -import com.nnk.springboot.domain.BidList; -import org.springframework.data.jpa.repository.JpaRepository; - - -public interface BidListRepository extends JpaRepository { - -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.BidList; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface BidListRepository extends JpaRepository { + +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/CurvePointRepository.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/CurvePointRepository.java index b01751b53e..bf59795293 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/CurvePointRepository.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/CurvePointRepository.java @@ -1,9 +1,10 @@ -package com.nnk.springboot.repositories; - -import com.nnk.springboot.domain.CurvePoint; -import org.springframework.data.jpa.repository.JpaRepository; - - -public interface CurvePointRepository extends JpaRepository { - -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.CurvePoint; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface CurvePointRepository extends JpaRepository { + +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RatingRepository.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RatingRepository.java index 7ded405e6d..fb70fa464f 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RatingRepository.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RatingRepository.java @@ -1,8 +1,10 @@ -package com.nnk.springboot.repositories; - -import com.nnk.springboot.domain.Rating; -import org.springframework.data.jpa.repository.JpaRepository; - -public interface RatingRepository extends JpaRepository { - -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.Rating; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface RatingRepository extends JpaRepository { + +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RuleNameRepository.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RuleNameRepository.java index 8053d1612e..78b32ee435 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RuleNameRepository.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/RuleNameRepository.java @@ -1,8 +1,9 @@ -package com.nnk.springboot.repositories; - -import com.nnk.springboot.domain.RuleName; -import org.springframework.data.jpa.repository.JpaRepository; - - -public interface RuleNameRepository extends JpaRepository { -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.RuleName; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface RuleNameRepository extends JpaRepository { +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/TradeRepository.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/TradeRepository.java index e8da38af02..d96fc627eb 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/TradeRepository.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/TradeRepository.java @@ -1,8 +1,9 @@ -package com.nnk.springboot.repositories; - -import com.nnk.springboot.domain.Trade; -import org.springframework.data.jpa.repository.JpaRepository; - - -public interface TradeRepository extends JpaRepository { -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.Trade; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface TradeRepository extends JpaRepository { +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/UserRepository.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/UserRepository.java index b6a3363949..c49988ffb6 100644 --- a/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/UserRepository.java +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/repositories/UserRepository.java @@ -1,12 +1,13 @@ -package com.nnk.springboot.repositories; - -import com.nnk.springboot.domain.User; -import org.springframework.data.jpa.repository.JpaRepository; -import org.springframework.data.jpa.repository.JpaSpecificationExecutor; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.query.Param; - - -public interface UserRepository extends JpaRepository, JpaSpecificationExecutor { - -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.User; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.data.jpa.repository.JpaSpecificationExecutor; +import org.springframework.stereotype.Repository; + +import java.util.Optional; + +@Repository +public interface UserRepository extends JpaRepository, JpaSpecificationExecutor { + Optional findByUsername(String username); +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/BidListService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/BidListService.java new file mode 100644 index 0000000000..2501df75aa --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/BidListService.java @@ -0,0 +1,33 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.BidList; +import com.nnk.springboot.repositories.BidListRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.List; +import java.util.Optional; + +@Service +public class BidListService { + + @Autowired + private BidListRepository bidListRepository; + + public List getAllBidList() { + return bidListRepository.findAll(); + } + + public Optional getBidListById(Integer id){ + return bidListRepository.findById(id); + } + + public BidList saveBidList(BidList bidList){ + return bidListRepository.save(bidList); + } + + public void deleteBidListById(Integer id){ + bidListRepository.deleteById(id); + } + +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/CurvePointService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/CurvePointService.java new file mode 100644 index 0000000000..c1f5cb422f --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/CurvePointService.java @@ -0,0 +1,33 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.CurvePoint; +import com.nnk.springboot.repositories.CurvePointRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.List; +import java.util.Optional; + +@Service +public class CurvePointService { + + @Autowired + private CurvePointRepository curvePointRepository; + + public List getAllCurvePoint(){ + return curvePointRepository.findAll(); + } + + public Optional getCurvePointById(Integer id){ + return curvePointRepository.findById(id); + } + + public CurvePoint saveCurvePoint( CurvePoint curvePoint){ + return curvePointRepository.save(curvePoint); + } + + public void deleteCurvePointById(Integer id){ + curvePointRepository.deleteById(id); + } +} + diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/CustomUserDetailsService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/CustomUserDetailsService.java new file mode 100644 index 0000000000..142e1991d4 --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/CustomUserDetailsService.java @@ -0,0 +1,61 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.User; +import com.nnk.springboot.repositories.UserRepository; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.security.core.userdetails.UserDetailsService; +import org.springframework.security.core.userdetails.UsernameNotFoundException; +import org.springframework.stereotype.Service; +import java.util.List; +import org.springframework.security.core.authority.SimpleGrantedAuthority; + +/** + * Custom implementation of UserDetailsService for loading user details from the database. + */ +@Service +public class CustomUserDetailsService implements UserDetailsService { + + private final UserRepository userRepository; + private static final Logger logger = LoggerFactory.getLogger(CustomUserDetailsService.class); + + /** + * Constructor for initializing the user repository. + * + * @param userRepository the user repository used to fetch user details + */ + public CustomUserDetailsService(UserRepository userRepository) { + this.userRepository = userRepository; + } + + /** + * Loads a user by their username. + * + * @param username the username of the user to be loaded + * @return the user details associated with the given username + * @throws UsernameNotFoundException if the user is not found + */ + @Override + public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException { + logger.info("Attempting to load user: " + username); + + User user = userRepository.findByUsername(username) + .orElseThrow(() -> { + logger.error("User not found: " + username); + return new UsernameNotFoundException("User not found"); + }); + + logger.info("Found user: " + user.getUsername()); + logger.info("User role from DB: " + user.getRole()); + + SimpleGrantedAuthority authority = new SimpleGrantedAuthority(user.getRole()); + logger.info("Granted authority: " + authority.getAuthority()); + + return org.springframework.security.core.userdetails.User.builder() + .username(user.getUsername()) + .password(user.getPassword()) + .authorities(List.of(authority)) + .build(); + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/RatingService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/RatingService.java new file mode 100644 index 0000000000..ea16f00029 --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/RatingService.java @@ -0,0 +1,32 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.Rating; +import com.nnk.springboot.repositories.RatingRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.List; +import java.util.Optional; + +@Service +public class RatingService { + + @Autowired + private RatingRepository ratingRepository; + + public List getAllRating(){ + return ratingRepository.findAll(); + } + + public Optional getRatingById(Integer id){ + return ratingRepository.findById(id); + } + + public Rating saveRating( Rating rating){ + return ratingRepository.save(rating); + } + + public void deleteRatingById(Integer id){ + ratingRepository.deleteById(id); + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/RuleNameService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/RuleNameService.java new file mode 100644 index 0000000000..f0480eca42 --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/RuleNameService.java @@ -0,0 +1,32 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.RuleName; +import com.nnk.springboot.repositories.RuleNameRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.List; +import java.util.Optional; + +@Service +public class RuleNameService { + + @Autowired + private RuleNameRepository ruleNameRepository; + + public List getAllRuleNames(){ + return ruleNameRepository.findAll(); + } + + public Optional getRuleNameById(Integer id){ + return ruleNameRepository.findById(id); + } + + public RuleName saveRuleName(RuleName ruleName){ + return ruleNameRepository.save(ruleName); + } + + public void deleteRuleNameById(Integer id){ + ruleNameRepository.deleteById(id); + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/TradeService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/TradeService.java new file mode 100644 index 0000000000..083dc2b645 --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/TradeService.java @@ -0,0 +1,32 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.Trade; +import com.nnk.springboot.repositories.TradeRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.List; +import java.util.Optional; + +@Service +public class TradeService { + + @Autowired + private TradeRepository tradeRepository; + + public List getAllTrades(){ + return tradeRepository.findAll(); + } + + public Optional getTradeById(Integer id){ + return tradeRepository.findById(id); + } + + public Trade saveTrade( Trade trade){ + return tradeRepository.save(trade); + } + + public void deleteTradeById( Integer id){ + tradeRepository.deleteById(id); + } +} diff --git a/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/UserService.java b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/UserService.java new file mode 100644 index 0000000000..1d9ec474bf --- /dev/null +++ b/Poseiden-skeleton/src/main/java/com/nnk/springboot/services/UserService.java @@ -0,0 +1,38 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.User; +import com.nnk.springboot.repositories.UserRepository; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Bean; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import org.springframework.security.crypto.password.PasswordEncoder; +import org.springframework.stereotype.Service; +import java.util.List; +import java.util.Optional; + +@Service +public class UserService { + + @Autowired + private UserRepository userRepository; + + @Autowired + private BCryptPasswordEncoder passwordEncoder; + + public List getAllUsers() { + return userRepository.findAll(); + } + + public Optional getUserById(Integer id) { + return userRepository.findById(id); + } + + public User saveUser(User user) { + user.setPassword(passwordEncoder.encode(user.getPassword())); + return userRepository.save(user); + } + + public void deleteUserById(Integer id) { + userRepository.deleteById(id); + } +} diff --git a/Poseiden-skeleton/src/main/resources/application-prod.properties b/Poseiden-skeleton/src/main/resources/application-prod.properties index 83f1ee5b32..65ac04435a 100644 --- a/Poseiden-skeleton/src/main/resources/application-prod.properties +++ b/Poseiden-skeleton/src/main/resources/application-prod.properties @@ -1,16 +1,17 @@ - -logging.level.org.springframework=INFO - -################### DataSource Configuration ########################## -spring.datasource.driver-class-name=com.mysql.jdbc.Driver -spring.datasource.url=jdbc:mysql://localhost:3306/test -spring.datasource.username=root -spring.datasource.password=admin - -spring.datasource.initialize=true - -################### Hibernate Configuration ########################## - -spring.jpa.hibernate.ddl-auto=update -spring.jpa.show-sql=true - +################### Logging Configuration ########################## +logging.level.org.springframework=INFO +logging.level.org.springframework.security=DEBUG + +################### DataSource Configuration ########################## +spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver +spring.datasource.url=jdbc:mysql://localhost:3306/test +spring.datasource.username=DB_USERNAME +spring.datasource.password=DB_PASSWORD + +spring.datasource.initialize=true + +################### Hibernate Configuration ########################## +spring.jpa.database-platform=org.hibernate.dialect.MySQL8Dialect +spring.jpa.hibernate.ddl-auto=update +spring.jpa.show-sql=true + diff --git a/Poseiden-skeleton/src/main/resources/application.properties b/Poseiden-skeleton/src/main/resources/application.properties index 8b194cc76e..92dd8eae2a 100644 --- a/Poseiden-skeleton/src/main/resources/application.properties +++ b/Poseiden-skeleton/src/main/resources/application.properties @@ -1,14 +1,17 @@ - -logging.level.org.springframework=INFO - -################### DataSource Configuration ########################## -spring.datasource.driver-class-name=com.mysql.jdbc.Driver -spring.datasource.url=jdbc:mysql://localhost:3306/demo -spring.datasource.username=root -spring.datasource.password= - -################### Hibernate Configuration ########################## - -spring.jpa.hibernate.ddl-auto=update -spring.jpa.show-sql=true - +################### Logging Configuration ########################## +logging.level.org.springframework=INFO +logging.level.org.springframework.security=DEBUG +logging.level.org.thymeleaf=TRACE + +################### DataSource Configuration ########################## +spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver +spring.datasource.url=jdbc:mysql://localhost:3306/demo +spring.datasource.username=root +spring.datasource.password=98741 + +################### Hibernate Configuration ########################## +spring.jpa.database-platform=org.hibernate.dialect.MySQL8Dialect +spring.jpa.hibernate.ddl-auto=update +spring.jpa.show-sql=true + + diff --git a/Poseiden-skeleton/src/main/resources/static/images/logo.png b/Poseiden-skeleton/src/main/resources/static/images/logo.png new file mode 100644 index 0000000000..e69de29bb2 diff --git a/Poseiden-skeleton/src/main/resources/templates/403.html b/Poseiden-skeleton/src/main/resources/templates/403.html index c0c3740bd2..1dd3bf82ee 100644 --- a/Poseiden-skeleton/src/main/resources/templates/403.html +++ b/Poseiden-skeleton/src/main/resources/templates/403.html @@ -1,16 +1,26 @@ - - - - Spring Boot - - -

Access Denied Exception

-
- Logged in user: [[${#httpServletRequest.remoteUser}]] -
- -
-
-

Error

- - \ No newline at end of file + + + + + Access Denied + + + +
+
+
+

403 - Access Denied

+
+ Logged in user: +
+

+
+ +
+ Return to Home +
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/bidList/add.html b/Poseiden-skeleton/src/main/resources/templates/bidList/add.html index f7291e602e..30d1ef6cfb 100644 --- a/Poseiden-skeleton/src/main/resources/templates/bidList/add.html +++ b/Poseiden-skeleton/src/main/resources/templates/bidList/add.html @@ -1,52 +1,47 @@ - - - - -Home - - - -
- -
-

Add New Bid

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- Cancel - -
-
-
-
- -
- + + + + + Add New Bid + + + +
+
+

Add New Bid

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/bidList/list.html b/Poseiden-skeleton/src/main/resources/templates/bidList/list.html index 62b67e22fc..4dddc9f4cc 100644 --- a/Poseiden-skeleton/src/main/resources/templates/bidList/list.html +++ b/Poseiden-skeleton/src/main/resources/templates/bidList/list.html @@ -1,55 +1,70 @@ - - - - -Home - - - -
-
-
- Bid List |  - Curve Points |  - Ratings |  - Trade |  - Rule -
-
- Logged in user: [[${#httpServletRequest.remoteUser}]] -
- -
-
-
-

Bid List

-
- Add New - - - - - - - - - - - - - - - - - - - -
IdAccountTypeBid QuantityAction
- Edit |  - Delete -
-
-
- + + + + + Bid List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+
+ Logged in user: +
+ +
+
+
+ +
+
+

Bid List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + +
IdAccountTypeBid QuantityAction
+ Edit | + Delete +
+
+
+ +
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/bidList/update.html b/Poseiden-skeleton/src/main/resources/templates/bidList/update.html index b86c634d3a..eca073f584 100644 --- a/Poseiden-skeleton/src/main/resources/templates/bidList/update.html +++ b/Poseiden-skeleton/src/main/resources/templates/bidList/update.html @@ -1,52 +1,48 @@ - - - - -Home - - - -
-
-

Update Bid

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- - Cancel - -
-
-
-
- -
- + + + + + Update Bid + + + +
+
+

Update Bid

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/curvePoint/add.html b/Poseiden-skeleton/src/main/resources/templates/curvePoint/add.html index a46e2afb74..c5f040625c 100644 --- a/Poseiden-skeleton/src/main/resources/templates/curvePoint/add.html +++ b/Poseiden-skeleton/src/main/resources/templates/curvePoint/add.html @@ -1,44 +1,47 @@ - - - - -Home - - - -
- -
-

Add New Curve Point

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
- -
-
- Cancel - -
-
-
-
- -
- + + + + + Add New Curve Point + + + +
+
+

Add New Curve Point

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/curvePoint/list.html b/Poseiden-skeleton/src/main/resources/templates/curvePoint/list.html index 3d31e22395..66bbf930a0 100644 --- a/Poseiden-skeleton/src/main/resources/templates/curvePoint/list.html +++ b/Poseiden-skeleton/src/main/resources/templates/curvePoint/list.html @@ -1,55 +1,71 @@ - - - - -Home - - - -
-
-
- Bid List |  - Curve Points |  - Ratings |  - Trade |  - Rule -
-
- Logged in user: [[${#httpServletRequest.remoteUser}]] -
- -
-
-
-

Curve Point List

-
- Add New - - - - - - - - - - - - - - - - - - - -
IdCurvePointIdTermValueAction
- Edit |  - Delete -
-
-
- + + + + + Curve Point List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+ +
+ Logged in user: +
+ +
+
+
+ +
+
+

Curve Point List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + +
IdCurve IdTermValueAction
+ Edit | + Delete +
+
+
+ +
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/curvePoint/update.html b/Poseiden-skeleton/src/main/resources/templates/curvePoint/update.html index ee8c798f8b..53ac437f7d 100644 --- a/Poseiden-skeleton/src/main/resources/templates/curvePoint/update.html +++ b/Poseiden-skeleton/src/main/resources/templates/curvePoint/update.html @@ -1,44 +1,48 @@ - - - - -Home - - - -
-
-

Update CurvePoint

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
- -
-
- - Cancel - -
-
-
-
- -
- + + + + + Update Curve Point + + + +
+
+

Update Curve Point

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/error.html b/Poseiden-skeleton/src/main/resources/templates/error.html new file mode 100644 index 0000000000..5ca332c1fb --- /dev/null +++ b/Poseiden-skeleton/src/main/resources/templates/error.html @@ -0,0 +1,20 @@ + + + + + Error + + + +
+
+
+

Something went wrong.

+

An unexpected error occurred.

+ Return to Home +
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/home.html b/Poseiden-skeleton/src/main/resources/templates/home.html index 47dd44deba..5de0dfe660 100644 --- a/Poseiden-skeleton/src/main/resources/templates/home.html +++ b/Poseiden-skeleton/src/main/resources/templates/home.html @@ -1,13 +1,36 @@ - - - - Spring Boot - - - -

HOME PAGE

-

- Login or create one User management -

- + + + + + Home + + + +
+
+
+

Welcome to Poseidon Capital Solutions

+

Manage your bids, curve points, ratings, trades, and rules efficiently.

+
+
+ +
+ Logged in user: +
+ +
+
+ + +
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/login.html b/Poseiden-skeleton/src/main/resources/templates/login.html new file mode 100644 index 0000000000..dbccdef31c --- /dev/null +++ b/Poseiden-skeleton/src/main/resources/templates/login.html @@ -0,0 +1,39 @@ + + + + + Login + + + +
+
+
+

Login to Poseidon Capital Solutions

+

Please enter your credentials to access the platform.

+
+
+
+
+
+ Invalid username or password. +
+
+
+ + +
+
+ + +
+
+ +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/logout.html b/Poseiden-skeleton/src/main/resources/templates/logout.html new file mode 100644 index 0000000000..a9dc56b726 --- /dev/null +++ b/Poseiden-skeleton/src/main/resources/templates/logout.html @@ -0,0 +1,24 @@ + + + + + Logout + + + +
+
+
+

You have been logged out

+

Thank you for using Poseidon Capital Solutions. We hope to see you again soon!

+
+
+
+ +
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/rating/add.html b/Poseiden-skeleton/src/main/resources/templates/rating/add.html index 0d8fa765bf..d05f0b66ed 100644 --- a/Poseiden-skeleton/src/main/resources/templates/rating/add.html +++ b/Poseiden-skeleton/src/main/resources/templates/rating/add.html @@ -1,59 +1,54 @@ - - - - -Home - - - -
- -
-

Add New Rating

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- Cancel - -
-
-
-
- -
- + + + + + Add New Rating + + + +
+
+

Add New Rating

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/rating/list.html b/Poseiden-skeleton/src/main/resources/templates/rating/list.html index 020d6ea74d..b8fbb177bb 100644 --- a/Poseiden-skeleton/src/main/resources/templates/rating/list.html +++ b/Poseiden-skeleton/src/main/resources/templates/rating/list.html @@ -1,57 +1,72 @@ - - - - -Home - - - -
-
-
- Bid List |  - Curve Points |  - Ratings |  - Trade |  - Rule -
-
- Logged in user: [[${#httpServletRequest.remoteUser}]] -
- -
-
-
-

Rating List

-
- Add New - - - - - - - - - - - - - - - - - - - - - -
IdMoodysRatingSandPRatingFitchRatingOrderAction
- Edit |  - Delete -
-
-
- + + + + + Rating List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+
+ Logged in user: +
+ +
+
+
+ +
+
+

Rating List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + + + +
IdMoodys RatingS&P RatingFitch RatingOrderAction
+ Edit | + Delete +
+
+
+ +
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/rating/update.html b/Poseiden-skeleton/src/main/resources/templates/rating/update.html index 488ed5c7c7..df39afe226 100644 --- a/Poseiden-skeleton/src/main/resources/templates/rating/update.html +++ b/Poseiden-skeleton/src/main/resources/templates/rating/update.html @@ -1,60 +1,55 @@ - - - - - Home - - - -
- -
-

Update Rating

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- - Cancel - -
-
-
-
- -
- + + + + + Update Rating + + + +
+
+

Update Rating

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/ruleName/add.html b/Poseiden-skeleton/src/main/resources/templates/ruleName/add.html index 210b6338f0..d8fcb69933 100644 --- a/Poseiden-skeleton/src/main/resources/templates/ruleName/add.html +++ b/Poseiden-skeleton/src/main/resources/templates/ruleName/add.html @@ -1,73 +1,68 @@ - - - - -Home - - - -
- -
-

Add New Rule

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- Cancel - -
-
-
-
- -
- + + + + + Add New Rule + + + +
+
+

Add New Rule

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/ruleName/list.html b/Poseiden-skeleton/src/main/resources/templates/ruleName/list.html index 739ec39d01..0aafee3af2 100644 --- a/Poseiden-skeleton/src/main/resources/templates/ruleName/list.html +++ b/Poseiden-skeleton/src/main/resources/templates/ruleName/list.html @@ -1,61 +1,77 @@ - - - - -Home - - - -
-
-
- Bid List |  - Curve Points |  - ruleNames |  - Trade |  - Rule -
-
- Logged in user: [[${#httpServletRequest.remoteUser}]] -
- -
-
-
-

Rule List

-
- Add New - - - - - - - - - - - - - - - - - - - - - - - - - -
IdNameDescriptionjsontemplatesqlsqlPartAction
- Edit |  - Delete -
-
-
- + + + + + Rule List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+ +
+ Logged in user: +
+ +
+
+
+ +
+
+

Rule List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + +
IdNameDescriptionJsonTemplateSQLSQL PartAction
+ Edit | + Delete +
+
+
+ +
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/ruleName/update.html b/Poseiden-skeleton/src/main/resources/templates/ruleName/update.html index f5c4437f84..eeeb96e081 100644 --- a/Poseiden-skeleton/src/main/resources/templates/ruleName/update.html +++ b/Poseiden-skeleton/src/main/resources/templates/ruleName/update.html @@ -1,75 +1,69 @@ - - - - - Home - - - -
- -
-

Update New Rule

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- - Cancel - -
-
- -
-
- -
- + + + + + Update Rule + + + +
+
+

Update Rule

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/trade/add.html b/Poseiden-skeleton/src/main/resources/templates/trade/add.html index 061e2118ba..b5681f3adb 100644 --- a/Poseiden-skeleton/src/main/resources/templates/trade/add.html +++ b/Poseiden-skeleton/src/main/resources/templates/trade/add.html @@ -1,52 +1,47 @@ - - - - -Home - - - -
- -
-

Add New Trade

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- Cancel - -
-
-
-
- -
- + + + + + Add New Trade + + + +
+
+

Add New Trade

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/trade/list.html b/Poseiden-skeleton/src/main/resources/templates/trade/list.html index debf1afa68..c463394f44 100644 --- a/Poseiden-skeleton/src/main/resources/templates/trade/list.html +++ b/Poseiden-skeleton/src/main/resources/templates/trade/list.html @@ -1,55 +1,70 @@ - - - - -Home - - - -
-
-
- Bid List |  - Curve Points |  - Ratings |  - Trade |  - Rule -
-
- Logged in user: [[${#httpServletRequest.remoteUser}]] -
- -
-
-
-

Trade List

-
- Add New - - - - - - - - - - - - - - - - - - - -
IdAccountTypeBuy QuantityAction
- Edit |  - Delete -
-
-
- + + + + + Trade List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+
+ Logged in user: +
+ +
+
+
+ +
+
+

Trade List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + +
IdAccountTypeBuy QuantityAction
+ Edit | + Delete +
+
+
+ +
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/trade/update.html b/Poseiden-skeleton/src/main/resources/templates/trade/update.html index 7948653a14..7de635788a 100644 --- a/Poseiden-skeleton/src/main/resources/templates/trade/update.html +++ b/Poseiden-skeleton/src/main/resources/templates/trade/update.html @@ -1,52 +1,48 @@ - - - - -Home - - - -
-
-

Update Trade

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
- - -
-
- - Cancel - -
-
-
-
- -
- + + + + + Update Trade + + + +
+
+

Update Trade

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/user/add.html b/Poseiden-skeleton/src/main/resources/templates/user/add.html index 09246f49b2..a52c71c332 100644 --- a/Poseiden-skeleton/src/main/resources/templates/user/add.html +++ b/Poseiden-skeleton/src/main/resources/templates/user/add.html @@ -1,61 +1,55 @@ - - - - -Home - - - -
- -
-

Add New User

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- - -

-
-
- - -
-
- Cancel - -
-
- -
-
- -
- + + + + + Add New User + + + +
+
+

Add New User

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ + +

+
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/user/list.html b/Poseiden-skeleton/src/main/resources/templates/user/list.html index d239df6977..fc1076688c 100644 --- a/Poseiden-skeleton/src/main/resources/templates/user/list.html +++ b/Poseiden-skeleton/src/main/resources/templates/user/list.html @@ -1,48 +1,42 @@ - - - - -Home - - - -
-
-
- -
-
- Home | Login -
-
-

User List

-
- Add New - - - - - - - - - - - - - - - - - - - -
IdFull NameUser NameRoleAction
- Edit |  - Delete -
-
-
- + + + + + User List + + + +
+
+

User List

+
+
+ Add New + + + + + + + + + + + + + + + + + + + +
IdFull NameUser NameRoleAction
+ Edit |  + Delete +
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/main/resources/templates/user/update.html b/Poseiden-skeleton/src/main/resources/templates/user/update.html index 0322cfdf39..d81814e7c0 100644 --- a/Poseiden-skeleton/src/main/resources/templates/user/update.html +++ b/Poseiden-skeleton/src/main/resources/templates/user/update.html @@ -1,61 +1,55 @@ - - - - - Home - - - -
- -
-

Update User

-
- -
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- -

-
-
-
- -
- - -
-
- - -
-
- - Cancel - -
-
- -
-
- -
- + + + + + Update User + + + +
+
+

Update User

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ + +
+
+
+
+ Cancel + +
+
+
+
+
+ \ No newline at end of file diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/PasswordEncodeTest.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/PasswordEncodeTest.java deleted file mode 100644 index 4f831d9024..0000000000 --- a/Poseiden-skeleton/src/test/java/com/nnk/springboot/PasswordEncodeTest.java +++ /dev/null @@ -1,24 +0,0 @@ -package com.nnk.springboot; - -import org.junit.Test; -import org.junit.runner.RunWith; -import org.springframework.boot.test.context.SpringBootTest; -import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; -import org.springframework.test.context.junit4.SpringRunner; - -/** - * Created by Khang Nguyen. - * Email: khang.nguyen@banvien.com - * Date: 09/03/2019 - * Time: 11:26 AM - */ -@RunWith(SpringRunner.class) -@SpringBootTest -public class PasswordEncodeTest { - @Test - public void testPassword() { - BCryptPasswordEncoder encoder = new BCryptPasswordEncoder(); - String pw = encoder.encode("123456"); - System.out.println("[ "+ pw + " ]"); - } -} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/RatingTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/RatingTests.java deleted file mode 100644 index 6c3ebf0b08..0000000000 --- a/Poseiden-skeleton/src/test/java/com/nnk/springboot/RatingTests.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.nnk.springboot; - -import com.nnk.springboot.domain.Rating; -import com.nnk.springboot.repositories.RatingRepository; -import org.junit.Assert; -import org.junit.Test; -import org.junit.runner.RunWith; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.test.context.SpringBootTest; -import org.springframework.test.context.junit4.SpringRunner; - -import java.util.List; -import java.util.Optional; - -@RunWith(SpringRunner.class) -@SpringBootTest -public class RatingTests { - - @Autowired - private RatingRepository ratingRepository; - - @Test - public void ratingTest() { - Rating rating = new Rating("Moodys Rating", "Sand PRating", "Fitch Rating", 10); - - // Save - rating = ratingRepository.save(rating); - Assert.assertNotNull(rating.getId()); - Assert.assertTrue(rating.getOrderNumber() == 10); - - // Update - rating.setOrderNumber(20); - rating = ratingRepository.save(rating); - Assert.assertTrue(rating.getOrderNumber() == 20); - - // Find - List listResult = ratingRepository.findAll(); - Assert.assertTrue(listResult.size() > 0); - - // Delete - Integer id = rating.getId(); - ratingRepository.delete(rating); - Optional ratingList = ratingRepository.findById(id); - Assert.assertFalse(ratingList.isPresent()); - } -} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/RuleTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/RuleTests.java deleted file mode 100644 index 541dab5412..0000000000 --- a/Poseiden-skeleton/src/test/java/com/nnk/springboot/RuleTests.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.nnk.springboot; - -import com.nnk.springboot.domain.RuleName; -import com.nnk.springboot.repositories.RuleNameRepository; -import org.junit.Assert; -import org.junit.Test; -import org.junit.runner.RunWith; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.test.context.SpringBootTest; -import org.springframework.test.context.junit4.SpringRunner; - -import java.util.List; -import java.util.Optional; - -@RunWith(SpringRunner.class) -@SpringBootTest -public class RuleTests { - - @Autowired - private RuleNameRepository ruleNameRepository; - - @Test - public void ruleTest() { - RuleName rule = new RuleName("Rule Name", "Description", "Json", "Template", "SQL", "SQL Part"); - - // Save - rule = ruleNameRepository.save(rule); - Assert.assertNotNull(rule.getId()); - Assert.assertTrue(rule.getName().equals("Rule Name")); - - // Update - rule.setName("Rule Name Update"); - rule = ruleNameRepository.save(rule); - Assert.assertTrue(rule.getName().equals("Rule Name Update")); - - // Find - List listResult = ruleNameRepository.findAll(); - Assert.assertTrue(listResult.size() > 0); - - // Delete - Integer id = rule.getId(); - ruleNameRepository.delete(rule); - Optional ruleList = ruleNameRepository.findById(id); - Assert.assertFalse(ruleList.isPresent()); - } -} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/BidTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/BidTests.java similarity index 52% rename from Poseiden-skeleton/src/test/java/com/nnk/springboot/BidTests.java rename to Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/BidTests.java index f1a4f40316..7c385266fe 100644 --- a/Poseiden-skeleton/src/test/java/com/nnk/springboot/BidTests.java +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/BidTests.java @@ -1,46 +1,54 @@ -package com.nnk.springboot; - -import com.nnk.springboot.domain.BidList; -import com.nnk.springboot.repositories.BidListRepository; -import org.junit.Assert; -import org.junit.Test; -import org.junit.runner.RunWith; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.test.context.SpringBootTest; -import org.springframework.test.context.junit4.SpringRunner; - -import java.util.List; -import java.util.Optional; - -@RunWith(SpringRunner.class) -@SpringBootTest -public class BidTests { - - @Autowired - private BidListRepository bidListRepository; - - @Test - public void bidListTest() { - BidList bid = new BidList("Account Test", "Type Test", 10d); - - // Save - bid = bidListRepository.save(bid); - Assert.assertNotNull(bid.getBidListId()); - Assert.assertEquals(bid.getBidQuantity(), 10d, 10d); - - // Update - bid.setBidQuantity(20d); - bid = bidListRepository.save(bid); - Assert.assertEquals(bid.getBidQuantity(), 20d, 20d); - - // Find - List listResult = bidListRepository.findAll(); - Assert.assertTrue(listResult.size() > 0); - - // Delete - Integer id = bid.getBidListId(); - bidListRepository.delete(bid); - Optional bidList = bidListRepository.findById(id); - Assert.assertFalse(bidList.isPresent()); - } -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.BidList; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.junit4.SpringRunner; + +import java.util.List; +import java.util.Optional; + +@RunWith(SpringRunner.class) +@SpringBootTest +public class BidTests { + + @Autowired + private BidListRepository bidListRepository; + + @Test + public void bidListTest() { + BidList bid = new BidList(); + bid.setAccount("Account Test"); + bid.setType("Type Test"); + bid.setBidQuantity(10d); + + // Save + bid = bidListRepository.save(bid); + Integer generatedId = bid.getBidListId(); + + Assert.assertNotNull(generatedId); + Assert.assertTrue(generatedId > 0); + Assert.assertEquals(10d, bid.getBidQuantity(), 0.001); + + // Update + bid.setBidQuantity(20d); + BidList updatedBid = bidListRepository.save(bid); + Assert.assertEquals(20d, updatedBid.getBidQuantity(), 0.001); + + // Find + List listResult = bidListRepository.findAll(); + Assert.assertTrue(listResult.size() > 0); + + // FindById + Optional foundBid = bidListRepository.findById(generatedId); + Assert.assertTrue(foundBid.isPresent()); + + // Delete + bidListRepository.delete(bid); + Optional deletedBid = bidListRepository.findById(generatedId); + Assert.assertFalse(deletedBid.isPresent()); + } +} \ No newline at end of file diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/CurvePointTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/CurvePointTests.java similarity index 51% rename from Poseiden-skeleton/src/test/java/com/nnk/springboot/CurvePointTests.java rename to Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/CurvePointTests.java index 854615c0a0..5ad24006c5 100644 --- a/Poseiden-skeleton/src/test/java/com/nnk/springboot/CurvePointTests.java +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/CurvePointTests.java @@ -1,47 +1,55 @@ -package com.nnk.springboot; - -import com.nnk.springboot.domain.CurvePoint; -import com.nnk.springboot.repositories.CurvePointRepository; -import org.junit.Assert; -import org.junit.Test; -import org.junit.runner.RunWith; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.test.context.SpringBootTest; -import org.springframework.test.context.junit4.SpringRunner; - -import java.util.List; -import java.util.Optional; - -@RunWith(SpringRunner.class) -@SpringBootTest -public class CurvePointTests { - - @Autowired - private CurvePointRepository curvePointRepository; - - @Test - public void curvePointTest() { - CurvePoint curvePoint = new CurvePoint(10, 10d, 30d); - - // Save - curvePoint = curvePointRepository.save(curvePoint); - Assert.assertNotNull(curvePoint.getId()); - Assert.assertTrue(curvePoint.getCurveId() == 10); - - // Update - curvePoint.setCurveId(20); - curvePoint = curvePointRepository.save(curvePoint); - Assert.assertTrue(curvePoint.getCurveId() == 20); - - // Find - List listResult = curvePointRepository.findAll(); - Assert.assertTrue(listResult.size() > 0); - - // Delete - Integer id = curvePoint.getId(); - curvePointRepository.delete(curvePoint); - Optional curvePointList = curvePointRepository.findById(id); - Assert.assertFalse(curvePointList.isPresent()); - } - -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.CurvePoint; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.junit4.SpringRunner; + +import java.util.List; +import java.util.Optional; + +@RunWith(SpringRunner.class) +@SpringBootTest +public class CurvePointTests { + + @Autowired + private CurvePointRepository curvePointRepository; + + @Test + public void curvePointTest() { + CurvePoint curvePoint = new CurvePoint(); + curvePoint.setCurveId(10); + curvePoint.setTerm(10d); + curvePoint.setValue(30d); + + // Save + curvePoint = curvePointRepository.save(curvePoint); + Integer generatedId = curvePoint.getId(); + + Assert.assertNotNull(generatedId); + Assert.assertTrue(generatedId > 0); + Assert.assertEquals(Integer.valueOf(10), curvePoint.getCurveId()); + + // Update + curvePoint.setCurveId(20); + CurvePoint updatedCurvePoint = curvePointRepository.save(curvePoint); + Assert.assertEquals(Integer.valueOf(20), updatedCurvePoint.getCurveId()); + + // Find + List listResult = curvePointRepository.findAll(); + Assert.assertTrue(listResult.size() > 0); + + // FindById + Optional foundCurvePoint = curvePointRepository.findById(generatedId); + Assert.assertTrue(foundCurvePoint.isPresent()); + + // Delete + curvePointRepository.delete(curvePoint); + Optional deletedCurvePoint = curvePointRepository.findById(generatedId); + Assert.assertFalse(deletedCurvePoint.isPresent()); + } + +} \ No newline at end of file diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/RatingTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/RatingTests.java new file mode 100644 index 0000000000..d32728ed69 --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/RatingTests.java @@ -0,0 +1,55 @@ +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.Rating; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.junit4.SpringRunner; + +import java.util.List; +import java.util.Optional; + +@RunWith(SpringRunner.class) +@SpringBootTest +public class RatingTests { + + @Autowired + private RatingRepository ratingRepository; + + @Test + public void ratingTest() { + Rating rating = new Rating(); + rating.setOrderNumber(10); + rating.setMoodysRating("Moodys Rating"); + rating.setFitchRating("Fitch Rating"); + rating.setSandPRating("Sand PRating"); + + // Save + rating = ratingRepository.save(rating); + Integer generatedId = rating.getId(); + + Assert.assertNotNull(generatedId); + Assert.assertTrue(generatedId > 0); + Assert.assertEquals(Integer.valueOf(10), rating.getOrderNumber()); + + // Update + rating.setOrderNumber(20); + Rating updatedRating = ratingRepository.save(rating); + Assert.assertEquals(Integer.valueOf(20), updatedRating.getOrderNumber()); + + // Find + List listResult = ratingRepository.findAll(); + Assert.assertTrue(listResult.size() > 0); + + // FindById + Optional foundRating = ratingRepository.findById(generatedId); + Assert.assertTrue(foundRating.isPresent()); + + // Delete + ratingRepository.delete(rating); + Optional deletedRating = ratingRepository.findById(generatedId); + Assert.assertFalse(deletedRating.isPresent()); + } +} \ No newline at end of file diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/RuleTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/RuleTests.java new file mode 100644 index 0000000000..ef41b2eb5e --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/RuleTests.java @@ -0,0 +1,57 @@ +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.RuleName; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.junit4.SpringRunner; + +import java.util.List; +import java.util.Optional; + +@RunWith(SpringRunner.class) +@SpringBootTest +public class RuleTests { + + @Autowired + private RuleNameRepository ruleNameRepository; + + @Test + public void ruleTest() { + RuleName rule = new RuleName(); + rule.setName("Rule Name"); + rule.setDescription("Description"); + rule.setJson("Json"); + rule.setTemplate("Template"); + rule.setSqlStr("SQL"); + rule.setSqlPart("SQL Part"); + + // Save + rule = ruleNameRepository.save(rule); + Integer generatedId = rule.getId(); + + Assert.assertNotNull(generatedId); + Assert.assertTrue(generatedId > 0); + Assert.assertEquals("Rule Name", rule.getName()); + + // Update + rule.setName("Rule Name Update"); + RuleName updatedRule = ruleNameRepository.save(rule); + Assert.assertEquals("Rule Name Update", updatedRule.getName()); + + // Find + List listResult = ruleNameRepository.findAll(); + Assert.assertTrue(listResult.size() > 0); + + // FindById + Optional foundRule = ruleNameRepository.findById(generatedId); + Assert.assertTrue(foundRule.isPresent()); + + // Delete + ruleNameRepository.delete(rule); + Optional deletedRule = ruleNameRepository.findById(generatedId); + Assert.assertFalse(deletedRule.isPresent()); + } +} \ No newline at end of file diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/TradeTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/TradeTests.java similarity index 53% rename from Poseiden-skeleton/src/test/java/com/nnk/springboot/TradeTests.java rename to Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/TradeTests.java index ed50409266..6fae222401 100644 --- a/Poseiden-skeleton/src/test/java/com/nnk/springboot/TradeTests.java +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/TradeTests.java @@ -1,46 +1,53 @@ -package com.nnk.springboot; - -import com.nnk.springboot.domain.Trade; -import com.nnk.springboot.repositories.TradeRepository; -import org.junit.Assert; -import org.junit.Test; -import org.junit.runner.RunWith; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.test.context.SpringBootTest; -import org.springframework.test.context.junit4.SpringRunner; - -import java.util.List; -import java.util.Optional; - -@RunWith(SpringRunner.class) -@SpringBootTest -public class TradeTests { - - @Autowired - private TradeRepository tradeRepository; - - @Test - public void tradeTest() { - Trade trade = new Trade("Trade Account", "Type"); - - // Save - trade = tradeRepository.save(trade); - Assert.assertNotNull(trade.getTradeId()); - Assert.assertTrue(trade.getAccount().equals("Trade Account")); - - // Update - trade.setAccount("Trade Account Update"); - trade = tradeRepository.save(trade); - Assert.assertTrue(trade.getAccount().equals("Trade Account Update")); - - // Find - List listResult = tradeRepository.findAll(); - Assert.assertTrue(listResult.size() > 0); - - // Delete - Integer id = trade.getTradeId(); - tradeRepository.delete(trade); - Optional tradeList = tradeRepository.findById(id); - Assert.assertFalse(tradeList.isPresent()); - } -} +package com.nnk.springboot.repositories; + +import com.nnk.springboot.domain.Trade; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.junit4.SpringRunner; + +import java.util.List; +import java.util.Optional; + +@RunWith(SpringRunner.class) +@SpringBootTest +public class TradeTests { + + @Autowired + private TradeRepository tradeRepository; + + @Test + public void tradeTest() { + Trade trade = new Trade(); + trade.setAccount("Trade Account"); + trade.setType("Type"); + + // Save + trade = tradeRepository.save(trade); + Integer generatedId = trade.getTradeId(); + + Assert.assertNotNull(generatedId); + Assert.assertTrue(generatedId > 0); + Assert.assertEquals("Trade Account", trade.getAccount()); + + // Update + trade.setAccount("Trade Account Update"); + Trade updatedTrade = tradeRepository.save(trade); + Assert.assertEquals("Trade Account Update", updatedTrade.getAccount()); + + // Find + List listResult = tradeRepository.findAll(); + Assert.assertTrue(listResult.size() > 0); + + // FindById + Optional foundTrade = tradeRepository.findById(generatedId); + Assert.assertTrue(foundTrade.isPresent()); + + // Delete + tradeRepository.delete(trade); + Optional deletedTrade = tradeRepository.findById(generatedId); + Assert.assertFalse(deletedTrade.isPresent()); + } +} \ No newline at end of file diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/UserTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/UserTests.java new file mode 100644 index 0000000000..dcad92d058 --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/repositories/UserTests.java @@ -0,0 +1,57 @@ +package com.nnk.springboot.repositories; + + +import com.nnk.springboot.domain.User; +import org.junit.Assert; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.test.context.junit4.SpringRunner; + +import java.util.List; +import java.util.Optional; + + +@RunWith(SpringRunner.class) +@SpringBootTest +public class UserTests { + + @Autowired + private UserRepository userRepository; + + @Test + public void UserTest() { + User user = new User(); + user.setFullname("Super Admin"); + user.setRole("ROLE_ADMIN"); + user.setUsername("SuperAdmin"); + user.setPassword("SuperAdmin@2025"); + + // Save + user = userRepository.save(user); + Integer generatedId = user.getId(); + + Assert.assertNotNull(generatedId); + Assert.assertTrue(generatedId > 0); + Assert.assertEquals("SuperAdmin", user.getUsername()); + + // Update + user.setUsername("Admin"); + User updatedUser= userRepository.save(user); + Assert.assertEquals("Admin", updatedUser.getUsername()); + + // Find + List listUsers = userRepository.findAll(); + Assert.assertTrue(listUsers.size() > 0); + + // FindById + Optional foundUser = userRepository.findById(generatedId); + Assert.assertTrue(foundUser.isPresent()); + + // Delete + userRepository.delete(user); + Optional deletedUser = userRepository.findById(generatedId); + Assert.assertFalse(deletedUser.isPresent()); + } +} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/BidListServiceTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/BidListServiceTests.java new file mode 100644 index 0000000000..59858df46d --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/BidListServiceTests.java @@ -0,0 +1,71 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.BidList; +import com.nnk.springboot.repositories.BidListRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class BidListServiceTests { + + @Mock + private BidListRepository bidListRepository; + + @InjectMocks + private BidListService bidListService; + + private BidList bid; + + @BeforeEach + void setUp() { + bid = new BidList(); + bid.setBidListId(1); + bid.setAccount("Account Test"); + bid.setType("Type Test"); + bid.setBidQuantity(10d); + } + + @Test + void testGetAllBidList() { + when(bidListRepository.findAll()).thenReturn(Arrays.asList(bid)); + + List result = bidListService.getAllBidList(); + assertFalse(result.isEmpty()); + assertEquals(1, result.size()); + assertEquals("Account Test", result.get(0).getAccount()); + } + + @Test + void testGetBidListById() { + when(bidListRepository.findById(1)).thenReturn(Optional.of(bid)); + + Optional result = bidListService.getBidListById(1); + assertTrue(result.isPresent()); + assertEquals("Account Test", result.get().getAccount()); + } + + @Test + void testSaveBidList() { + when(bidListRepository.save(bid)).thenReturn(bid); + + BidList savedBid = bidListService.saveBidList(bid); + assertNotNull(savedBid); + assertEquals(10d, savedBid.getBidQuantity()); + } + + @Test + void testDeleteBidListById() { + doNothing().when(bidListRepository).deleteById(1); + bidListService.deleteBidListById(1); + verify(bidListRepository, times(1)).deleteById(1); + } +} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/CurvePointServiceTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/CurvePointServiceTests.java new file mode 100644 index 0000000000..daf1dc170f --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/CurvePointServiceTests.java @@ -0,0 +1,71 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.CurvePoint; +import com.nnk.springboot.repositories.CurvePointRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class CurvePointServiceTests { + + @Mock + private CurvePointRepository curvePointRepository; + + @InjectMocks + private CurvePointService curvePointService; + + private CurvePoint curvePoint; + + @BeforeEach + void setUp() { + curvePoint = new CurvePoint(); + curvePoint.setId(1); + curvePoint.setCurveId(10); + curvePoint.setTerm(10d); + curvePoint.setValue(30d); + } + + @Test + void testGetAllCurvePoint() { + when(curvePointRepository.findAll()).thenReturn(Arrays.asList(curvePoint)); + + List result = curvePointService.getAllCurvePoint(); + assertFalse(result.isEmpty()); + assertEquals(1, result.size()); + assertEquals(10, result.get(0).getCurveId()); + } + + @Test + void testGetCurvePointById() { + when(curvePointRepository.findById(1)).thenReturn(Optional.of(curvePoint)); + + Optional result = curvePointService.getCurvePointById(1); + assertTrue(result.isPresent()); + assertEquals(10, result.get().getCurveId()); + } + + @Test + void testSaveCurvePoint() { + when(curvePointRepository.save(curvePoint)).thenReturn(curvePoint); + + CurvePoint savedCurvePoint = curvePointService.saveCurvePoint(curvePoint); + assertNotNull(savedCurvePoint); + assertEquals(30d, savedCurvePoint.getValue()); + } + + @Test + void testDeleteCurvePointById() { + doNothing().when(curvePointRepository).deleteById(1); + curvePointService.deleteCurvePointById(1); + verify(curvePointRepository, times(1)).deleteById(1); + } +} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/RatingServiceTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/RatingServiceTests.java new file mode 100644 index 0000000000..478fe6e1c8 --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/RatingServiceTests.java @@ -0,0 +1,72 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.Rating; +import com.nnk.springboot.repositories.RatingRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class RatingServiceTests { + + @Mock + private RatingRepository ratingRepository; + + @InjectMocks + private RatingService ratingService; + + private Rating rating; + + @BeforeEach + void setUp() { + rating = new Rating(); + rating.setId(1); + rating.setOrderNumber(10); + rating.setMoodysRating("Moodys Rating"); + rating.setFitchRating("Fitch Rating"); + rating.setSandPRating("Sand PRating"); + } + + @Test + void testGetAllRating() { + when(ratingRepository.findAll()).thenReturn(Arrays.asList(rating)); + + List result = ratingService.getAllRating(); + assertFalse(result.isEmpty()); + assertEquals(1, result.size()); + assertEquals("Moodys Rating", result.get(0).getMoodysRating()); + } + + @Test + void testGetRatingById() { + when(ratingRepository.findById(1)).thenReturn(Optional.of(rating)); + + Optional result = ratingService.getRatingById(1); + assertTrue(result.isPresent()); + assertEquals(10, result.get().getOrderNumber()); + } + + @Test + void testSaveRating() { + when(ratingRepository.save(rating)).thenReturn(rating); + + Rating savedRating = ratingService.saveRating(rating); + assertNotNull(savedRating); + assertEquals("Fitch Rating", savedRating.getFitchRating()); + } + + @Test + void testDeleteRatingById() { + doNothing().when(ratingRepository).deleteById(1); + ratingService.deleteRatingById(1); + verify(ratingRepository, times(1)).deleteById(1); + } +} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/RuleNameServiceTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/RuleNameServiceTests.java new file mode 100644 index 0000000000..9cb7aef0e0 --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/RuleNameServiceTests.java @@ -0,0 +1,74 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.RuleName; +import com.nnk.springboot.repositories.RuleNameRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class RuleNameServiceTests { + + @Mock + private RuleNameRepository ruleNameRepository; + + @InjectMocks + private RuleNameService ruleNameService; + + private RuleName ruleName; + + @BeforeEach + void setUp() { + ruleName = new RuleName(); + ruleName.setId(1); + ruleName.setName("Rule Name"); + ruleName.setDescription("Description"); + ruleName.setJson("Json"); + ruleName.setTemplate("Template"); + ruleName.setSqlStr("SQL"); + ruleName.setSqlPart("SQL Part"); + } + + @Test + void testGetAllRuleNames() { + when(ruleNameRepository.findAll()).thenReturn(Arrays.asList(ruleName)); + + List result = ruleNameService.getAllRuleNames(); + assertFalse(result.isEmpty()); + assertEquals(1, result.size()); + assertEquals("Rule Name", result.get(0).getName()); + } + + @Test + void testGetRuleNameById() { + when(ruleNameRepository.findById(1)).thenReturn(Optional.of(ruleName)); + + Optional result = ruleNameService.getRuleNameById(1); + assertTrue(result.isPresent()); + assertEquals("Rule Name", result.get().getName()); + } + + @Test + void testSaveRuleName() { + when(ruleNameRepository.save(ruleName)).thenReturn(ruleName); + + RuleName savedRuleName = ruleNameService.saveRuleName(ruleName); + assertNotNull(savedRuleName); + assertEquals("Description", savedRuleName.getDescription()); + } + + @Test + void testDeleteRuleNameById() { + doNothing().when(ruleNameRepository).deleteById(1); + ruleNameService.deleteRuleNameById(1); + verify(ruleNameRepository, times(1)).deleteById(1); + } +} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/TradeServiceTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/TradeServiceTests.java new file mode 100644 index 0000000000..280ccb6201 --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/TradeServiceTests.java @@ -0,0 +1,70 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.Trade; +import com.nnk.springboot.repositories.TradeRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.junit.jupiter.MockitoExtension; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +@ExtendWith(MockitoExtension.class) +public class TradeServiceTests { + + @Mock + private TradeRepository tradeRepository; + + @InjectMocks + private TradeService tradeService; + + private Trade trade; + + @BeforeEach + void setUp() { + trade = new Trade(); + trade.setTradeId(1); + trade.setAccount("Trade Account"); + trade.setType("Type"); + } + + @Test + void testGetAllTrades() { + when(tradeRepository.findAll()).thenReturn(Arrays.asList(trade)); + + List result = tradeService.getAllTrades(); + assertFalse(result.isEmpty()); + assertEquals(1, result.size()); + assertEquals("Trade Account", result.get(0).getAccount()); + } + + @Test + void testGetTradeById() { + when(tradeRepository.findById(1)).thenReturn(Optional.of(trade)); + + Optional result = tradeService.getTradeById(1); + assertTrue(result.isPresent()); + assertEquals("Trade Account", result.get().getAccount()); + } + + @Test + void testSaveTrade() { + when(tradeRepository.save(trade)).thenReturn(trade); + + Trade savedTrade = tradeService.saveTrade(trade); + assertNotNull(savedTrade); + assertEquals("Type", savedTrade.getType()); + } + + @Test + void testDeleteTradeById() { + doNothing().when(tradeRepository).deleteById(1); + tradeService.deleteTradeById(1); + verify(tradeRepository, times(1)).deleteById(1); + } +} diff --git a/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/UserServiceTests.java b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/UserServiceTests.java new file mode 100644 index 0000000000..3224d55b04 --- /dev/null +++ b/Poseiden-skeleton/src/test/java/com/nnk/springboot/services/UserServiceTests.java @@ -0,0 +1,93 @@ +package com.nnk.springboot.services; + +import com.nnk.springboot.domain.User; +import com.nnk.springboot.repositories.UserRepository; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.mockito.InjectMocks; +import org.mockito.Mock; +import org.mockito.MockitoAnnotations; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; +import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.Mockito.*; + +class UserServiceTests { + + @InjectMocks + private UserService userService; + + @Mock + private UserRepository userRepository; + + @Mock + private BCryptPasswordEncoder passwordEncoder; + + @BeforeEach + void setUp() { + MockitoAnnotations.openMocks(this); + } + + @Test + void getAllUsers_shouldReturnListOfUsers() { + List users = Arrays.asList(new User(), new User()); + when(userRepository.findAll()).thenReturn(users); + + List result = userService.getAllUsers(); + + assertEquals(2, result.size()); + verify(userRepository, times(1)).findAll(); + } + + @Test + void getUserById_shouldReturnUser_whenIdExists() { + User user = new User(); + user.setId(1); + when(userRepository.findById(1)).thenReturn(Optional.of(user)); + + Optional result = userService.getUserById(1); + + assertTrue(result.isPresent()); + assertEquals(1, result.get().getId()); + } + + @Test + void saveUser_shouldEncodePasswordAndSaveUser() { + User user = new User(); + user.setPassword("rawPassword"); + when(passwordEncoder.encode("rawPassword")).thenReturn("encodedPassword"); + when(userRepository.save(user)).thenReturn(user); + + User result = userService.saveUser(user); + + assertEquals("encodedPassword", result.getPassword()); + verify(userRepository, times(1)).save(user); + } + + @Test + void updateUser_shouldEncodePasswordAndUpdateUser() { + User user = new User(); + user.setId(1); + user.setPassword("newPassword"); + when(passwordEncoder.encode("newPassword")).thenReturn("encodedNewPassword"); + when(userRepository.save(user)).thenReturn(user); + + user.setId(1); + User result = userService.saveUser(user); + + assertEquals(1, result.getId()); + assertEquals("encodedNewPassword", result.getPassword()); + verify(userRepository, times(1)).save(user); + } + + @Test + void deleteUserById_shouldDeleteUser_whenIdExists() { + doNothing().when(userRepository).deleteById(1); + + userService.deleteUserById(1); + + verify(userRepository, times(1)).deleteById(1); + } +} diff --git a/Poseiden-skeleton/target/classes/application-prod.properties b/Poseiden-skeleton/target/classes/application-prod.properties new file mode 100644 index 0000000000..65ac04435a --- /dev/null +++ b/Poseiden-skeleton/target/classes/application-prod.properties @@ -0,0 +1,17 @@ +################### Logging Configuration ########################## +logging.level.org.springframework=INFO +logging.level.org.springframework.security=DEBUG + +################### DataSource Configuration ########################## +spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver +spring.datasource.url=jdbc:mysql://localhost:3306/test +spring.datasource.username=DB_USERNAME +spring.datasource.password=DB_PASSWORD + +spring.datasource.initialize=true + +################### Hibernate Configuration ########################## +spring.jpa.database-platform=org.hibernate.dialect.MySQL8Dialect +spring.jpa.hibernate.ddl-auto=update +spring.jpa.show-sql=true + diff --git a/Poseiden-skeleton/target/classes/application.properties b/Poseiden-skeleton/target/classes/application.properties new file mode 100644 index 0000000000..92dd8eae2a --- /dev/null +++ b/Poseiden-skeleton/target/classes/application.properties @@ -0,0 +1,17 @@ +################### Logging Configuration ########################## +logging.level.org.springframework=INFO +logging.level.org.springframework.security=DEBUG +logging.level.org.thymeleaf=TRACE + +################### DataSource Configuration ########################## +spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver +spring.datasource.url=jdbc:mysql://localhost:3306/demo +spring.datasource.username=root +spring.datasource.password=98741 + +################### Hibernate Configuration ########################## +spring.jpa.database-platform=org.hibernate.dialect.MySQL8Dialect +spring.jpa.hibernate.ddl-auto=update +spring.jpa.show-sql=true + + diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/Application.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/Application.class new file mode 100644 index 0000000000..0221aff8b2 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/Application.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/config/SecurityConfig.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/config/SecurityConfig.class new file mode 100644 index 0000000000..acf4e0ec5b Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/config/SecurityConfig.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/BidListController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/BidListController.class new file mode 100644 index 0000000000..4f8b187003 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/BidListController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/CurvePointController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/CurvePointController.class new file mode 100644 index 0000000000..159a591843 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/CurvePointController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/HomeController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/HomeController.class new file mode 100644 index 0000000000..7f7d4783bc Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/HomeController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/LoginController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/LoginController.class new file mode 100644 index 0000000000..4fccac4d5b Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/LoginController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/LogoutController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/LogoutController.class new file mode 100644 index 0000000000..a979e23fab Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/LogoutController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/RatingController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/RatingController.class new file mode 100644 index 0000000000..a615908dbf Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/RatingController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/RuleNameController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/RuleNameController.class new file mode 100644 index 0000000000..49eb3bcf85 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/RuleNameController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/TradeController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/TradeController.class new file mode 100644 index 0000000000..f051c0d809 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/TradeController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/UserController.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/UserController.class new file mode 100644 index 0000000000..d50b5c95ff Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/controllers/UserController.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/BidList.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/BidList.class new file mode 100644 index 0000000000..6f6420d3c4 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/BidList.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/CurvePoint.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/CurvePoint.class new file mode 100644 index 0000000000..3bfb4b5f10 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/CurvePoint.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/Rating.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/Rating.class new file mode 100644 index 0000000000..ea4486ca6a Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/Rating.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/RuleName.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/RuleName.class new file mode 100644 index 0000000000..c8e0aa2628 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/RuleName.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/Trade.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/Trade.class new file mode 100644 index 0000000000..3bd91eca7f Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/Trade.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/User.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/User.class new file mode 100644 index 0000000000..a38d059f59 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/domain/User.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/BidListRepository.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/BidListRepository.class new file mode 100644 index 0000000000..c3f159c2ae Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/BidListRepository.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/CurvePointRepository.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/CurvePointRepository.class new file mode 100644 index 0000000000..2738594974 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/CurvePointRepository.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/RatingRepository.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/RatingRepository.class new file mode 100644 index 0000000000..5cb5ec7a77 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/RatingRepository.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/RuleNameRepository.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/RuleNameRepository.class new file mode 100644 index 0000000000..a9d0afef0d Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/RuleNameRepository.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/TradeRepository.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/TradeRepository.class new file mode 100644 index 0000000000..ba2cb63c1b Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/TradeRepository.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/UserRepository.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/UserRepository.class new file mode 100644 index 0000000000..81b28aaf7a Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/repositories/UserRepository.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/BidListService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/BidListService.class new file mode 100644 index 0000000000..f78a046351 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/BidListService.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/CurvePointService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/CurvePointService.class new file mode 100644 index 0000000000..a300223d11 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/CurvePointService.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/CustomUserDetailsService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/CustomUserDetailsService.class new file mode 100644 index 0000000000..0115a7c481 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/CustomUserDetailsService.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/RatingService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/RatingService.class new file mode 100644 index 0000000000..6f5b95ca2d Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/RatingService.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/RuleNameService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/RuleNameService.class new file mode 100644 index 0000000000..6d7824bbb7 Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/RuleNameService.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/TradeService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/TradeService.class new file mode 100644 index 0000000000..9881ca625f Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/TradeService.class differ diff --git a/Poseiden-skeleton/target/classes/com/nnk/springboot/services/UserService.class b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/UserService.class new file mode 100644 index 0000000000..2e3484148b Binary files /dev/null and b/Poseiden-skeleton/target/classes/com/nnk/springboot/services/UserService.class differ diff --git a/Poseiden-skeleton/target/classes/messages.properties b/Poseiden-skeleton/target/classes/messages.properties new file mode 100644 index 0000000000..5a5e3cc198 --- /dev/null +++ b/Poseiden-skeleton/target/classes/messages.properties @@ -0,0 +1 @@ +app.title=Simple App - Demo \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/static/css/bootstrap.min.css b/Poseiden-skeleton/target/classes/static/css/bootstrap.min.css new file mode 100644 index 0000000000..92e3fe8712 --- /dev/null +++ b/Poseiden-skeleton/target/classes/static/css/bootstrap.min.css @@ -0,0 +1,7 @@ +/*! + * Bootstrap v4.3.1 (https://getbootstrap.com/) + * Copyright 2011-2019 The Bootstrap Authors + * Copyright 2011-2019 Twitter, Inc. + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) + */:root{--blue:#007bff;--indigo:#6610f2;--purple:#6f42c1;--pink:#e83e8c;--red:#dc3545;--orange:#fd7e14;--yellow:#ffc107;--green:#28a745;--teal:#20c997;--cyan:#17a2b8;--white:#fff;--gray:#6c757d;--gray-dark:#343a40;--primary:#007bff;--secondary:#6c757d;--success:#28a745;--info:#17a2b8;--warning:#ffc107;--danger:#dc3545;--light:#f8f9fa;--dark:#343a40;--breakpoint-xs:0;--breakpoint-sm:576px;--breakpoint-md:768px;--breakpoint-lg:992px;--breakpoint-xl:1200px;--font-family-sans-serif:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--font-family-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace}*,::after,::before{box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}article,aside,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-size:1rem;font-weight:400;line-height:1.5;color:#212529;text-align:left;background-color:#fff}[tabindex="-1"]:focus{outline:0!important}hr{box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem}p{margin-top:0;margin-bottom:1rem}abbr[data-original-title],abbr[title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;border-bottom:0;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#007bff;text-decoration:none;background-color:transparent}a:hover{color:#0056b3;text-decoration:underline}a:not([href]):not([tabindex]){color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus,a:not([href]):not([tabindex]):hover{color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus{outline:0}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em}pre{margin-top:0;margin-bottom:1rem;overflow:auto}figure{margin:0 0 1rem}img{vertical-align:middle;border-style:none}svg{overflow:hidden;vertical-align:middle}table{border-collapse:collapse}caption{padding-top:.75rem;padding-bottom:.75rem;color:#6c757d;text-align:left;caption-side:bottom}th{text-align:inherit}label{display:inline-block;margin-bottom:.5rem}button{border-radius:0}button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}select{word-wrap:normal}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{padding:0;border-style:none}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=date],input[type=datetime-local],input[type=month],input[type=time]{-webkit-appearance:listbox}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;max-width:100%;padding:0;margin-bottom:.5rem;font-size:1.5rem;line-height:inherit;color:inherit;white-space:normal}progress{vertical-align:baseline}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:none}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item;cursor:pointer}template{display:none}[hidden]{display:none!important}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-bottom:.5rem;font-weight:500;line-height:1.2}.h1,h1{font-size:2.5rem}.h2,h2{font-size:2rem}.h3,h3{font-size:1.75rem}.h4,h4{font-size:1.5rem}.h5,h5{font-size:1.25rem}.h6,h6{font-size:1rem}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:6rem;font-weight:300;line-height:1.2}.display-2{font-size:5.5rem;font-weight:300;line-height:1.2}.display-3{font-size:4.5rem;font-weight:300;line-height:1.2}.display-4{font-size:3.5rem;font-weight:300;line-height:1.2}hr{margin-top:1rem;margin-bottom:1rem;border:0;border-top:1px solid rgba(0,0,0,.1)}.small,small{font-size:80%;font-weight:400}.mark,mark{padding:.2em;background-color:#fcf8e3}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:90%;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote-footer{display:block;font-size:80%;color:#6c757d}.blockquote-footer::before{content:"\2014\00A0"}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:#fff;border:1px solid #dee2e6;border-radius:.25rem;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:90%;color:#6c757d}code{font-size:87.5%;color:#e83e8c;word-break:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:87.5%;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:100%;font-weight:700}pre{display:block;font-size:87.5%;color:#212529}pre code{font-size:inherit;color:inherit;word-break:normal}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:576px){.container{max-width:540px}}@media (min-width:768px){.container{max-width:720px}}@media (min-width:992px){.container{max-width:960px}}@media (min-width:1200px){.container{max-width:1140px}}.container-fluid{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-15px;margin-left:-15px}.no-gutters{margin-right:0;margin-left:0}.no-gutters>.col,.no-gutters>[class*=col-]{padding-right:0;padding-left:0}.col,.col-1,.col-10,.col-11,.col-12,.col-2,.col-3,.col-4,.col-5,.col-6,.col-7,.col-8,.col-9,.col-auto,.col-lg,.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-auto,.col-md,.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-auto,.col-sm,.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-auto,.col-xl,.col-xl-1,.col-xl-10,.col-xl-11,.col-xl-12,.col-xl-2,.col-xl-3,.col-xl-4,.col-xl-5,.col-xl-6,.col-xl-7,.col-xl-8,.col-xl-9,.col-xl-auto{position:relative;width:100%;padding-right:15px;padding-left:15px}.col{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-first{-ms-flex-order:-1;order:-1}.order-last{-ms-flex-order:13;order:13}.order-0{-ms-flex-order:0;order:0}.order-1{-ms-flex-order:1;order:1}.order-2{-ms-flex-order:2;order:2}.order-3{-ms-flex-order:3;order:3}.order-4{-ms-flex-order:4;order:4}.order-5{-ms-flex-order:5;order:5}.order-6{-ms-flex-order:6;order:6}.order-7{-ms-flex-order:7;order:7}.order-8{-ms-flex-order:8;order:8}.order-9{-ms-flex-order:9;order:9}.order-10{-ms-flex-order:10;order:10}.order-11{-ms-flex-order:11;order:11}.order-12{-ms-flex-order:12;order:12}.offset-1{margin-left:8.333333%}.offset-2{margin-left:16.666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.333333%}.offset-5{margin-left:41.666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.333333%}.offset-8{margin-left:66.666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.333333%}.offset-11{margin-left:91.666667%}@media (min-width:576px){.col-sm{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-sm-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-sm-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-sm-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-sm-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-sm-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-sm-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-sm-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-sm-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-sm-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-sm-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-sm-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-sm-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-sm-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-sm-first{-ms-flex-order:-1;order:-1}.order-sm-last{-ms-flex-order:13;order:13}.order-sm-0{-ms-flex-order:0;order:0}.order-sm-1{-ms-flex-order:1;order:1}.order-sm-2{-ms-flex-order:2;order:2}.order-sm-3{-ms-flex-order:3;order:3}.order-sm-4{-ms-flex-order:4;order:4}.order-sm-5{-ms-flex-order:5;order:5}.order-sm-6{-ms-flex-order:6;order:6}.order-sm-7{-ms-flex-order:7;order:7}.order-sm-8{-ms-flex-order:8;order:8}.order-sm-9{-ms-flex-order:9;order:9}.order-sm-10{-ms-flex-order:10;order:10}.order-sm-11{-ms-flex-order:11;order:11}.order-sm-12{-ms-flex-order:12;order:12}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.333333%}.offset-sm-2{margin-left:16.666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.333333%}.offset-sm-5{margin-left:41.666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.333333%}.offset-sm-8{margin-left:66.666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.333333%}.offset-sm-11{margin-left:91.666667%}}@media (min-width:768px){.col-md{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-md-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-md-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-md-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-md-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-md-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-md-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-md-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-md-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-md-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-md-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-md-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-md-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-md-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-md-first{-ms-flex-order:-1;order:-1}.order-md-last{-ms-flex-order:13;order:13}.order-md-0{-ms-flex-order:0;order:0}.order-md-1{-ms-flex-order:1;order:1}.order-md-2{-ms-flex-order:2;order:2}.order-md-3{-ms-flex-order:3;order:3}.order-md-4{-ms-flex-order:4;order:4}.order-md-5{-ms-flex-order:5;order:5}.order-md-6{-ms-flex-order:6;order:6}.order-md-7{-ms-flex-order:7;order:7}.order-md-8{-ms-flex-order:8;order:8}.order-md-9{-ms-flex-order:9;order:9}.order-md-10{-ms-flex-order:10;order:10}.order-md-11{-ms-flex-order:11;order:11}.order-md-12{-ms-flex-order:12;order:12}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.333333%}.offset-md-2{margin-left:16.666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.333333%}.offset-md-5{margin-left:41.666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.333333%}.offset-md-8{margin-left:66.666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.333333%}.offset-md-11{margin-left:91.666667%}}@media (min-width:992px){.col-lg{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-lg-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-lg-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-lg-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-lg-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-lg-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-lg-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-lg-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-lg-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-lg-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-lg-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-lg-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-lg-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-lg-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-lg-first{-ms-flex-order:-1;order:-1}.order-lg-last{-ms-flex-order:13;order:13}.order-lg-0{-ms-flex-order:0;order:0}.order-lg-1{-ms-flex-order:1;order:1}.order-lg-2{-ms-flex-order:2;order:2}.order-lg-3{-ms-flex-order:3;order:3}.order-lg-4{-ms-flex-order:4;order:4}.order-lg-5{-ms-flex-order:5;order:5}.order-lg-6{-ms-flex-order:6;order:6}.order-lg-7{-ms-flex-order:7;order:7}.order-lg-8{-ms-flex-order:8;order:8}.order-lg-9{-ms-flex-order:9;order:9}.order-lg-10{-ms-flex-order:10;order:10}.order-lg-11{-ms-flex-order:11;order:11}.order-lg-12{-ms-flex-order:12;order:12}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.333333%}.offset-lg-2{margin-left:16.666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.333333%}.offset-lg-5{margin-left:41.666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.333333%}.offset-lg-8{margin-left:66.666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.333333%}.offset-lg-11{margin-left:91.666667%}}@media (min-width:1200px){.col-xl{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-xl-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-xl-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-xl-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-xl-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-xl-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-xl-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-xl-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-xl-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-xl-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-xl-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-xl-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-xl-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-xl-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-xl-first{-ms-flex-order:-1;order:-1}.order-xl-last{-ms-flex-order:13;order:13}.order-xl-0{-ms-flex-order:0;order:0}.order-xl-1{-ms-flex-order:1;order:1}.order-xl-2{-ms-flex-order:2;order:2}.order-xl-3{-ms-flex-order:3;order:3}.order-xl-4{-ms-flex-order:4;order:4}.order-xl-5{-ms-flex-order:5;order:5}.order-xl-6{-ms-flex-order:6;order:6}.order-xl-7{-ms-flex-order:7;order:7}.order-xl-8{-ms-flex-order:8;order:8}.order-xl-9{-ms-flex-order:9;order:9}.order-xl-10{-ms-flex-order:10;order:10}.order-xl-11{-ms-flex-order:11;order:11}.order-xl-12{-ms-flex-order:12;order:12}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.333333%}.offset-xl-2{margin-left:16.666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.333333%}.offset-xl-5{margin-left:41.666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.333333%}.offset-xl-8{margin-left:66.666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.333333%}.offset-xl-11{margin-left:91.666667%}}.table{width:100%;margin-bottom:1rem;color:#212529}.table td,.table th{padding:.75rem;vertical-align:top;border-top:1px solid #dee2e6}.table thead th{vertical-align:bottom;border-bottom:2px solid #dee2e6}.table tbody+tbody{border-top:2px solid #dee2e6}.table-sm td,.table-sm th{padding:.3rem}.table-bordered{border:1px solid #dee2e6}.table-bordered td,.table-bordered th{border:1px solid #dee2e6}.table-bordered thead td,.table-bordered thead th{border-bottom-width:2px}.table-borderless tbody+tbody,.table-borderless td,.table-borderless th,.table-borderless thead th{border:0}.table-striped tbody tr:nth-of-type(odd){background-color:rgba(0,0,0,.05)}.table-hover tbody tr:hover{color:#212529;background-color:rgba(0,0,0,.075)}.table-primary,.table-primary>td,.table-primary>th{background-color:#b8daff}.table-primary tbody+tbody,.table-primary td,.table-primary th,.table-primary thead th{border-color:#7abaff}.table-hover .table-primary:hover{background-color:#9fcdff}.table-hover .table-primary:hover>td,.table-hover .table-primary:hover>th{background-color:#9fcdff}.table-secondary,.table-secondary>td,.table-secondary>th{background-color:#d6d8db}.table-secondary tbody+tbody,.table-secondary td,.table-secondary th,.table-secondary thead th{border-color:#b3b7bb}.table-hover .table-secondary:hover{background-color:#c8cbcf}.table-hover .table-secondary:hover>td,.table-hover .table-secondary:hover>th{background-color:#c8cbcf}.table-success,.table-success>td,.table-success>th{background-color:#c3e6cb}.table-success tbody+tbody,.table-success td,.table-success th,.table-success thead th{border-color:#8fd19e}.table-hover .table-success:hover{background-color:#b1dfbb}.table-hover .table-success:hover>td,.table-hover .table-success:hover>th{background-color:#b1dfbb}.table-info,.table-info>td,.table-info>th{background-color:#bee5eb}.table-info tbody+tbody,.table-info td,.table-info th,.table-info thead th{border-color:#86cfda}.table-hover .table-info:hover{background-color:#abdde5}.table-hover .table-info:hover>td,.table-hover .table-info:hover>th{background-color:#abdde5}.table-warning,.table-warning>td,.table-warning>th{background-color:#ffeeba}.table-warning tbody+tbody,.table-warning td,.table-warning th,.table-warning thead th{border-color:#ffdf7e}.table-hover .table-warning:hover{background-color:#ffe8a1}.table-hover .table-warning:hover>td,.table-hover .table-warning:hover>th{background-color:#ffe8a1}.table-danger,.table-danger>td,.table-danger>th{background-color:#f5c6cb}.table-danger tbody+tbody,.table-danger td,.table-danger th,.table-danger thead th{border-color:#ed969e}.table-hover .table-danger:hover{background-color:#f1b0b7}.table-hover .table-danger:hover>td,.table-hover .table-danger:hover>th{background-color:#f1b0b7}.table-light,.table-light>td,.table-light>th{background-color:#fdfdfe}.table-light tbody+tbody,.table-light td,.table-light th,.table-light thead th{border-color:#fbfcfc}.table-hover .table-light:hover{background-color:#ececf6}.table-hover .table-light:hover>td,.table-hover .table-light:hover>th{background-color:#ececf6}.table-dark,.table-dark>td,.table-dark>th{background-color:#c6c8ca}.table-dark tbody+tbody,.table-dark td,.table-dark th,.table-dark thead th{border-color:#95999c}.table-hover .table-dark:hover{background-color:#b9bbbe}.table-hover .table-dark:hover>td,.table-hover .table-dark:hover>th{background-color:#b9bbbe}.table-active,.table-active>td,.table-active>th{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover>td,.table-hover .table-active:hover>th{background-color:rgba(0,0,0,.075)}.table .thead-dark th{color:#fff;background-color:#343a40;border-color:#454d55}.table .thead-light th{color:#495057;background-color:#e9ecef;border-color:#dee2e6}.table-dark{color:#fff;background-color:#343a40}.table-dark td,.table-dark th,.table-dark thead th{border-color:#454d55}.table-dark.table-bordered{border:0}.table-dark.table-striped tbody tr:nth-of-type(odd){background-color:rgba(255,255,255,.05)}.table-dark.table-hover tbody tr:hover{color:#fff;background-color:rgba(255,255,255,.075)}@media (max-width:575.98px){.table-responsive-sm{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-sm>.table-bordered{border:0}}@media (max-width:767.98px){.table-responsive-md{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-md>.table-bordered{border:0}}@media (max-width:991.98px){.table-responsive-lg{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-lg>.table-bordered{border:0}}@media (max-width:1199.98px){.table-responsive-xl{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-xl>.table-bordered{border:0}}.table-responsive{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive>.table-bordered{border:0}.form-control{display:block;width:100%;height:calc(1.5em + .75rem + 2px);padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;background-color:#fff;background-clip:padding-box;border:1px solid #ced4da;border-radius:.25rem;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control{transition:none}}.form-control::-ms-expand{background-color:transparent;border:0}.form-control:focus{color:#495057;background-color:#fff;border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.form-control::-webkit-input-placeholder{color:#6c757d;opacity:1}.form-control::-moz-placeholder{color:#6c757d;opacity:1}.form-control:-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::placeholder{color:#6c757d;opacity:1}.form-control:disabled,.form-control[readonly]{background-color:#e9ecef;opacity:1}select.form-control:focus::-ms-value{color:#495057;background-color:#fff}.form-control-file,.form-control-range{display:block;width:100%}.col-form-label{padding-top:calc(.375rem + 1px);padding-bottom:calc(.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + 1px);padding-bottom:calc(.5rem + 1px);font-size:1.25rem;line-height:1.5}.col-form-label-sm{padding-top:calc(.25rem + 1px);padding-bottom:calc(.25rem + 1px);font-size:.875rem;line-height:1.5}.form-control-plaintext{display:block;width:100%;padding-top:.375rem;padding-bottom:.375rem;margin-bottom:0;line-height:1.5;color:#212529;background-color:transparent;border:solid transparent;border-width:1px 0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm{padding-right:0;padding-left:0}.form-control-sm{height:calc(1.5em + .5rem + 2px);padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.form-control-lg{height:calc(1.5em + 1rem + 2px);padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}select.form-control[multiple],select.form-control[size]{height:auto}textarea.form-control{height:auto}.form-group{margin-bottom:1rem}.form-text{display:block;margin-top:.25rem}.form-row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-5px;margin-left:-5px}.form-row>.col,.form-row>[class*=col-]{padding-right:5px;padding-left:5px}.form-check{position:relative;display:block;padding-left:1.25rem}.form-check-input{position:absolute;margin-top:.3rem;margin-left:-1.25rem}.form-check-input:disabled~.form-check-label{color:#6c757d}.form-check-label{margin-bottom:0}.form-check-inline{display:-ms-inline-flexbox;display:inline-flex;-ms-flex-align:center;align-items:center;padding-left:0;margin-right:.75rem}.form-check-inline .form-check-input{position:static;margin-top:0;margin-right:.3125rem;margin-left:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#28a745}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;line-height:1.5;color:#fff;background-color:rgba(40,167,69,.9);border-radius:.25rem}.form-control.is-valid,.was-validated .form-control:valid{border-color:#28a745;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%2328a745' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:center right calc(.375em + .1875rem);background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-valid:focus,.was-validated .form-control:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.form-control.is-valid~.valid-feedback,.form-control.is-valid~.valid-tooltip,.was-validated .form-control:valid~.valid-feedback,.was-validated .form-control:valid~.valid-tooltip{display:block}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.custom-select.is-valid,.was-validated .custom-select:valid{border-color:#28a745;padding-right:calc((1em + .75rem) * 3 / 4 + 1.75rem);background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%2328a745' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e") #fff no-repeat center right 1.75rem/calc(.75em + .375rem) calc(.75em + .375rem)}.custom-select.is-valid:focus,.was-validated .custom-select:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.custom-select.is-valid~.valid-feedback,.custom-select.is-valid~.valid-tooltip,.was-validated .custom-select:valid~.valid-feedback,.was-validated .custom-select:valid~.valid-tooltip{display:block}.form-control-file.is-valid~.valid-feedback,.form-control-file.is-valid~.valid-tooltip,.was-validated .form-control-file:valid~.valid-feedback,.was-validated .form-control-file:valid~.valid-tooltip{display:block}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:#28a745}.form-check-input.is-valid~.valid-feedback,.form-check-input.is-valid~.valid-tooltip,.was-validated .form-check-input:valid~.valid-feedback,.was-validated .form-check-input:valid~.valid-tooltip{display:block}.custom-control-input.is-valid~.custom-control-label,.was-validated .custom-control-input:valid~.custom-control-label{color:#28a745}.custom-control-input.is-valid~.custom-control-label::before,.was-validated .custom-control-input:valid~.custom-control-label::before{border-color:#28a745}.custom-control-input.is-valid~.valid-feedback,.custom-control-input.is-valid~.valid-tooltip,.was-validated .custom-control-input:valid~.valid-feedback,.was-validated .custom-control-input:valid~.valid-tooltip{display:block}.custom-control-input.is-valid:checked~.custom-control-label::before,.was-validated .custom-control-input:valid:checked~.custom-control-label::before{border-color:#34ce57;background-color:#34ce57}.custom-control-input.is-valid:focus~.custom-control-label::before,.was-validated .custom-control-input:valid:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.custom-control-input.is-valid:focus:not(:checked)~.custom-control-label::before,.was-validated .custom-control-input:valid:focus:not(:checked)~.custom-control-label::before{border-color:#28a745}.custom-file-input.is-valid~.custom-file-label,.was-validated .custom-file-input:valid~.custom-file-label{border-color:#28a745}.custom-file-input.is-valid~.valid-feedback,.custom-file-input.is-valid~.valid-tooltip,.was-validated .custom-file-input:valid~.valid-feedback,.was-validated .custom-file-input:valid~.valid-tooltip{display:block}.custom-file-input.is-valid:focus~.custom-file-label,.was-validated .custom-file-input:valid:focus~.custom-file-label{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#dc3545}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;line-height:1.5;color:#fff;background-color:rgba(220,53,69,.9);border-radius:.25rem}.form-control.is-invalid,.was-validated .form-control:invalid{border-color:#dc3545;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23dc3545' viewBox='-2 -2 7 7'%3e%3cpath stroke='%23dc3545' d='M0 0l3 3m0-3L0 3'/%3e%3ccircle r='.5'/%3e%3ccircle cx='3' r='.5'/%3e%3ccircle cy='3' r='.5'/%3e%3ccircle cx='3' cy='3' r='.5'/%3e%3c/svg%3E");background-repeat:no-repeat;background-position:center right calc(.375em + .1875rem);background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-invalid:focus,.was-validated .form-control:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-control.is-invalid~.invalid-feedback,.form-control.is-invalid~.invalid-tooltip,.was-validated .form-control:invalid~.invalid-feedback,.was-validated .form-control:invalid~.invalid-tooltip{display:block}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.custom-select.is-invalid,.was-validated .custom-select:invalid{border-color:#dc3545;padding-right:calc((1em + .75rem) * 3 / 4 + 1.75rem);background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23dc3545' viewBox='-2 -2 7 7'%3e%3cpath stroke='%23dc3545' d='M0 0l3 3m0-3L0 3'/%3e%3ccircle r='.5'/%3e%3ccircle cx='3' r='.5'/%3e%3ccircle cy='3' r='.5'/%3e%3ccircle cx='3' cy='3' r='.5'/%3e%3c/svg%3E") #fff no-repeat center right 1.75rem/calc(.75em + .375rem) calc(.75em + .375rem)}.custom-select.is-invalid:focus,.was-validated .custom-select:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.custom-select.is-invalid~.invalid-feedback,.custom-select.is-invalid~.invalid-tooltip,.was-validated .custom-select:invalid~.invalid-feedback,.was-validated .custom-select:invalid~.invalid-tooltip{display:block}.form-control-file.is-invalid~.invalid-feedback,.form-control-file.is-invalid~.invalid-tooltip,.was-validated .form-control-file:invalid~.invalid-feedback,.was-validated .form-control-file:invalid~.invalid-tooltip{display:block}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:#dc3545}.form-check-input.is-invalid~.invalid-feedback,.form-check-input.is-invalid~.invalid-tooltip,.was-validated .form-check-input:invalid~.invalid-feedback,.was-validated .form-check-input:invalid~.invalid-tooltip{display:block}.custom-control-input.is-invalid~.custom-control-label,.was-validated .custom-control-input:invalid~.custom-control-label{color:#dc3545}.custom-control-input.is-invalid~.custom-control-label::before,.was-validated .custom-control-input:invalid~.custom-control-label::before{border-color:#dc3545}.custom-control-input.is-invalid~.invalid-feedback,.custom-control-input.is-invalid~.invalid-tooltip,.was-validated .custom-control-input:invalid~.invalid-feedback,.was-validated .custom-control-input:invalid~.invalid-tooltip{display:block}.custom-control-input.is-invalid:checked~.custom-control-label::before,.was-validated .custom-control-input:invalid:checked~.custom-control-label::before{border-color:#e4606d;background-color:#e4606d}.custom-control-input.is-invalid:focus~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.custom-control-input.is-invalid:focus:not(:checked)~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus:not(:checked)~.custom-control-label::before{border-color:#dc3545}.custom-file-input.is-invalid~.custom-file-label,.was-validated .custom-file-input:invalid~.custom-file-label{border-color:#dc3545}.custom-file-input.is-invalid~.invalid-feedback,.custom-file-input.is-invalid~.invalid-tooltip,.was-validated .custom-file-input:invalid~.invalid-feedback,.was-validated .custom-file-input:invalid~.invalid-tooltip{display:block}.custom-file-input.is-invalid:focus~.custom-file-label,.was-validated .custom-file-input:invalid:focus~.custom-file-label{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-inline{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center}.form-inline .form-check{width:100%}@media (min-width:576px){.form-inline label{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;margin-bottom:0}.form-inline .form-group{display:-ms-flexbox;display:flex;-ms-flex:0 0 auto;flex:0 0 auto;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center;margin-bottom:0}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-plaintext{display:inline-block}.form-inline .custom-select,.form-inline .input-group{width:auto}.form-inline .form-check{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:auto;padding-left:0}.form-inline .form-check-input{position:relative;-ms-flex-negative:0;flex-shrink:0;margin-top:0;margin-right:.25rem;margin-left:0}.form-inline .custom-control{-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center}.form-inline .custom-control-label{margin-bottom:0}}.btn{display:inline-block;font-weight:400;color:#212529;text-align:center;vertical-align:middle;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:transparent;border:1px solid transparent;padding:.375rem .75rem;font-size:1rem;line-height:1.5;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.btn{transition:none}}.btn:hover{color:#212529;text-decoration:none}.btn.focus,.btn:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.btn.disabled,.btn:disabled{opacity:.65}a.btn.disabled,fieldset:disabled a.btn{pointer-events:none}.btn-primary{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:hover{color:#fff;background-color:#0069d9;border-color:#0062cc}.btn-primary.focus,.btn-primary:focus{box-shadow:0 0 0 .2rem rgba(38,143,255,.5)}.btn-primary.disabled,.btn-primary:disabled{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:not(:disabled):not(.disabled).active,.btn-primary:not(:disabled):not(.disabled):active,.show>.btn-primary.dropdown-toggle{color:#fff;background-color:#0062cc;border-color:#005cbf}.btn-primary:not(:disabled):not(.disabled).active:focus,.btn-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(38,143,255,.5)}.btn-secondary{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:hover{color:#fff;background-color:#5a6268;border-color:#545b62}.btn-secondary.focus,.btn-secondary:focus{box-shadow:0 0 0 .2rem rgba(130,138,145,.5)}.btn-secondary.disabled,.btn-secondary:disabled{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:not(:disabled):not(.disabled).active,.btn-secondary:not(:disabled):not(.disabled):active,.show>.btn-secondary.dropdown-toggle{color:#fff;background-color:#545b62;border-color:#4e555b}.btn-secondary:not(:disabled):not(.disabled).active:focus,.btn-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(130,138,145,.5)}.btn-success{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:hover{color:#fff;background-color:#218838;border-color:#1e7e34}.btn-success.focus,.btn-success:focus{box-shadow:0 0 0 .2rem rgba(72,180,97,.5)}.btn-success.disabled,.btn-success:disabled{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:not(:disabled):not(.disabled).active,.btn-success:not(:disabled):not(.disabled):active,.show>.btn-success.dropdown-toggle{color:#fff;background-color:#1e7e34;border-color:#1c7430}.btn-success:not(:disabled):not(.disabled).active:focus,.btn-success:not(:disabled):not(.disabled):active:focus,.show>.btn-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(72,180,97,.5)}.btn-info{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:hover{color:#fff;background-color:#138496;border-color:#117a8b}.btn-info.focus,.btn-info:focus{box-shadow:0 0 0 .2rem rgba(58,176,195,.5)}.btn-info.disabled,.btn-info:disabled{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:not(:disabled):not(.disabled).active,.btn-info:not(:disabled):not(.disabled):active,.show>.btn-info.dropdown-toggle{color:#fff;background-color:#117a8b;border-color:#10707f}.btn-info:not(:disabled):not(.disabled).active:focus,.btn-info:not(:disabled):not(.disabled):active:focus,.show>.btn-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(58,176,195,.5)}.btn-warning{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:hover{color:#212529;background-color:#e0a800;border-color:#d39e00}.btn-warning.focus,.btn-warning:focus{box-shadow:0 0 0 .2rem rgba(222,170,12,.5)}.btn-warning.disabled,.btn-warning:disabled{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:not(:disabled):not(.disabled).active,.btn-warning:not(:disabled):not(.disabled):active,.show>.btn-warning.dropdown-toggle{color:#212529;background-color:#d39e00;border-color:#c69500}.btn-warning:not(:disabled):not(.disabled).active:focus,.btn-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(222,170,12,.5)}.btn-danger{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:hover{color:#fff;background-color:#c82333;border-color:#bd2130}.btn-danger.focus,.btn-danger:focus{box-shadow:0 0 0 .2rem rgba(225,83,97,.5)}.btn-danger.disabled,.btn-danger:disabled{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:not(:disabled):not(.disabled).active,.btn-danger:not(:disabled):not(.disabled):active,.show>.btn-danger.dropdown-toggle{color:#fff;background-color:#bd2130;border-color:#b21f2d}.btn-danger:not(:disabled):not(.disabled).active:focus,.btn-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(225,83,97,.5)}.btn-light{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:hover{color:#212529;background-color:#e2e6ea;border-color:#dae0e5}.btn-light.focus,.btn-light:focus{box-shadow:0 0 0 .2rem rgba(216,217,219,.5)}.btn-light.disabled,.btn-light:disabled{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:not(:disabled):not(.disabled).active,.btn-light:not(:disabled):not(.disabled):active,.show>.btn-light.dropdown-toggle{color:#212529;background-color:#dae0e5;border-color:#d3d9df}.btn-light:not(:disabled):not(.disabled).active:focus,.btn-light:not(:disabled):not(.disabled):active:focus,.show>.btn-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(216,217,219,.5)}.btn-dark{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:hover{color:#fff;background-color:#23272b;border-color:#1d2124}.btn-dark.focus,.btn-dark:focus{box-shadow:0 0 0 .2rem rgba(82,88,93,.5)}.btn-dark.disabled,.btn-dark:disabled{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:not(:disabled):not(.disabled).active,.btn-dark:not(:disabled):not(.disabled):active,.show>.btn-dark.dropdown-toggle{color:#fff;background-color:#1d2124;border-color:#171a1d}.btn-dark:not(:disabled):not(.disabled).active:focus,.btn-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(82,88,93,.5)}.btn-outline-primary{color:#007bff;border-color:#007bff}.btn-outline-primary:hover{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary.focus,.btn-outline-primary:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-primary.disabled,.btn-outline-primary:disabled{color:#007bff;background-color:transparent}.btn-outline-primary:not(:disabled):not(.disabled).active,.btn-outline-primary:not(:disabled):not(.disabled):active,.show>.btn-outline-primary.dropdown-toggle{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary:not(:disabled):not(.disabled).active:focus,.btn-outline-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-secondary{color:#6c757d;border-color:#6c757d}.btn-outline-secondary:hover{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary.focus,.btn-outline-secondary:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-secondary.disabled,.btn-outline-secondary:disabled{color:#6c757d;background-color:transparent}.btn-outline-secondary:not(:disabled):not(.disabled).active,.btn-outline-secondary:not(:disabled):not(.disabled):active,.show>.btn-outline-secondary.dropdown-toggle{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary:not(:disabled):not(.disabled).active:focus,.btn-outline-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-success{color:#28a745;border-color:#28a745}.btn-outline-success:hover{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success.focus,.btn-outline-success:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-success.disabled,.btn-outline-success:disabled{color:#28a745;background-color:transparent}.btn-outline-success:not(:disabled):not(.disabled).active,.btn-outline-success:not(:disabled):not(.disabled):active,.show>.btn-outline-success.dropdown-toggle{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success:not(:disabled):not(.disabled).active:focus,.btn-outline-success:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-info{color:#17a2b8;border-color:#17a2b8}.btn-outline-info:hover{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info.focus,.btn-outline-info:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-info.disabled,.btn-outline-info:disabled{color:#17a2b8;background-color:transparent}.btn-outline-info:not(:disabled):not(.disabled).active,.btn-outline-info:not(:disabled):not(.disabled):active,.show>.btn-outline-info.dropdown-toggle{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info:not(:disabled):not(.disabled).active:focus,.btn-outline-info:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-warning{color:#ffc107;border-color:#ffc107}.btn-outline-warning:hover{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning.focus,.btn-outline-warning:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-warning.disabled,.btn-outline-warning:disabled{color:#ffc107;background-color:transparent}.btn-outline-warning:not(:disabled):not(.disabled).active,.btn-outline-warning:not(:disabled):not(.disabled):active,.show>.btn-outline-warning.dropdown-toggle{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning:not(:disabled):not(.disabled).active:focus,.btn-outline-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-danger{color:#dc3545;border-color:#dc3545}.btn-outline-danger:hover{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger.focus,.btn-outline-danger:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-danger.disabled,.btn-outline-danger:disabled{color:#dc3545;background-color:transparent}.btn-outline-danger:not(:disabled):not(.disabled).active,.btn-outline-danger:not(:disabled):not(.disabled):active,.show>.btn-outline-danger.dropdown-toggle{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger:not(:disabled):not(.disabled).active:focus,.btn-outline-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-light{color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:hover{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light.focus,.btn-outline-light:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-light.disabled,.btn-outline-light:disabled{color:#f8f9fa;background-color:transparent}.btn-outline-light:not(:disabled):not(.disabled).active,.btn-outline-light:not(:disabled):not(.disabled):active,.show>.btn-outline-light.dropdown-toggle{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:not(:disabled):not(.disabled).active:focus,.btn-outline-light:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-dark{color:#343a40;border-color:#343a40}.btn-outline-dark:hover{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark.focus,.btn-outline-dark:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-outline-dark.disabled,.btn-outline-dark:disabled{color:#343a40;background-color:transparent}.btn-outline-dark:not(:disabled):not(.disabled).active,.btn-outline-dark:not(:disabled):not(.disabled):active,.show>.btn-outline-dark.dropdown-toggle{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark:not(:disabled):not(.disabled).active:focus,.btn-outline-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-link{font-weight:400;color:#007bff;text-decoration:none}.btn-link:hover{color:#0056b3;text-decoration:underline}.btn-link.focus,.btn-link:focus{text-decoration:underline;box-shadow:none}.btn-link.disabled,.btn-link:disabled{color:#6c757d;pointer-events:none}.btn-group-lg>.btn,.btn-lg{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.btn-group-sm>.btn,.btn-sm{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:.5rem}input[type=button].btn-block,input[type=reset].btn-block,input[type=submit].btn-block{width:100%}.fade{transition:opacity .15s linear}@media (prefers-reduced-motion:reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{position:relative;height:0;overflow:hidden;transition:height .35s ease}@media (prefers-reduced-motion:reduce){.collapsing{transition:none}}.dropdown,.dropleft,.dropright,.dropup{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:10rem;padding:.5rem 0;margin:.125rem 0 0;font-size:1rem;color:#212529;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.15);border-radius:.25rem}.dropdown-menu-left{right:auto;left:0}.dropdown-menu-right{right:0;left:auto}@media (min-width:576px){.dropdown-menu-sm-left{right:auto;left:0}.dropdown-menu-sm-right{right:0;left:auto}}@media (min-width:768px){.dropdown-menu-md-left{right:auto;left:0}.dropdown-menu-md-right{right:0;left:auto}}@media (min-width:992px){.dropdown-menu-lg-left{right:auto;left:0}.dropdown-menu-lg-right{right:0;left:auto}}@media (min-width:1200px){.dropdown-menu-xl-left{right:auto;left:0}.dropdown-menu-xl-right{right:0;left:auto}}.dropup .dropdown-menu{top:auto;bottom:100%;margin-top:0;margin-bottom:.125rem}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-menu{top:0;right:auto;left:100%;margin-top:0;margin-left:.125rem}.dropright .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropright .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-toggle::after{vertical-align:0}.dropleft .dropdown-menu{top:0;right:100%;left:auto;margin-top:0;margin-right:.125rem}.dropleft .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropleft .dropdown-toggle::after{display:none}.dropleft .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropleft .dropdown-toggle:empty::after{margin-left:0}.dropleft .dropdown-toggle::before{vertical-align:0}.dropdown-menu[x-placement^=bottom],.dropdown-menu[x-placement^=left],.dropdown-menu[x-placement^=right],.dropdown-menu[x-placement^=top]{right:auto;bottom:auto}.dropdown-divider{height:0;margin:.5rem 0;overflow:hidden;border-top:1px solid #e9ecef}.dropdown-item{display:block;width:100%;padding:.25rem 1.5rem;clear:both;font-weight:400;color:#212529;text-align:inherit;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:focus,.dropdown-item:hover{color:#16181b;text-decoration:none;background-color:#f8f9fa}.dropdown-item.active,.dropdown-item:active{color:#fff;text-decoration:none;background-color:#007bff}.dropdown-item.disabled,.dropdown-item:disabled{color:#6c757d;pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:.5rem 1.5rem;margin-bottom:0;font-size:.875rem;color:#6c757d;white-space:nowrap}.dropdown-item-text{display:block;padding:.25rem 1.5rem;color:#212529}.btn-group,.btn-group-vertical{position:relative;display:-ms-inline-flexbox;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;-ms-flex:1 1 auto;flex:1 1 auto}.btn-group-vertical>.btn:hover,.btn-group>.btn:hover{z-index:1}.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus{z-index:1}.btn-toolbar{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-pack:start;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group>.btn-group:not(:first-child),.btn-group>.btn:not(:first-child){margin-left:-1px}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropright .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropleft .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:center;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn-group:not(:first-child),.btn-group-vertical>.btn:not(:first-child){margin-top:-1px}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn:not(:first-child){border-top-left-radius:0;border-top-right-radius:0}.btn-group-toggle>.btn,.btn-group-toggle>.btn-group>.btn{margin-bottom:0}.btn-group-toggle>.btn input[type=checkbox],.btn-group-toggle>.btn input[type=radio],.btn-group-toggle>.btn-group>.btn input[type=checkbox],.btn-group-toggle>.btn-group>.btn input[type=radio]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:stretch;align-items:stretch;width:100%}.input-group>.custom-file,.input-group>.custom-select,.input-group>.form-control,.input-group>.form-control-plaintext{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;width:1%;margin-bottom:0}.input-group>.custom-file+.custom-file,.input-group>.custom-file+.custom-select,.input-group>.custom-file+.form-control,.input-group>.custom-select+.custom-file,.input-group>.custom-select+.custom-select,.input-group>.custom-select+.form-control,.input-group>.form-control+.custom-file,.input-group>.form-control+.custom-select,.input-group>.form-control+.form-control,.input-group>.form-control-plaintext+.custom-file,.input-group>.form-control-plaintext+.custom-select,.input-group>.form-control-plaintext+.form-control{margin-left:-1px}.input-group>.custom-file .custom-file-input:focus~.custom-file-label,.input-group>.custom-select:focus,.input-group>.form-control:focus{z-index:3}.input-group>.custom-file .custom-file-input:focus{z-index:4}.input-group>.custom-select:not(:last-child),.input-group>.form-control:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-select:not(:first-child),.input-group>.form-control:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.custom-file{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center}.input-group>.custom-file:not(:last-child) .custom-file-label,.input-group>.custom-file:not(:last-child) .custom-file-label::after{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-file:not(:first-child) .custom-file-label{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-append,.input-group-prepend{display:-ms-flexbox;display:flex}.input-group-append .btn,.input-group-prepend .btn{position:relative;z-index:2}.input-group-append .btn:focus,.input-group-prepend .btn:focus{z-index:3}.input-group-append .btn+.btn,.input-group-append .btn+.input-group-text,.input-group-append .input-group-text+.btn,.input-group-append .input-group-text+.input-group-text,.input-group-prepend .btn+.btn,.input-group-prepend .btn+.input-group-text,.input-group-prepend .input-group-text+.btn,.input-group-prepend .input-group-text+.input-group-text{margin-left:-1px}.input-group-prepend{margin-right:-1px}.input-group-append{margin-left:-1px}.input-group-text{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.375rem .75rem;margin-bottom:0;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;text-align:center;white-space:nowrap;background-color:#e9ecef;border:1px solid #ced4da;border-radius:.25rem}.input-group-text input[type=checkbox],.input-group-text input[type=radio]{margin-top:0}.input-group-lg>.custom-select,.input-group-lg>.form-control:not(textarea){height:calc(1.5em + 1rem + 2px)}.input-group-lg>.custom-select,.input-group-lg>.form-control,.input-group-lg>.input-group-append>.btn,.input-group-lg>.input-group-append>.input-group-text,.input-group-lg>.input-group-prepend>.btn,.input-group-lg>.input-group-prepend>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.input-group-sm>.custom-select,.input-group-sm>.form-control:not(textarea){height:calc(1.5em + .5rem + 2px)}.input-group-sm>.custom-select,.input-group-sm>.form-control,.input-group-sm>.input-group-append>.btn,.input-group-sm>.input-group-append>.input-group-text,.input-group-sm>.input-group-prepend>.btn,.input-group-sm>.input-group-prepend>.input-group-text{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.input-group-lg>.custom-select,.input-group-sm>.custom-select{padding-right:1.75rem}.input-group>.input-group-append:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group>.input-group-append:last-child>.input-group-text:not(:last-child),.input-group>.input-group-append:not(:last-child)>.btn,.input-group>.input-group-append:not(:last-child)>.input-group-text,.input-group>.input-group-prepend>.btn,.input-group>.input-group-prepend>.input-group-text{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.input-group-append>.btn,.input-group>.input-group-append>.input-group-text,.input-group>.input-group-prepend:first-child>.btn:not(:first-child),.input-group>.input-group-prepend:first-child>.input-group-text:not(:first-child),.input-group>.input-group-prepend:not(:first-child)>.btn,.input-group>.input-group-prepend:not(:first-child)>.input-group-text{border-top-left-radius:0;border-bottom-left-radius:0}.custom-control{position:relative;display:block;min-height:1.5rem;padding-left:1.5rem}.custom-control-inline{display:-ms-inline-flexbox;display:inline-flex;margin-right:1rem}.custom-control-input{position:absolute;z-index:-1;opacity:0}.custom-control-input:checked~.custom-control-label::before{color:#fff;border-color:#007bff;background-color:#007bff}.custom-control-input:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-control-input:focus:not(:checked)~.custom-control-label::before{border-color:#80bdff}.custom-control-input:not(:disabled):active~.custom-control-label::before{color:#fff;background-color:#b3d7ff;border-color:#b3d7ff}.custom-control-input:disabled~.custom-control-label{color:#6c757d}.custom-control-input:disabled~.custom-control-label::before{background-color:#e9ecef}.custom-control-label{position:relative;margin-bottom:0;vertical-align:top}.custom-control-label::before{position:absolute;top:.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;pointer-events:none;content:"";background-color:#fff;border:#adb5bd solid 1px}.custom-control-label::after{position:absolute;top:.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;content:"";background:no-repeat 50%/50% 50%}.custom-checkbox .custom-control-label::before{border-radius:.25rem}.custom-checkbox .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23fff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26 2.974 7.25 8 2.193z'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::before{border-color:#007bff;background-color:#007bff}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 4'%3e%3cpath stroke='%23fff' d='M0 2h4'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-checkbox .custom-control-input:disabled:indeterminate~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-radio .custom-control-label::before{border-radius:50%}.custom-radio .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e")}.custom-radio .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-switch{padding-left:2.25rem}.custom-switch .custom-control-label::before{left:-2.25rem;width:1.75rem;pointer-events:all;border-radius:.5rem}.custom-switch .custom-control-label::after{top:calc(.25rem + 2px);left:calc(-2.25rem + 2px);width:calc(1rem - 4px);height:calc(1rem - 4px);background-color:#adb5bd;border-radius:.5rem;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-transform .15s ease-in-out;transition:transform .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:transform .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-transform .15s ease-in-out}@media (prefers-reduced-motion:reduce){.custom-switch .custom-control-label::after{transition:none}}.custom-switch .custom-control-input:checked~.custom-control-label::after{background-color:#fff;-webkit-transform:translateX(.75rem);transform:translateX(.75rem)}.custom-switch .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-select{display:inline-block;width:100%;height:calc(1.5em + .75rem + 2px);padding:.375rem 1.75rem .375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;vertical-align:middle;background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px;background-color:#fff;border:1px solid #ced4da;border-radius:.25rem;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-select:focus{border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-select:focus::-ms-value{color:#495057;background-color:#fff}.custom-select[multiple],.custom-select[size]:not([size="1"]){height:auto;padding-right:.75rem;background-image:none}.custom-select:disabled{color:#6c757d;background-color:#e9ecef}.custom-select::-ms-expand{display:none}.custom-select-sm{height:calc(1.5em + .5rem + 2px);padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem}.custom-select-lg{height:calc(1.5em + 1rem + 2px);padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem}.custom-file{position:relative;display:inline-block;width:100%;height:calc(1.5em + .75rem + 2px);margin-bottom:0}.custom-file-input{position:relative;z-index:2;width:100%;height:calc(1.5em + .75rem + 2px);margin:0;opacity:0}.custom-file-input:focus~.custom-file-label{border-color:#80bdff;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-file-input:disabled~.custom-file-label{background-color:#e9ecef}.custom-file-input:lang(en)~.custom-file-label::after{content:"Browse"}.custom-file-input~.custom-file-label[data-browse]::after{content:attr(data-browse)}.custom-file-label{position:absolute;top:0;right:0;left:0;z-index:1;height:calc(1.5em + .75rem + 2px);padding:.375rem .75rem;font-weight:400;line-height:1.5;color:#495057;background-color:#fff;border:1px solid #ced4da;border-radius:.25rem}.custom-file-label::after{position:absolute;top:0;right:0;bottom:0;z-index:3;display:block;height:calc(1.5em + .75rem);padding:.375rem .75rem;line-height:1.5;color:#495057;content:"Browse";background-color:#e9ecef;border-left:inherit;border-radius:0 .25rem .25rem 0}.custom-range{width:100%;height:calc(1rem + .4rem);padding:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-range:focus{outline:0}.custom-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range:focus::-ms-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range::-moz-focus-outer{border:0}.custom-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;background-color:#007bff;border:0;border-radius:1rem;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;-webkit-appearance:none;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-webkit-slider-thumb{transition:none}}.custom-range::-webkit-slider-thumb:active{background-color:#b3d7ff}.custom-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-moz-range-thumb{width:1rem;height:1rem;background-color:#007bff;border:0;border-radius:1rem;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;-moz-appearance:none;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-moz-range-thumb{transition:none}}.custom-range::-moz-range-thumb:active{background-color:#b3d7ff}.custom-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-ms-thumb{width:1rem;height:1rem;margin-top:0;margin-right:.2rem;margin-left:.2rem;background-color:#007bff;border:0;border-radius:1rem;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-ms-thumb{transition:none}}.custom-range::-ms-thumb:active{background-color:#b3d7ff}.custom-range::-ms-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:transparent;border-color:transparent;border-width:.5rem}.custom-range::-ms-fill-lower{background-color:#dee2e6;border-radius:1rem}.custom-range::-ms-fill-upper{margin-right:15px;background-color:#dee2e6;border-radius:1rem}.custom-range:disabled::-webkit-slider-thumb{background-color:#adb5bd}.custom-range:disabled::-webkit-slider-runnable-track{cursor:default}.custom-range:disabled::-moz-range-thumb{background-color:#adb5bd}.custom-range:disabled::-moz-range-track{cursor:default}.custom-range:disabled::-ms-thumb{background-color:#adb5bd}.custom-control-label::before,.custom-file-label,.custom-select{transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.custom-control-label::before,.custom-file-label,.custom-select{transition:none}}.nav{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:.5rem 1rem}.nav-link:focus,.nav-link:hover{text-decoration:none}.nav-link.disabled{color:#6c757d;pointer-events:none;cursor:default}.nav-tabs{border-bottom:1px solid #dee2e6}.nav-tabs .nav-item{margin-bottom:-1px}.nav-tabs .nav-link{border:1px solid transparent;border-top-left-radius:.25rem;border-top-right-radius:.25rem}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{border-color:#e9ecef #e9ecef #dee2e6}.nav-tabs .nav-link.disabled{color:#6c757d;background-color:transparent;border-color:transparent}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:#495057;background-color:#fff;border-color:#dee2e6 #dee2e6 #fff}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.nav-pills .nav-link{border-radius:.25rem}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:#fff;background-color:#007bff}.nav-fill .nav-item{-ms-flex:1 1 auto;flex:1 1 auto;text-align:center}.nav-justified .nav-item{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;text-align:center}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between;padding:.5rem 1rem}.navbar>.container,.navbar>.container-fluid{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between}.navbar-brand{display:inline-block;padding-top:.3125rem;padding-bottom:.3125rem;margin-right:1rem;font-size:1.25rem;line-height:inherit;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{text-decoration:none}.navbar-nav{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link{padding-right:0;padding-left:0}.navbar-nav .dropdown-menu{position:static;float:none}.navbar-text{display:inline-block;padding-top:.5rem;padding-bottom:.5rem}.navbar-collapse{-ms-flex-preferred-size:100%;flex-basis:100%;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center}.navbar-toggler{padding:.25rem .75rem;font-size:1.25rem;line-height:1;background-color:transparent;border:1px solid transparent;border-radius:.25rem}.navbar-toggler:focus,.navbar-toggler:hover{text-decoration:none}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;content:"";background:no-repeat center center;background-size:100% 100%}@media (max-width:575.98px){.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:576px){.navbar-expand-sm{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-sm .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-sm .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}}@media (max-width:767.98px){.navbar-expand-md>.container,.navbar-expand-md>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:768px){.navbar-expand-md{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-md .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-md>.container,.navbar-expand-md>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-md .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}}@media (max-width:991.98px){.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:992px){.navbar-expand-lg{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-lg .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-lg .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}}@media (max-width:1199.98px){.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:1200px){.navbar-expand-xl{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-xl .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-xl .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}}.navbar-expand{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand>.container,.navbar-expand>.container-fluid{padding-right:0;padding-left:0}.navbar-expand .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand>.container,.navbar-expand>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-light .navbar-brand{color:rgba(0,0,0,.9)}.navbar-light .navbar-brand:focus,.navbar-light .navbar-brand:hover{color:rgba(0,0,0,.9)}.navbar-light .navbar-nav .nav-link{color:rgba(0,0,0,.5)}.navbar-light .navbar-nav .nav-link:focus,.navbar-light .navbar-nav .nav-link:hover{color:rgba(0,0,0,.7)}.navbar-light .navbar-nav .nav-link.disabled{color:rgba(0,0,0,.3)}.navbar-light .navbar-nav .active>.nav-link,.navbar-light .navbar-nav .nav-link.active,.navbar-light .navbar-nav .nav-link.show,.navbar-light .navbar-nav .show>.nav-link{color:rgba(0,0,0,.9)}.navbar-light .navbar-toggler{color:rgba(0,0,0,.5);border-color:rgba(0,0,0,.1)}.navbar-light .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg viewBox='0 0 30 30' xmlns='http://www.w3.org/2000/svg'%3e%3cpath stroke='rgba(0, 0, 0, 0.5)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-light .navbar-text{color:rgba(0,0,0,.5)}.navbar-light .navbar-text a{color:rgba(0,0,0,.9)}.navbar-light .navbar-text a:focus,.navbar-light .navbar-text a:hover{color:rgba(0,0,0,.9)}.navbar-dark .navbar-brand{color:#fff}.navbar-dark .navbar-brand:focus,.navbar-dark .navbar-brand:hover{color:#fff}.navbar-dark .navbar-nav .nav-link{color:rgba(255,255,255,.5)}.navbar-dark .navbar-nav .nav-link:focus,.navbar-dark .navbar-nav .nav-link:hover{color:rgba(255,255,255,.75)}.navbar-dark .navbar-nav .nav-link.disabled{color:rgba(255,255,255,.25)}.navbar-dark .navbar-nav .active>.nav-link,.navbar-dark .navbar-nav .nav-link.active,.navbar-dark .navbar-nav .nav-link.show,.navbar-dark .navbar-nav .show>.nav-link{color:#fff}.navbar-dark .navbar-toggler{color:rgba(255,255,255,.5);border-color:rgba(255,255,255,.1)}.navbar-dark .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg viewBox='0 0 30 30' xmlns='http://www.w3.org/2000/svg'%3e%3cpath stroke='rgba(255, 255, 255, 0.5)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-dark .navbar-text{color:rgba(255,255,255,.5)}.navbar-dark .navbar-text a{color:#fff}.navbar-dark .navbar-text a:focus,.navbar-dark .navbar-text a:hover{color:#fff}.card{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;min-width:0;word-wrap:break-word;background-color:#fff;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}.card>hr{margin-right:0;margin-left:0}.card>.list-group:first-child .list-group-item:first-child{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.card>.list-group:last-child .list-group-item:last-child{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.card-body{-ms-flex:1 1 auto;flex:1 1 auto;padding:1.25rem}.card-title{margin-bottom:.75rem}.card-subtitle{margin-top:-.375rem;margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link:hover{text-decoration:none}.card-link+.card-link{margin-left:1.25rem}.card-header{padding:.75rem 1.25rem;margin-bottom:0;background-color:rgba(0,0,0,.03);border-bottom:1px solid rgba(0,0,0,.125)}.card-header:first-child{border-radius:calc(.25rem - 1px) calc(.25rem - 1px) 0 0}.card-header+.list-group .list-group-item:first-child{border-top:0}.card-footer{padding:.75rem 1.25rem;background-color:rgba(0,0,0,.03);border-top:1px solid rgba(0,0,0,.125)}.card-footer:last-child{border-radius:0 0 calc(.25rem - 1px) calc(.25rem - 1px)}.card-header-tabs{margin-right:-.625rem;margin-bottom:-.75rem;margin-left:-.625rem;border-bottom:0}.card-header-pills{margin-right:-.625rem;margin-left:-.625rem}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:1.25rem}.card-img{width:100%;border-radius:calc(.25rem - 1px)}.card-img-top{width:100%;border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card-img-bottom{width:100%;border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card-deck{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}.card-deck .card{margin-bottom:15px}@media (min-width:576px){.card-deck{-ms-flex-flow:row wrap;flex-flow:row wrap;margin-right:-15px;margin-left:-15px}.card-deck .card{display:-ms-flexbox;display:flex;-ms-flex:1 0 0%;flex:1 0 0%;-ms-flex-direction:column;flex-direction:column;margin-right:15px;margin-bottom:0;margin-left:15px}}.card-group{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}.card-group>.card{margin-bottom:15px}@media (min-width:576px){.card-group{-ms-flex-flow:row wrap;flex-flow:row wrap}.card-group>.card{-ms-flex:1 0 0%;flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:not(:last-child) .card-header,.card-group>.card:not(:last-child) .card-img-top{border-top-right-radius:0}.card-group>.card:not(:last-child) .card-footer,.card-group>.card:not(:last-child) .card-img-bottom{border-bottom-right-radius:0}.card-group>.card:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:not(:first-child) .card-header,.card-group>.card:not(:first-child) .card-img-top{border-top-left-radius:0}.card-group>.card:not(:first-child) .card-footer,.card-group>.card:not(:first-child) .card-img-bottom{border-bottom-left-radius:0}}.card-columns .card{margin-bottom:.75rem}@media (min-width:576px){.card-columns{-webkit-column-count:3;-moz-column-count:3;column-count:3;-webkit-column-gap:1.25rem;-moz-column-gap:1.25rem;column-gap:1.25rem;orphans:1;widows:1}.card-columns .card{display:inline-block;width:100%}}.accordion>.card{overflow:hidden}.accordion>.card:not(:first-of-type) .card-header:first-child{border-radius:0}.accordion>.card:not(:first-of-type):not(:last-of-type){border-bottom:0;border-radius:0}.accordion>.card:first-of-type{border-bottom:0;border-bottom-right-radius:0;border-bottom-left-radius:0}.accordion>.card:last-of-type{border-top-left-radius:0;border-top-right-radius:0}.accordion>.card .card-header{margin-bottom:-1px}.breadcrumb{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding:.75rem 1rem;margin-bottom:1rem;list-style:none;background-color:#e9ecef;border-radius:.25rem}.breadcrumb-item+.breadcrumb-item{padding-left:.5rem}.breadcrumb-item+.breadcrumb-item::before{display:inline-block;padding-right:.5rem;color:#6c757d;content:"/"}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:underline}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:none}.breadcrumb-item.active{color:#6c757d}.pagination{display:-ms-flexbox;display:flex;padding-left:0;list-style:none;border-radius:.25rem}.page-link{position:relative;display:block;padding:.5rem .75rem;margin-left:-1px;line-height:1.25;color:#007bff;background-color:#fff;border:1px solid #dee2e6}.page-link:hover{z-index:2;color:#0056b3;text-decoration:none;background-color:#e9ecef;border-color:#dee2e6}.page-link:focus{z-index:2;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.page-item:first-child .page-link{margin-left:0;border-top-left-radius:.25rem;border-bottom-left-radius:.25rem}.page-item:last-child .page-link{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.page-item.active .page-link{z-index:1;color:#fff;background-color:#007bff;border-color:#007bff}.page-item.disabled .page-link{color:#6c757d;pointer-events:none;cursor:auto;background-color:#fff;border-color:#dee2e6}.pagination-lg .page-link{padding:.75rem 1.5rem;font-size:1.25rem;line-height:1.5}.pagination-lg .page-item:first-child .page-link{border-top-left-radius:.3rem;border-bottom-left-radius:.3rem}.pagination-lg .page-item:last-child .page-link{border-top-right-radius:.3rem;border-bottom-right-radius:.3rem}.pagination-sm .page-link{padding:.25rem .5rem;font-size:.875rem;line-height:1.5}.pagination-sm .page-item:first-child .page-link{border-top-left-radius:.2rem;border-bottom-left-radius:.2rem}.pagination-sm .page-item:last-child .page-link{border-top-right-radius:.2rem;border-bottom-right-radius:.2rem}.badge{display:inline-block;padding:.25em .4em;font-size:75%;font-weight:700;line-height:1;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.badge{transition:none}}a.badge:focus,a.badge:hover{text-decoration:none}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.badge-pill{padding-right:.6em;padding-left:.6em;border-radius:10rem}.badge-primary{color:#fff;background-color:#007bff}a.badge-primary:focus,a.badge-primary:hover{color:#fff;background-color:#0062cc}a.badge-primary.focus,a.badge-primary:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.badge-secondary{color:#fff;background-color:#6c757d}a.badge-secondary:focus,a.badge-secondary:hover{color:#fff;background-color:#545b62}a.badge-secondary.focus,a.badge-secondary:focus{outline:0;box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.badge-success{color:#fff;background-color:#28a745}a.badge-success:focus,a.badge-success:hover{color:#fff;background-color:#1e7e34}a.badge-success.focus,a.badge-success:focus{outline:0;box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.badge-info{color:#fff;background-color:#17a2b8}a.badge-info:focus,a.badge-info:hover{color:#fff;background-color:#117a8b}a.badge-info.focus,a.badge-info:focus{outline:0;box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.badge-warning{color:#212529;background-color:#ffc107}a.badge-warning:focus,a.badge-warning:hover{color:#212529;background-color:#d39e00}a.badge-warning.focus,a.badge-warning:focus{outline:0;box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.badge-danger{color:#fff;background-color:#dc3545}a.badge-danger:focus,a.badge-danger:hover{color:#fff;background-color:#bd2130}a.badge-danger.focus,a.badge-danger:focus{outline:0;box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.badge-light{color:#212529;background-color:#f8f9fa}a.badge-light:focus,a.badge-light:hover{color:#212529;background-color:#dae0e5}a.badge-light.focus,a.badge-light:focus{outline:0;box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.badge-dark{color:#fff;background-color:#343a40}a.badge-dark:focus,a.badge-dark:hover{color:#fff;background-color:#1d2124}a.badge-dark.focus,a.badge-dark:focus{outline:0;box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.jumbotron{padding:2rem 1rem;margin-bottom:2rem;background-color:#e9ecef;border-radius:.3rem}@media (min-width:576px){.jumbotron{padding:4rem 2rem}}.jumbotron-fluid{padding-right:0;padding-left:0;border-radius:0}.alert{position:relative;padding:.75rem 1.25rem;margin-bottom:1rem;border:1px solid transparent;border-radius:.25rem}.alert-heading{color:inherit}.alert-link{font-weight:700}.alert-dismissible{padding-right:4rem}.alert-dismissible .close{position:absolute;top:0;right:0;padding:.75rem 1.25rem;color:inherit}.alert-primary{color:#004085;background-color:#cce5ff;border-color:#b8daff}.alert-primary hr{border-top-color:#9fcdff}.alert-primary .alert-link{color:#002752}.alert-secondary{color:#383d41;background-color:#e2e3e5;border-color:#d6d8db}.alert-secondary hr{border-top-color:#c8cbcf}.alert-secondary .alert-link{color:#202326}.alert-success{color:#155724;background-color:#d4edda;border-color:#c3e6cb}.alert-success hr{border-top-color:#b1dfbb}.alert-success .alert-link{color:#0b2e13}.alert-info{color:#0c5460;background-color:#d1ecf1;border-color:#bee5eb}.alert-info hr{border-top-color:#abdde5}.alert-info .alert-link{color:#062c33}.alert-warning{color:#856404;background-color:#fff3cd;border-color:#ffeeba}.alert-warning hr{border-top-color:#ffe8a1}.alert-warning .alert-link{color:#533f03}.alert-danger{color:#721c24;background-color:#f8d7da;border-color:#f5c6cb}.alert-danger hr{border-top-color:#f1b0b7}.alert-danger .alert-link{color:#491217}.alert-light{color:#818182;background-color:#fefefe;border-color:#fdfdfe}.alert-light hr{border-top-color:#ececf6}.alert-light .alert-link{color:#686868}.alert-dark{color:#1b1e21;background-color:#d6d8d9;border-color:#c6c8ca}.alert-dark hr{border-top-color:#b9bbbe}.alert-dark .alert-link{color:#040505}@-webkit-keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}.progress{display:-ms-flexbox;display:flex;height:1rem;overflow:hidden;font-size:.75rem;background-color:#e9ecef;border-radius:.25rem}.progress-bar{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;color:#fff;text-align:center;white-space:nowrap;background-color:#007bff;transition:width .6s ease}@media (prefers-reduced-motion:reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:1rem 1rem}.progress-bar-animated{-webkit-animation:progress-bar-stripes 1s linear infinite;animation:progress-bar-stripes 1s linear infinite}@media (prefers-reduced-motion:reduce){.progress-bar-animated{-webkit-animation:none;animation:none}}.media{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start}.media-body{-ms-flex:1;flex:1}.list-group{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0}.list-group-item-action{width:100%;color:#495057;text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{z-index:1;color:#495057;text-decoration:none;background-color:#f8f9fa}.list-group-item-action:active{color:#212529;background-color:#e9ecef}.list-group-item{position:relative;display:block;padding:.75rem 1.25rem;margin-bottom:-1px;background-color:#fff;border:1px solid rgba(0,0,0,.125)}.list-group-item:first-child{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.list-group-item.disabled,.list-group-item:disabled{color:#6c757d;pointer-events:none;background-color:#fff}.list-group-item.active{z-index:2;color:#fff;background-color:#007bff;border-color:#007bff}.list-group-horizontal{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal .list-group-item{margin-right:-1px;margin-bottom:0}.list-group-horizontal .list-group-item:first-child{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal .list-group-item:last-child{margin-right:0;border-top-right-radius:.25rem;border-bottom-right-radius:.25rem;border-bottom-left-radius:0}@media (min-width:576px){.list-group-horizontal-sm{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-sm .list-group-item{margin-right:-1px;margin-bottom:0}.list-group-horizontal-sm .list-group-item:first-child{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-sm .list-group-item:last-child{margin-right:0;border-top-right-radius:.25rem;border-bottom-right-radius:.25rem;border-bottom-left-radius:0}}@media (min-width:768px){.list-group-horizontal-md{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-md .list-group-item{margin-right:-1px;margin-bottom:0}.list-group-horizontal-md .list-group-item:first-child{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-md .list-group-item:last-child{margin-right:0;border-top-right-radius:.25rem;border-bottom-right-radius:.25rem;border-bottom-left-radius:0}}@media (min-width:992px){.list-group-horizontal-lg{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-lg .list-group-item{margin-right:-1px;margin-bottom:0}.list-group-horizontal-lg .list-group-item:first-child{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-lg .list-group-item:last-child{margin-right:0;border-top-right-radius:.25rem;border-bottom-right-radius:.25rem;border-bottom-left-radius:0}}@media (min-width:1200px){.list-group-horizontal-xl{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-xl .list-group-item{margin-right:-1px;margin-bottom:0}.list-group-horizontal-xl .list-group-item:first-child{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-xl .list-group-item:last-child{margin-right:0;border-top-right-radius:.25rem;border-bottom-right-radius:.25rem;border-bottom-left-radius:0}}.list-group-flush .list-group-item{border-right:0;border-left:0;border-radius:0}.list-group-flush .list-group-item:last-child{margin-bottom:-1px}.list-group-flush:first-child .list-group-item:first-child{border-top:0}.list-group-flush:last-child .list-group-item:last-child{margin-bottom:0;border-bottom:0}.list-group-item-primary{color:#004085;background-color:#b8daff}.list-group-item-primary.list-group-item-action:focus,.list-group-item-primary.list-group-item-action:hover{color:#004085;background-color:#9fcdff}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#004085;border-color:#004085}.list-group-item-secondary{color:#383d41;background-color:#d6d8db}.list-group-item-secondary.list-group-item-action:focus,.list-group-item-secondary.list-group-item-action:hover{color:#383d41;background-color:#c8cbcf}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#383d41;border-color:#383d41}.list-group-item-success{color:#155724;background-color:#c3e6cb}.list-group-item-success.list-group-item-action:focus,.list-group-item-success.list-group-item-action:hover{color:#155724;background-color:#b1dfbb}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#155724;border-color:#155724}.list-group-item-info{color:#0c5460;background-color:#bee5eb}.list-group-item-info.list-group-item-action:focus,.list-group-item-info.list-group-item-action:hover{color:#0c5460;background-color:#abdde5}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#0c5460;border-color:#0c5460}.list-group-item-warning{color:#856404;background-color:#ffeeba}.list-group-item-warning.list-group-item-action:focus,.list-group-item-warning.list-group-item-action:hover{color:#856404;background-color:#ffe8a1}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#856404;border-color:#856404}.list-group-item-danger{color:#721c24;background-color:#f5c6cb}.list-group-item-danger.list-group-item-action:focus,.list-group-item-danger.list-group-item-action:hover{color:#721c24;background-color:#f1b0b7}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#721c24;border-color:#721c24}.list-group-item-light{color:#818182;background-color:#fdfdfe}.list-group-item-light.list-group-item-action:focus,.list-group-item-light.list-group-item-action:hover{color:#818182;background-color:#ececf6}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#818182;border-color:#818182}.list-group-item-dark{color:#1b1e21;background-color:#c6c8ca}.list-group-item-dark.list-group-item-action:focus,.list-group-item-dark.list-group-item-action:hover{color:#1b1e21;background-color:#b9bbbe}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#1b1e21;border-color:#1b1e21}.close{float:right;font-size:1.5rem;font-weight:700;line-height:1;color:#000;text-shadow:0 1px 0 #fff;opacity:.5}.close:hover{color:#000;text-decoration:none}.close:not(:disabled):not(.disabled):focus,.close:not(:disabled):not(.disabled):hover{opacity:.75}button.close{padding:0;background-color:transparent;border:0;-webkit-appearance:none;-moz-appearance:none;appearance:none}a.close.disabled{pointer-events:none}.toast{max-width:350px;overflow:hidden;font-size:.875rem;background-color:rgba(255,255,255,.85);background-clip:padding-box;border:1px solid rgba(0,0,0,.1);box-shadow:0 .25rem .75rem rgba(0,0,0,.1);-webkit-backdrop-filter:blur(10px);backdrop-filter:blur(10px);opacity:0;border-radius:.25rem}.toast:not(:last-child){margin-bottom:.75rem}.toast.showing{opacity:1}.toast.show{display:block;opacity:1}.toast.hide{display:none}.toast-header{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.25rem .75rem;color:#6c757d;background-color:rgba(255,255,255,.85);background-clip:padding-box;border-bottom:1px solid rgba(0,0,0,.05)}.toast-body{padding:.75rem}.modal-open{overflow:hidden}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal{position:fixed;top:0;left:0;z-index:1050;display:none;width:100%;height:100%;overflow:hidden;outline:0}.modal-dialog{position:relative;width:auto;margin:.5rem;pointer-events:none}.modal.fade .modal-dialog{transition:-webkit-transform .3s ease-out;transition:transform .3s ease-out;transition:transform .3s ease-out,-webkit-transform .3s ease-out;-webkit-transform:translate(0,-50px);transform:translate(0,-50px)}@media (prefers-reduced-motion:reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{-webkit-transform:none;transform:none}.modal-dialog-scrollable{display:-ms-flexbox;display:flex;max-height:calc(100% - 1rem)}.modal-dialog-scrollable .modal-content{max-height:calc(100vh - 1rem);overflow:hidden}.modal-dialog-scrollable .modal-footer,.modal-dialog-scrollable .modal-header{-ms-flex-negative:0;flex-shrink:0}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;min-height:calc(100% - 1rem)}.modal-dialog-centered::before{display:block;height:calc(100vh - 1rem);content:""}.modal-dialog-centered.modal-dialog-scrollable{-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;height:100%}.modal-dialog-centered.modal-dialog-scrollable .modal-content{max-height:none}.modal-dialog-centered.modal-dialog-scrollable::before{content:none}.modal-content{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;width:100%;pointer-events:auto;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem;outline:0}.modal-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:.5}.modal-header{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:justify;justify-content:space-between;padding:1rem 1rem;border-bottom:1px solid #dee2e6;border-top-left-radius:.3rem;border-top-right-radius:.3rem}.modal-header .close{padding:1rem 1rem;margin:-1rem -1rem -1rem auto}.modal-title{margin-bottom:0;line-height:1.5}.modal-body{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;padding:1rem}.modal-footer{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:end;justify-content:flex-end;padding:1rem;border-top:1px solid #dee2e6;border-bottom-right-radius:.3rem;border-bottom-left-radius:.3rem}.modal-footer>:not(:first-child){margin-left:.25rem}.modal-footer>:not(:last-child){margin-right:.25rem}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width:576px){.modal-dialog{max-width:500px;margin:1.75rem auto}.modal-dialog-scrollable{max-height:calc(100% - 3.5rem)}.modal-dialog-scrollable .modal-content{max-height:calc(100vh - 3.5rem)}.modal-dialog-centered{min-height:calc(100% - 3.5rem)}.modal-dialog-centered::before{height:calc(100vh - 3.5rem)}.modal-sm{max-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{max-width:800px}}@media (min-width:1200px){.modal-xl{max-width:1140px}}.tooltip{position:absolute;z-index:1070;display:block;margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;opacity:0}.tooltip.show{opacity:.9}.tooltip .arrow{position:absolute;display:block;width:.8rem;height:.4rem}.tooltip .arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[x-placement^=top],.bs-tooltip-top{padding:.4rem 0}.bs-tooltip-auto[x-placement^=top] .arrow,.bs-tooltip-top .arrow{bottom:0}.bs-tooltip-auto[x-placement^=top] .arrow::before,.bs-tooltip-top .arrow::before{top:0;border-width:.4rem .4rem 0;border-top-color:#000}.bs-tooltip-auto[x-placement^=right],.bs-tooltip-right{padding:0 .4rem}.bs-tooltip-auto[x-placement^=right] .arrow,.bs-tooltip-right .arrow{left:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=right] .arrow::before,.bs-tooltip-right .arrow::before{right:0;border-width:.4rem .4rem .4rem 0;border-right-color:#000}.bs-tooltip-auto[x-placement^=bottom],.bs-tooltip-bottom{padding:.4rem 0}.bs-tooltip-auto[x-placement^=bottom] .arrow,.bs-tooltip-bottom .arrow{top:0}.bs-tooltip-auto[x-placement^=bottom] .arrow::before,.bs-tooltip-bottom .arrow::before{bottom:0;border-width:0 .4rem .4rem;border-bottom-color:#000}.bs-tooltip-auto[x-placement^=left],.bs-tooltip-left{padding:0 .4rem}.bs-tooltip-auto[x-placement^=left] .arrow,.bs-tooltip-left .arrow{right:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=left] .arrow::before,.bs-tooltip-left .arrow::before{left:0;border-width:.4rem 0 .4rem .4rem;border-left-color:#000}.tooltip-inner{max-width:200px;padding:.25rem .5rem;color:#fff;text-align:center;background-color:#000;border-radius:.25rem}.popover{position:absolute;top:0;left:0;z-index:1060;display:block;max-width:276px;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem}.popover .arrow{position:absolute;display:block;width:1rem;height:.5rem;margin:0 .3rem}.popover .arrow::after,.popover .arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid}.bs-popover-auto[x-placement^=top],.bs-popover-top{margin-bottom:.5rem}.bs-popover-auto[x-placement^=top]>.arrow,.bs-popover-top>.arrow{bottom:calc((.5rem + 1px) * -1)}.bs-popover-auto[x-placement^=top]>.arrow::before,.bs-popover-top>.arrow::before{bottom:0;border-width:.5rem .5rem 0;border-top-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=top]>.arrow::after,.bs-popover-top>.arrow::after{bottom:1px;border-width:.5rem .5rem 0;border-top-color:#fff}.bs-popover-auto[x-placement^=right],.bs-popover-right{margin-left:.5rem}.bs-popover-auto[x-placement^=right]>.arrow,.bs-popover-right>.arrow{left:calc((.5rem + 1px) * -1);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=right]>.arrow::before,.bs-popover-right>.arrow::before{left:0;border-width:.5rem .5rem .5rem 0;border-right-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=right]>.arrow::after,.bs-popover-right>.arrow::after{left:1px;border-width:.5rem .5rem .5rem 0;border-right-color:#fff}.bs-popover-auto[x-placement^=bottom],.bs-popover-bottom{margin-top:.5rem}.bs-popover-auto[x-placement^=bottom]>.arrow,.bs-popover-bottom>.arrow{top:calc((.5rem + 1px) * -1)}.bs-popover-auto[x-placement^=bottom]>.arrow::before,.bs-popover-bottom>.arrow::before{top:0;border-width:0 .5rem .5rem .5rem;border-bottom-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=bottom]>.arrow::after,.bs-popover-bottom>.arrow::after{top:1px;border-width:0 .5rem .5rem .5rem;border-bottom-color:#fff}.bs-popover-auto[x-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:1rem;margin-left:-.5rem;content:"";border-bottom:1px solid #f7f7f7}.bs-popover-auto[x-placement^=left],.bs-popover-left{margin-right:.5rem}.bs-popover-auto[x-placement^=left]>.arrow,.bs-popover-left>.arrow{right:calc((.5rem + 1px) * -1);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=left]>.arrow::before,.bs-popover-left>.arrow::before{right:0;border-width:.5rem 0 .5rem .5rem;border-left-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=left]>.arrow::after,.bs-popover-left>.arrow::after{right:1px;border-width:.5rem 0 .5rem .5rem;border-left-color:#fff}.popover-header{padding:.5rem .75rem;margin-bottom:0;font-size:1rem;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.popover-header:empty{display:none}.popover-body{padding:.5rem .75rem;color:#212529}.carousel{position:relative}.carousel.pointer-event{-ms-touch-action:pan-y;touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden;transition:-webkit-transform .6s ease-in-out;transition:transform .6s ease-in-out;transition:transform .6s ease-in-out,-webkit-transform .6s ease-in-out}@media (prefers-reduced-motion:reduce){.carousel-item{transition:none}}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.active.carousel-item-right,.carousel-item-next:not(.carousel-item-left){-webkit-transform:translateX(100%);transform:translateX(100%)}.active.carousel-item-left,.carousel-item-prev:not(.carousel-item-right){-webkit-transform:translateX(-100%);transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;-webkit-transform:none;transform:none}.carousel-fade .carousel-item-next.carousel-item-left,.carousel-fade .carousel-item-prev.carousel-item-right,.carousel-fade .carousel-item.active{z-index:1;opacity:1}.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{z-index:0;opacity:0;transition:0s .6s opacity}@media (prefers-reduced-motion:reduce){.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{transition:none}}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;z-index:1;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:15%;color:#fff;text-align:center;opacity:.5;transition:opacity .15s ease}@media (prefers-reduced-motion:reduce){.carousel-control-next,.carousel-control-prev{transition:none}}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:20px;height:20px;background:no-repeat 50%/100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 8 8'%3e%3cpath d='M5.25 0l-4 4 4 4 1.5-1.5-2.5-2.5 2.5-2.5-1.5-1.5z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 8 8'%3e%3cpath d='M2.75 0l-1.5 1.5 2.5 2.5-2.5 2.5 1.5 1.5 4-4-4-4z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:15;display:-ms-flexbox;display:flex;-ms-flex-pack:center;justify-content:center;padding-left:0;margin-right:15%;margin-left:15%;list-style:none}.carousel-indicators li{box-sizing:content-box;-ms-flex:0 1 auto;flex:0 1 auto;width:30px;height:3px;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5;transition:opacity .6s ease}@media (prefers-reduced-motion:reduce){.carousel-indicators li{transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center}@-webkit-keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}.spinner-border{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;border:.25em solid currentColor;border-right-color:transparent;border-radius:50%;-webkit-animation:spinner-border .75s linear infinite;animation:spinner-border .75s linear infinite}.spinner-border-sm{width:1rem;height:1rem;border-width:.2em}@-webkit-keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1}}@keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1}}.spinner-grow{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;background-color:currentColor;border-radius:50%;opacity:0;-webkit-animation:spinner-grow .75s linear infinite;animation:spinner-grow .75s linear infinite}.spinner-grow-sm{width:1rem;height:1rem}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.bg-primary{background-color:#007bff!important}a.bg-primary:focus,a.bg-primary:hover,button.bg-primary:focus,button.bg-primary:hover{background-color:#0062cc!important}.bg-secondary{background-color:#6c757d!important}a.bg-secondary:focus,a.bg-secondary:hover,button.bg-secondary:focus,button.bg-secondary:hover{background-color:#545b62!important}.bg-success{background-color:#28a745!important}a.bg-success:focus,a.bg-success:hover,button.bg-success:focus,button.bg-success:hover{background-color:#1e7e34!important}.bg-info{background-color:#17a2b8!important}a.bg-info:focus,a.bg-info:hover,button.bg-info:focus,button.bg-info:hover{background-color:#117a8b!important}.bg-warning{background-color:#ffc107!important}a.bg-warning:focus,a.bg-warning:hover,button.bg-warning:focus,button.bg-warning:hover{background-color:#d39e00!important}.bg-danger{background-color:#dc3545!important}a.bg-danger:focus,a.bg-danger:hover,button.bg-danger:focus,button.bg-danger:hover{background-color:#bd2130!important}.bg-light{background-color:#f8f9fa!important}a.bg-light:focus,a.bg-light:hover,button.bg-light:focus,button.bg-light:hover{background-color:#dae0e5!important}.bg-dark{background-color:#343a40!important}a.bg-dark:focus,a.bg-dark:hover,button.bg-dark:focus,button.bg-dark:hover{background-color:#1d2124!important}.bg-white{background-color:#fff!important}.bg-transparent{background-color:transparent!important}.border{border:1px solid #dee2e6!important}.border-top{border-top:1px solid #dee2e6!important}.border-right{border-right:1px solid #dee2e6!important}.border-bottom{border-bottom:1px solid #dee2e6!important}.border-left{border-left:1px solid #dee2e6!important}.border-0{border:0!important}.border-top-0{border-top:0!important}.border-right-0{border-right:0!important}.border-bottom-0{border-bottom:0!important}.border-left-0{border-left:0!important}.border-primary{border-color:#007bff!important}.border-secondary{border-color:#6c757d!important}.border-success{border-color:#28a745!important}.border-info{border-color:#17a2b8!important}.border-warning{border-color:#ffc107!important}.border-danger{border-color:#dc3545!important}.border-light{border-color:#f8f9fa!important}.border-dark{border-color:#343a40!important}.border-white{border-color:#fff!important}.rounded-sm{border-radius:.2rem!important}.rounded{border-radius:.25rem!important}.rounded-top{border-top-left-radius:.25rem!important;border-top-right-radius:.25rem!important}.rounded-right{border-top-right-radius:.25rem!important;border-bottom-right-radius:.25rem!important}.rounded-bottom{border-bottom-right-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-left{border-top-left-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-lg{border-radius:.3rem!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:50rem!important}.rounded-0{border-radius:0!important}.clearfix::after{display:block;clear:both;content:""}.d-none{display:none!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:-ms-flexbox!important;display:flex!important}.d-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}@media (min-width:576px){.d-sm-none{display:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:-ms-flexbox!important;display:flex!important}.d-sm-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:768px){.d-md-none{display:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:-ms-flexbox!important;display:flex!important}.d-md-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:992px){.d-lg-none{display:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:-ms-flexbox!important;display:flex!important}.d-lg-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:1200px){.d-xl-none{display:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:-ms-flexbox!important;display:flex!important}.d-xl-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media print{.d-print-none{display:none!important}.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:-ms-flexbox!important;display:flex!important}.d-print-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}.embed-responsive{position:relative;display:block;width:100%;padding:0;overflow:hidden}.embed-responsive::before{display:block;content:""}.embed-responsive .embed-responsive-item,.embed-responsive embed,.embed-responsive iframe,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-21by9::before{padding-top:42.857143%}.embed-responsive-16by9::before{padding-top:56.25%}.embed-responsive-4by3::before{padding-top:75%}.embed-responsive-1by1::before{padding-top:100%}.flex-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-center{-ms-flex-align:center!important;align-items:center!important}.align-items-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}@media (min-width:576px){.flex-sm-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-sm-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-sm-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-sm-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-sm-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-sm-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-sm-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-sm-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-sm-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-sm-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-sm-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-sm-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-sm-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-sm-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-sm-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-sm-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-sm-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-sm-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-sm-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-sm-center{-ms-flex-align:center!important;align-items:center!important}.align-items-sm-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-sm-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-sm-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-sm-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-sm-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-sm-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-sm-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-sm-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-sm-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-sm-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-sm-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-sm-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-sm-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-sm-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:768px){.flex-md-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-md-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-md-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-md-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-md-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-md-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-md-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-md-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-md-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-md-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-md-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-md-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-md-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-md-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-md-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-md-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-md-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-md-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-md-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-md-center{-ms-flex-align:center!important;align-items:center!important}.align-items-md-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-md-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-md-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-md-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-md-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-md-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-md-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-md-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-md-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-md-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-md-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-md-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-md-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-md-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:992px){.flex-lg-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-lg-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-lg-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-lg-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-lg-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-lg-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-lg-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-lg-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-lg-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-lg-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-lg-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-lg-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-lg-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-lg-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-lg-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-lg-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-lg-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-lg-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-lg-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-lg-center{-ms-flex-align:center!important;align-items:center!important}.align-items-lg-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-lg-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-lg-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-lg-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-lg-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-lg-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-lg-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-lg-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-lg-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-lg-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-lg-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-lg-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-lg-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-lg-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:1200px){.flex-xl-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-xl-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-xl-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-xl-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-xl-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-xl-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-xl-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-xl-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-xl-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-xl-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-xl-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-xl-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-xl-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-xl-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-xl-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-xl-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-xl-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-xl-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-xl-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-xl-center{-ms-flex-align:center!important;align-items:center!important}.align-items-xl-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-xl-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-xl-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-xl-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-xl-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-xl-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-xl-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-xl-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-xl-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-xl-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-xl-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-xl-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-xl-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-xl-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}.float-left{float:left!important}.float-right{float:right!important}.float-none{float:none!important}@media (min-width:576px){.float-sm-left{float:left!important}.float-sm-right{float:right!important}.float-sm-none{float:none!important}}@media (min-width:768px){.float-md-left{float:left!important}.float-md-right{float:right!important}.float-md-none{float:none!important}}@media (min-width:992px){.float-lg-left{float:left!important}.float-lg-right{float:right!important}.float-lg-none{float:none!important}}@media (min-width:1200px){.float-xl-left{float:left!important}.float-xl-right{float:right!important}.float-xl-none{float:none!important}}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}@supports ((position:-webkit-sticky) or (position:sticky)){.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}.sr-only{position:absolute;width:1px;height:1px;padding:0;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;overflow:visible;clip:auto;white-space:normal}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075)!important}.shadow{box-shadow:0 .5rem 1rem rgba(0,0,0,.15)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175)!important}.shadow-none{box-shadow:none!important}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mw-100{max-width:100%!important}.mh-100{max-height:100%!important}.min-vw-100{min-width:100vw!important}.min-vh-100{min-height:100vh!important}.vw-100{width:100vw!important}.vh-100{height:100vh!important}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;pointer-events:auto;content:"";background-color:rgba(0,0,0,0)}.m-0{margin:0!important}.mt-0,.my-0{margin-top:0!important}.mr-0,.mx-0{margin-right:0!important}.mb-0,.my-0{margin-bottom:0!important}.ml-0,.mx-0{margin-left:0!important}.m-1{margin:.25rem!important}.mt-1,.my-1{margin-top:.25rem!important}.mr-1,.mx-1{margin-right:.25rem!important}.mb-1,.my-1{margin-bottom:.25rem!important}.ml-1,.mx-1{margin-left:.25rem!important}.m-2{margin:.5rem!important}.mt-2,.my-2{margin-top:.5rem!important}.mr-2,.mx-2{margin-right:.5rem!important}.mb-2,.my-2{margin-bottom:.5rem!important}.ml-2,.mx-2{margin-left:.5rem!important}.m-3{margin:1rem!important}.mt-3,.my-3{margin-top:1rem!important}.mr-3,.mx-3{margin-right:1rem!important}.mb-3,.my-3{margin-bottom:1rem!important}.ml-3,.mx-3{margin-left:1rem!important}.m-4{margin:1.5rem!important}.mt-4,.my-4{margin-top:1.5rem!important}.mr-4,.mx-4{margin-right:1.5rem!important}.mb-4,.my-4{margin-bottom:1.5rem!important}.ml-4,.mx-4{margin-left:1.5rem!important}.m-5{margin:3rem!important}.mt-5,.my-5{margin-top:3rem!important}.mr-5,.mx-5{margin-right:3rem!important}.mb-5,.my-5{margin-bottom:3rem!important}.ml-5,.mx-5{margin-left:3rem!important}.p-0{padding:0!important}.pt-0,.py-0{padding-top:0!important}.pr-0,.px-0{padding-right:0!important}.pb-0,.py-0{padding-bottom:0!important}.pl-0,.px-0{padding-left:0!important}.p-1{padding:.25rem!important}.pt-1,.py-1{padding-top:.25rem!important}.pr-1,.px-1{padding-right:.25rem!important}.pb-1,.py-1{padding-bottom:.25rem!important}.pl-1,.px-1{padding-left:.25rem!important}.p-2{padding:.5rem!important}.pt-2,.py-2{padding-top:.5rem!important}.pr-2,.px-2{padding-right:.5rem!important}.pb-2,.py-2{padding-bottom:.5rem!important}.pl-2,.px-2{padding-left:.5rem!important}.p-3{padding:1rem!important}.pt-3,.py-3{padding-top:1rem!important}.pr-3,.px-3{padding-right:1rem!important}.pb-3,.py-3{padding-bottom:1rem!important}.pl-3,.px-3{padding-left:1rem!important}.p-4{padding:1.5rem!important}.pt-4,.py-4{padding-top:1.5rem!important}.pr-4,.px-4{padding-right:1.5rem!important}.pb-4,.py-4{padding-bottom:1.5rem!important}.pl-4,.px-4{padding-left:1.5rem!important}.p-5{padding:3rem!important}.pt-5,.py-5{padding-top:3rem!important}.pr-5,.px-5{padding-right:3rem!important}.pb-5,.py-5{padding-bottom:3rem!important}.pl-5,.px-5{padding-left:3rem!important}.m-n1{margin:-.25rem!important}.mt-n1,.my-n1{margin-top:-.25rem!important}.mr-n1,.mx-n1{margin-right:-.25rem!important}.mb-n1,.my-n1{margin-bottom:-.25rem!important}.ml-n1,.mx-n1{margin-left:-.25rem!important}.m-n2{margin:-.5rem!important}.mt-n2,.my-n2{margin-top:-.5rem!important}.mr-n2,.mx-n2{margin-right:-.5rem!important}.mb-n2,.my-n2{margin-bottom:-.5rem!important}.ml-n2,.mx-n2{margin-left:-.5rem!important}.m-n3{margin:-1rem!important}.mt-n3,.my-n3{margin-top:-1rem!important}.mr-n3,.mx-n3{margin-right:-1rem!important}.mb-n3,.my-n3{margin-bottom:-1rem!important}.ml-n3,.mx-n3{margin-left:-1rem!important}.m-n4{margin:-1.5rem!important}.mt-n4,.my-n4{margin-top:-1.5rem!important}.mr-n4,.mx-n4{margin-right:-1.5rem!important}.mb-n4,.my-n4{margin-bottom:-1.5rem!important}.ml-n4,.mx-n4{margin-left:-1.5rem!important}.m-n5{margin:-3rem!important}.mt-n5,.my-n5{margin-top:-3rem!important}.mr-n5,.mx-n5{margin-right:-3rem!important}.mb-n5,.my-n5{margin-bottom:-3rem!important}.ml-n5,.mx-n5{margin-left:-3rem!important}.m-auto{margin:auto!important}.mt-auto,.my-auto{margin-top:auto!important}.mr-auto,.mx-auto{margin-right:auto!important}.mb-auto,.my-auto{margin-bottom:auto!important}.ml-auto,.mx-auto{margin-left:auto!important}@media (min-width:576px){.m-sm-0{margin:0!important}.mt-sm-0,.my-sm-0{margin-top:0!important}.mr-sm-0,.mx-sm-0{margin-right:0!important}.mb-sm-0,.my-sm-0{margin-bottom:0!important}.ml-sm-0,.mx-sm-0{margin-left:0!important}.m-sm-1{margin:.25rem!important}.mt-sm-1,.my-sm-1{margin-top:.25rem!important}.mr-sm-1,.mx-sm-1{margin-right:.25rem!important}.mb-sm-1,.my-sm-1{margin-bottom:.25rem!important}.ml-sm-1,.mx-sm-1{margin-left:.25rem!important}.m-sm-2{margin:.5rem!important}.mt-sm-2,.my-sm-2{margin-top:.5rem!important}.mr-sm-2,.mx-sm-2{margin-right:.5rem!important}.mb-sm-2,.my-sm-2{margin-bottom:.5rem!important}.ml-sm-2,.mx-sm-2{margin-left:.5rem!important}.m-sm-3{margin:1rem!important}.mt-sm-3,.my-sm-3{margin-top:1rem!important}.mr-sm-3,.mx-sm-3{margin-right:1rem!important}.mb-sm-3,.my-sm-3{margin-bottom:1rem!important}.ml-sm-3,.mx-sm-3{margin-left:1rem!important}.m-sm-4{margin:1.5rem!important}.mt-sm-4,.my-sm-4{margin-top:1.5rem!important}.mr-sm-4,.mx-sm-4{margin-right:1.5rem!important}.mb-sm-4,.my-sm-4{margin-bottom:1.5rem!important}.ml-sm-4,.mx-sm-4{margin-left:1.5rem!important}.m-sm-5{margin:3rem!important}.mt-sm-5,.my-sm-5{margin-top:3rem!important}.mr-sm-5,.mx-sm-5{margin-right:3rem!important}.mb-sm-5,.my-sm-5{margin-bottom:3rem!important}.ml-sm-5,.mx-sm-5{margin-left:3rem!important}.p-sm-0{padding:0!important}.pt-sm-0,.py-sm-0{padding-top:0!important}.pr-sm-0,.px-sm-0{padding-right:0!important}.pb-sm-0,.py-sm-0{padding-bottom:0!important}.pl-sm-0,.px-sm-0{padding-left:0!important}.p-sm-1{padding:.25rem!important}.pt-sm-1,.py-sm-1{padding-top:.25rem!important}.pr-sm-1,.px-sm-1{padding-right:.25rem!important}.pb-sm-1,.py-sm-1{padding-bottom:.25rem!important}.pl-sm-1,.px-sm-1{padding-left:.25rem!important}.p-sm-2{padding:.5rem!important}.pt-sm-2,.py-sm-2{padding-top:.5rem!important}.pr-sm-2,.px-sm-2{padding-right:.5rem!important}.pb-sm-2,.py-sm-2{padding-bottom:.5rem!important}.pl-sm-2,.px-sm-2{padding-left:.5rem!important}.p-sm-3{padding:1rem!important}.pt-sm-3,.py-sm-3{padding-top:1rem!important}.pr-sm-3,.px-sm-3{padding-right:1rem!important}.pb-sm-3,.py-sm-3{padding-bottom:1rem!important}.pl-sm-3,.px-sm-3{padding-left:1rem!important}.p-sm-4{padding:1.5rem!important}.pt-sm-4,.py-sm-4{padding-top:1.5rem!important}.pr-sm-4,.px-sm-4{padding-right:1.5rem!important}.pb-sm-4,.py-sm-4{padding-bottom:1.5rem!important}.pl-sm-4,.px-sm-4{padding-left:1.5rem!important}.p-sm-5{padding:3rem!important}.pt-sm-5,.py-sm-5{padding-top:3rem!important}.pr-sm-5,.px-sm-5{padding-right:3rem!important}.pb-sm-5,.py-sm-5{padding-bottom:3rem!important}.pl-sm-5,.px-sm-5{padding-left:3rem!important}.m-sm-n1{margin:-.25rem!important}.mt-sm-n1,.my-sm-n1{margin-top:-.25rem!important}.mr-sm-n1,.mx-sm-n1{margin-right:-.25rem!important}.mb-sm-n1,.my-sm-n1{margin-bottom:-.25rem!important}.ml-sm-n1,.mx-sm-n1{margin-left:-.25rem!important}.m-sm-n2{margin:-.5rem!important}.mt-sm-n2,.my-sm-n2{margin-top:-.5rem!important}.mr-sm-n2,.mx-sm-n2{margin-right:-.5rem!important}.mb-sm-n2,.my-sm-n2{margin-bottom:-.5rem!important}.ml-sm-n2,.mx-sm-n2{margin-left:-.5rem!important}.m-sm-n3{margin:-1rem!important}.mt-sm-n3,.my-sm-n3{margin-top:-1rem!important}.mr-sm-n3,.mx-sm-n3{margin-right:-1rem!important}.mb-sm-n3,.my-sm-n3{margin-bottom:-1rem!important}.ml-sm-n3,.mx-sm-n3{margin-left:-1rem!important}.m-sm-n4{margin:-1.5rem!important}.mt-sm-n4,.my-sm-n4{margin-top:-1.5rem!important}.mr-sm-n4,.mx-sm-n4{margin-right:-1.5rem!important}.mb-sm-n4,.my-sm-n4{margin-bottom:-1.5rem!important}.ml-sm-n4,.mx-sm-n4{margin-left:-1.5rem!important}.m-sm-n5{margin:-3rem!important}.mt-sm-n5,.my-sm-n5{margin-top:-3rem!important}.mr-sm-n5,.mx-sm-n5{margin-right:-3rem!important}.mb-sm-n5,.my-sm-n5{margin-bottom:-3rem!important}.ml-sm-n5,.mx-sm-n5{margin-left:-3rem!important}.m-sm-auto{margin:auto!important}.mt-sm-auto,.my-sm-auto{margin-top:auto!important}.mr-sm-auto,.mx-sm-auto{margin-right:auto!important}.mb-sm-auto,.my-sm-auto{margin-bottom:auto!important}.ml-sm-auto,.mx-sm-auto{margin-left:auto!important}}@media (min-width:768px){.m-md-0{margin:0!important}.mt-md-0,.my-md-0{margin-top:0!important}.mr-md-0,.mx-md-0{margin-right:0!important}.mb-md-0,.my-md-0{margin-bottom:0!important}.ml-md-0,.mx-md-0{margin-left:0!important}.m-md-1{margin:.25rem!important}.mt-md-1,.my-md-1{margin-top:.25rem!important}.mr-md-1,.mx-md-1{margin-right:.25rem!important}.mb-md-1,.my-md-1{margin-bottom:.25rem!important}.ml-md-1,.mx-md-1{margin-left:.25rem!important}.m-md-2{margin:.5rem!important}.mt-md-2,.my-md-2{margin-top:.5rem!important}.mr-md-2,.mx-md-2{margin-right:.5rem!important}.mb-md-2,.my-md-2{margin-bottom:.5rem!important}.ml-md-2,.mx-md-2{margin-left:.5rem!important}.m-md-3{margin:1rem!important}.mt-md-3,.my-md-3{margin-top:1rem!important}.mr-md-3,.mx-md-3{margin-right:1rem!important}.mb-md-3,.my-md-3{margin-bottom:1rem!important}.ml-md-3,.mx-md-3{margin-left:1rem!important}.m-md-4{margin:1.5rem!important}.mt-md-4,.my-md-4{margin-top:1.5rem!important}.mr-md-4,.mx-md-4{margin-right:1.5rem!important}.mb-md-4,.my-md-4{margin-bottom:1.5rem!important}.ml-md-4,.mx-md-4{margin-left:1.5rem!important}.m-md-5{margin:3rem!important}.mt-md-5,.my-md-5{margin-top:3rem!important}.mr-md-5,.mx-md-5{margin-right:3rem!important}.mb-md-5,.my-md-5{margin-bottom:3rem!important}.ml-md-5,.mx-md-5{margin-left:3rem!important}.p-md-0{padding:0!important}.pt-md-0,.py-md-0{padding-top:0!important}.pr-md-0,.px-md-0{padding-right:0!important}.pb-md-0,.py-md-0{padding-bottom:0!important}.pl-md-0,.px-md-0{padding-left:0!important}.p-md-1{padding:.25rem!important}.pt-md-1,.py-md-1{padding-top:.25rem!important}.pr-md-1,.px-md-1{padding-right:.25rem!important}.pb-md-1,.py-md-1{padding-bottom:.25rem!important}.pl-md-1,.px-md-1{padding-left:.25rem!important}.p-md-2{padding:.5rem!important}.pt-md-2,.py-md-2{padding-top:.5rem!important}.pr-md-2,.px-md-2{padding-right:.5rem!important}.pb-md-2,.py-md-2{padding-bottom:.5rem!important}.pl-md-2,.px-md-2{padding-left:.5rem!important}.p-md-3{padding:1rem!important}.pt-md-3,.py-md-3{padding-top:1rem!important}.pr-md-3,.px-md-3{padding-right:1rem!important}.pb-md-3,.py-md-3{padding-bottom:1rem!important}.pl-md-3,.px-md-3{padding-left:1rem!important}.p-md-4{padding:1.5rem!important}.pt-md-4,.py-md-4{padding-top:1.5rem!important}.pr-md-4,.px-md-4{padding-right:1.5rem!important}.pb-md-4,.py-md-4{padding-bottom:1.5rem!important}.pl-md-4,.px-md-4{padding-left:1.5rem!important}.p-md-5{padding:3rem!important}.pt-md-5,.py-md-5{padding-top:3rem!important}.pr-md-5,.px-md-5{padding-right:3rem!important}.pb-md-5,.py-md-5{padding-bottom:3rem!important}.pl-md-5,.px-md-5{padding-left:3rem!important}.m-md-n1{margin:-.25rem!important}.mt-md-n1,.my-md-n1{margin-top:-.25rem!important}.mr-md-n1,.mx-md-n1{margin-right:-.25rem!important}.mb-md-n1,.my-md-n1{margin-bottom:-.25rem!important}.ml-md-n1,.mx-md-n1{margin-left:-.25rem!important}.m-md-n2{margin:-.5rem!important}.mt-md-n2,.my-md-n2{margin-top:-.5rem!important}.mr-md-n2,.mx-md-n2{margin-right:-.5rem!important}.mb-md-n2,.my-md-n2{margin-bottom:-.5rem!important}.ml-md-n2,.mx-md-n2{margin-left:-.5rem!important}.m-md-n3{margin:-1rem!important}.mt-md-n3,.my-md-n3{margin-top:-1rem!important}.mr-md-n3,.mx-md-n3{margin-right:-1rem!important}.mb-md-n3,.my-md-n3{margin-bottom:-1rem!important}.ml-md-n3,.mx-md-n3{margin-left:-1rem!important}.m-md-n4{margin:-1.5rem!important}.mt-md-n4,.my-md-n4{margin-top:-1.5rem!important}.mr-md-n4,.mx-md-n4{margin-right:-1.5rem!important}.mb-md-n4,.my-md-n4{margin-bottom:-1.5rem!important}.ml-md-n4,.mx-md-n4{margin-left:-1.5rem!important}.m-md-n5{margin:-3rem!important}.mt-md-n5,.my-md-n5{margin-top:-3rem!important}.mr-md-n5,.mx-md-n5{margin-right:-3rem!important}.mb-md-n5,.my-md-n5{margin-bottom:-3rem!important}.ml-md-n5,.mx-md-n5{margin-left:-3rem!important}.m-md-auto{margin:auto!important}.mt-md-auto,.my-md-auto{margin-top:auto!important}.mr-md-auto,.mx-md-auto{margin-right:auto!important}.mb-md-auto,.my-md-auto{margin-bottom:auto!important}.ml-md-auto,.mx-md-auto{margin-left:auto!important}}@media (min-width:992px){.m-lg-0{margin:0!important}.mt-lg-0,.my-lg-0{margin-top:0!important}.mr-lg-0,.mx-lg-0{margin-right:0!important}.mb-lg-0,.my-lg-0{margin-bottom:0!important}.ml-lg-0,.mx-lg-0{margin-left:0!important}.m-lg-1{margin:.25rem!important}.mt-lg-1,.my-lg-1{margin-top:.25rem!important}.mr-lg-1,.mx-lg-1{margin-right:.25rem!important}.mb-lg-1,.my-lg-1{margin-bottom:.25rem!important}.ml-lg-1,.mx-lg-1{margin-left:.25rem!important}.m-lg-2{margin:.5rem!important}.mt-lg-2,.my-lg-2{margin-top:.5rem!important}.mr-lg-2,.mx-lg-2{margin-right:.5rem!important}.mb-lg-2,.my-lg-2{margin-bottom:.5rem!important}.ml-lg-2,.mx-lg-2{margin-left:.5rem!important}.m-lg-3{margin:1rem!important}.mt-lg-3,.my-lg-3{margin-top:1rem!important}.mr-lg-3,.mx-lg-3{margin-right:1rem!important}.mb-lg-3,.my-lg-3{margin-bottom:1rem!important}.ml-lg-3,.mx-lg-3{margin-left:1rem!important}.m-lg-4{margin:1.5rem!important}.mt-lg-4,.my-lg-4{margin-top:1.5rem!important}.mr-lg-4,.mx-lg-4{margin-right:1.5rem!important}.mb-lg-4,.my-lg-4{margin-bottom:1.5rem!important}.ml-lg-4,.mx-lg-4{margin-left:1.5rem!important}.m-lg-5{margin:3rem!important}.mt-lg-5,.my-lg-5{margin-top:3rem!important}.mr-lg-5,.mx-lg-5{margin-right:3rem!important}.mb-lg-5,.my-lg-5{margin-bottom:3rem!important}.ml-lg-5,.mx-lg-5{margin-left:3rem!important}.p-lg-0{padding:0!important}.pt-lg-0,.py-lg-0{padding-top:0!important}.pr-lg-0,.px-lg-0{padding-right:0!important}.pb-lg-0,.py-lg-0{padding-bottom:0!important}.pl-lg-0,.px-lg-0{padding-left:0!important}.p-lg-1{padding:.25rem!important}.pt-lg-1,.py-lg-1{padding-top:.25rem!important}.pr-lg-1,.px-lg-1{padding-right:.25rem!important}.pb-lg-1,.py-lg-1{padding-bottom:.25rem!important}.pl-lg-1,.px-lg-1{padding-left:.25rem!important}.p-lg-2{padding:.5rem!important}.pt-lg-2,.py-lg-2{padding-top:.5rem!important}.pr-lg-2,.px-lg-2{padding-right:.5rem!important}.pb-lg-2,.py-lg-2{padding-bottom:.5rem!important}.pl-lg-2,.px-lg-2{padding-left:.5rem!important}.p-lg-3{padding:1rem!important}.pt-lg-3,.py-lg-3{padding-top:1rem!important}.pr-lg-3,.px-lg-3{padding-right:1rem!important}.pb-lg-3,.py-lg-3{padding-bottom:1rem!important}.pl-lg-3,.px-lg-3{padding-left:1rem!important}.p-lg-4{padding:1.5rem!important}.pt-lg-4,.py-lg-4{padding-top:1.5rem!important}.pr-lg-4,.px-lg-4{padding-right:1.5rem!important}.pb-lg-4,.py-lg-4{padding-bottom:1.5rem!important}.pl-lg-4,.px-lg-4{padding-left:1.5rem!important}.p-lg-5{padding:3rem!important}.pt-lg-5,.py-lg-5{padding-top:3rem!important}.pr-lg-5,.px-lg-5{padding-right:3rem!important}.pb-lg-5,.py-lg-5{padding-bottom:3rem!important}.pl-lg-5,.px-lg-5{padding-left:3rem!important}.m-lg-n1{margin:-.25rem!important}.mt-lg-n1,.my-lg-n1{margin-top:-.25rem!important}.mr-lg-n1,.mx-lg-n1{margin-right:-.25rem!important}.mb-lg-n1,.my-lg-n1{margin-bottom:-.25rem!important}.ml-lg-n1,.mx-lg-n1{margin-left:-.25rem!important}.m-lg-n2{margin:-.5rem!important}.mt-lg-n2,.my-lg-n2{margin-top:-.5rem!important}.mr-lg-n2,.mx-lg-n2{margin-right:-.5rem!important}.mb-lg-n2,.my-lg-n2{margin-bottom:-.5rem!important}.ml-lg-n2,.mx-lg-n2{margin-left:-.5rem!important}.m-lg-n3{margin:-1rem!important}.mt-lg-n3,.my-lg-n3{margin-top:-1rem!important}.mr-lg-n3,.mx-lg-n3{margin-right:-1rem!important}.mb-lg-n3,.my-lg-n3{margin-bottom:-1rem!important}.ml-lg-n3,.mx-lg-n3{margin-left:-1rem!important}.m-lg-n4{margin:-1.5rem!important}.mt-lg-n4,.my-lg-n4{margin-top:-1.5rem!important}.mr-lg-n4,.mx-lg-n4{margin-right:-1.5rem!important}.mb-lg-n4,.my-lg-n4{margin-bottom:-1.5rem!important}.ml-lg-n4,.mx-lg-n4{margin-left:-1.5rem!important}.m-lg-n5{margin:-3rem!important}.mt-lg-n5,.my-lg-n5{margin-top:-3rem!important}.mr-lg-n5,.mx-lg-n5{margin-right:-3rem!important}.mb-lg-n5,.my-lg-n5{margin-bottom:-3rem!important}.ml-lg-n5,.mx-lg-n5{margin-left:-3rem!important}.m-lg-auto{margin:auto!important}.mt-lg-auto,.my-lg-auto{margin-top:auto!important}.mr-lg-auto,.mx-lg-auto{margin-right:auto!important}.mb-lg-auto,.my-lg-auto{margin-bottom:auto!important}.ml-lg-auto,.mx-lg-auto{margin-left:auto!important}}@media (min-width:1200px){.m-xl-0{margin:0!important}.mt-xl-0,.my-xl-0{margin-top:0!important}.mr-xl-0,.mx-xl-0{margin-right:0!important}.mb-xl-0,.my-xl-0{margin-bottom:0!important}.ml-xl-0,.mx-xl-0{margin-left:0!important}.m-xl-1{margin:.25rem!important}.mt-xl-1,.my-xl-1{margin-top:.25rem!important}.mr-xl-1,.mx-xl-1{margin-right:.25rem!important}.mb-xl-1,.my-xl-1{margin-bottom:.25rem!important}.ml-xl-1,.mx-xl-1{margin-left:.25rem!important}.m-xl-2{margin:.5rem!important}.mt-xl-2,.my-xl-2{margin-top:.5rem!important}.mr-xl-2,.mx-xl-2{margin-right:.5rem!important}.mb-xl-2,.my-xl-2{margin-bottom:.5rem!important}.ml-xl-2,.mx-xl-2{margin-left:.5rem!important}.m-xl-3{margin:1rem!important}.mt-xl-3,.my-xl-3{margin-top:1rem!important}.mr-xl-3,.mx-xl-3{margin-right:1rem!important}.mb-xl-3,.my-xl-3{margin-bottom:1rem!important}.ml-xl-3,.mx-xl-3{margin-left:1rem!important}.m-xl-4{margin:1.5rem!important}.mt-xl-4,.my-xl-4{margin-top:1.5rem!important}.mr-xl-4,.mx-xl-4{margin-right:1.5rem!important}.mb-xl-4,.my-xl-4{margin-bottom:1.5rem!important}.ml-xl-4,.mx-xl-4{margin-left:1.5rem!important}.m-xl-5{margin:3rem!important}.mt-xl-5,.my-xl-5{margin-top:3rem!important}.mr-xl-5,.mx-xl-5{margin-right:3rem!important}.mb-xl-5,.my-xl-5{margin-bottom:3rem!important}.ml-xl-5,.mx-xl-5{margin-left:3rem!important}.p-xl-0{padding:0!important}.pt-xl-0,.py-xl-0{padding-top:0!important}.pr-xl-0,.px-xl-0{padding-right:0!important}.pb-xl-0,.py-xl-0{padding-bottom:0!important}.pl-xl-0,.px-xl-0{padding-left:0!important}.p-xl-1{padding:.25rem!important}.pt-xl-1,.py-xl-1{padding-top:.25rem!important}.pr-xl-1,.px-xl-1{padding-right:.25rem!important}.pb-xl-1,.py-xl-1{padding-bottom:.25rem!important}.pl-xl-1,.px-xl-1{padding-left:.25rem!important}.p-xl-2{padding:.5rem!important}.pt-xl-2,.py-xl-2{padding-top:.5rem!important}.pr-xl-2,.px-xl-2{padding-right:.5rem!important}.pb-xl-2,.py-xl-2{padding-bottom:.5rem!important}.pl-xl-2,.px-xl-2{padding-left:.5rem!important}.p-xl-3{padding:1rem!important}.pt-xl-3,.py-xl-3{padding-top:1rem!important}.pr-xl-3,.px-xl-3{padding-right:1rem!important}.pb-xl-3,.py-xl-3{padding-bottom:1rem!important}.pl-xl-3,.px-xl-3{padding-left:1rem!important}.p-xl-4{padding:1.5rem!important}.pt-xl-4,.py-xl-4{padding-top:1.5rem!important}.pr-xl-4,.px-xl-4{padding-right:1.5rem!important}.pb-xl-4,.py-xl-4{padding-bottom:1.5rem!important}.pl-xl-4,.px-xl-4{padding-left:1.5rem!important}.p-xl-5{padding:3rem!important}.pt-xl-5,.py-xl-5{padding-top:3rem!important}.pr-xl-5,.px-xl-5{padding-right:3rem!important}.pb-xl-5,.py-xl-5{padding-bottom:3rem!important}.pl-xl-5,.px-xl-5{padding-left:3rem!important}.m-xl-n1{margin:-.25rem!important}.mt-xl-n1,.my-xl-n1{margin-top:-.25rem!important}.mr-xl-n1,.mx-xl-n1{margin-right:-.25rem!important}.mb-xl-n1,.my-xl-n1{margin-bottom:-.25rem!important}.ml-xl-n1,.mx-xl-n1{margin-left:-.25rem!important}.m-xl-n2{margin:-.5rem!important}.mt-xl-n2,.my-xl-n2{margin-top:-.5rem!important}.mr-xl-n2,.mx-xl-n2{margin-right:-.5rem!important}.mb-xl-n2,.my-xl-n2{margin-bottom:-.5rem!important}.ml-xl-n2,.mx-xl-n2{margin-left:-.5rem!important}.m-xl-n3{margin:-1rem!important}.mt-xl-n3,.my-xl-n3{margin-top:-1rem!important}.mr-xl-n3,.mx-xl-n3{margin-right:-1rem!important}.mb-xl-n3,.my-xl-n3{margin-bottom:-1rem!important}.ml-xl-n3,.mx-xl-n3{margin-left:-1rem!important}.m-xl-n4{margin:-1.5rem!important}.mt-xl-n4,.my-xl-n4{margin-top:-1.5rem!important}.mr-xl-n4,.mx-xl-n4{margin-right:-1.5rem!important}.mb-xl-n4,.my-xl-n4{margin-bottom:-1.5rem!important}.ml-xl-n4,.mx-xl-n4{margin-left:-1.5rem!important}.m-xl-n5{margin:-3rem!important}.mt-xl-n5,.my-xl-n5{margin-top:-3rem!important}.mr-xl-n5,.mx-xl-n5{margin-right:-3rem!important}.mb-xl-n5,.my-xl-n5{margin-bottom:-3rem!important}.ml-xl-n5,.mx-xl-n5{margin-left:-3rem!important}.m-xl-auto{margin:auto!important}.mt-xl-auto,.my-xl-auto{margin-top:auto!important}.mr-xl-auto,.mx-xl-auto{margin-right:auto!important}.mb-xl-auto,.my-xl-auto{margin-bottom:auto!important}.ml-xl-auto,.mx-xl-auto{margin-left:auto!important}}.text-monospace{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace!important}.text-justify{text-align:justify!important}.text-wrap{white-space:normal!important}.text-nowrap{white-space:nowrap!important}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.text-left{text-align:left!important}.text-right{text-align:right!important}.text-center{text-align:center!important}@media (min-width:576px){.text-sm-left{text-align:left!important}.text-sm-right{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.text-md-left{text-align:left!important}.text-md-right{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.text-lg-left{text-align:left!important}.text-lg-right{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.text-xl-left{text-align:left!important}.text-xl-right{text-align:right!important}.text-xl-center{text-align:center!important}}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.font-weight-light{font-weight:300!important}.font-weight-lighter{font-weight:lighter!important}.font-weight-normal{font-weight:400!important}.font-weight-bold{font-weight:700!important}.font-weight-bolder{font-weight:bolder!important}.font-italic{font-style:italic!important}.text-white{color:#fff!important}.text-primary{color:#007bff!important}a.text-primary:focus,a.text-primary:hover{color:#0056b3!important}.text-secondary{color:#6c757d!important}a.text-secondary:focus,a.text-secondary:hover{color:#494f54!important}.text-success{color:#28a745!important}a.text-success:focus,a.text-success:hover{color:#19692c!important}.text-info{color:#17a2b8!important}a.text-info:focus,a.text-info:hover{color:#0f6674!important}.text-warning{color:#ffc107!important}a.text-warning:focus,a.text-warning:hover{color:#ba8b00!important}.text-danger{color:#dc3545!important}a.text-danger:focus,a.text-danger:hover{color:#a71d2a!important}.text-light{color:#f8f9fa!important}a.text-light:focus,a.text-light:hover{color:#cbd3da!important}.text-dark{color:#343a40!important}a.text-dark:focus,a.text-dark:hover{color:#121416!important}.text-body{color:#212529!important}.text-muted{color:#6c757d!important}.text-black-50{color:rgba(0,0,0,.5)!important}.text-white-50{color:rgba(255,255,255,.5)!important}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.text-decoration-none{text-decoration:none!important}.text-break{word-break:break-word!important;overflow-wrap:break-word!important}.text-reset{color:inherit!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}@media print{*,::after,::before{text-shadow:none!important;box-shadow:none!important}a:not(.btn){text-decoration:underline}abbr[title]::after{content:" (" attr(title) ")"}pre{white-space:pre-wrap!important}blockquote,pre{border:1px solid #adb5bd;page-break-inside:avoid}thead{display:table-header-group}img,tr{page-break-inside:avoid}h2,h3,p{orphans:3;widows:3}h2,h3{page-break-after:avoid}@page{size:a3}body{min-width:992px!important}.container{min-width:992px!important}.navbar{display:none}.badge{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered td,.table-bordered th{border:1px solid #dee2e6!important}.table-dark{color:inherit}.table-dark tbody+tbody,.table-dark td,.table-dark th,.table-dark thead th{border-color:#dee2e6}.table .thead-dark th{color:inherit;border-color:#dee2e6}} +/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/static/images/logo.png b/Poseiden-skeleton/target/classes/static/images/logo.png new file mode 100644 index 0000000000..e69de29bb2 diff --git a/Poseiden-skeleton/target/classes/templates/403.html b/Poseiden-skeleton/target/classes/templates/403.html new file mode 100644 index 0000000000..1dd3bf82ee --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/403.html @@ -0,0 +1,26 @@ + + + + + Access Denied + + + +
+
+
+

403 - Access Denied

+
+ Logged in user: +
+

+
+ +
+ Return to Home +
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/bidList/add.html b/Poseiden-skeleton/target/classes/templates/bidList/add.html new file mode 100644 index 0000000000..30d1ef6cfb --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/bidList/add.html @@ -0,0 +1,47 @@ + + + + + Add New Bid + + + +
+
+

Add New Bid

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/bidList/list.html b/Poseiden-skeleton/target/classes/templates/bidList/list.html new file mode 100644 index 0000000000..4dddc9f4cc --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/bidList/list.html @@ -0,0 +1,70 @@ + + + + + Bid List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+
+ Logged in user: +
+ +
+
+
+ +
+
+

Bid List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + +
IdAccountTypeBid QuantityAction
+ Edit | + Delete +
+
+
+ +
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/bidList/update.html b/Poseiden-skeleton/target/classes/templates/bidList/update.html new file mode 100644 index 0000000000..eca073f584 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/bidList/update.html @@ -0,0 +1,48 @@ + + + + + Update Bid + + + +
+
+

Update Bid

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/curvePoint/add.html b/Poseiden-skeleton/target/classes/templates/curvePoint/add.html new file mode 100644 index 0000000000..c5f040625c --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/curvePoint/add.html @@ -0,0 +1,47 @@ + + + + + Add New Curve Point + + + +
+
+

Add New Curve Point

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/curvePoint/list.html b/Poseiden-skeleton/target/classes/templates/curvePoint/list.html new file mode 100644 index 0000000000..66bbf930a0 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/curvePoint/list.html @@ -0,0 +1,71 @@ + + + + + Curve Point List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+ +
+ Logged in user: +
+ +
+
+
+ +
+
+

Curve Point List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + +
IdCurve IdTermValueAction
+ Edit | + Delete +
+
+
+ +
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/curvePoint/update.html b/Poseiden-skeleton/target/classes/templates/curvePoint/update.html new file mode 100644 index 0000000000..53ac437f7d --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/curvePoint/update.html @@ -0,0 +1,48 @@ + + + + + Update Curve Point + + + +
+
+

Update Curve Point

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/error.html b/Poseiden-skeleton/target/classes/templates/error.html new file mode 100644 index 0000000000..5ca332c1fb --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/error.html @@ -0,0 +1,20 @@ + + + + + Error + + + +
+
+
+

Something went wrong.

+

An unexpected error occurred.

+ Return to Home +
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/home.html b/Poseiden-skeleton/target/classes/templates/home.html new file mode 100644 index 0000000000..5de0dfe660 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/home.html @@ -0,0 +1,36 @@ + + + + + Home + + + +
+
+
+

Welcome to Poseidon Capital Solutions

+

Manage your bids, curve points, ratings, trades, and rules efficiently.

+
+
+ +
+ Logged in user: +
+ +
+
+ + +
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/login.html b/Poseiden-skeleton/target/classes/templates/login.html new file mode 100644 index 0000000000..dbccdef31c --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/login.html @@ -0,0 +1,39 @@ + + + + + Login + + + +
+
+
+

Login to Poseidon Capital Solutions

+

Please enter your credentials to access the platform.

+
+
+
+
+
+ Invalid username or password. +
+
+
+ + +
+
+ + +
+
+ +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/logout.html b/Poseiden-skeleton/target/classes/templates/logout.html new file mode 100644 index 0000000000..a9dc56b726 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/logout.html @@ -0,0 +1,24 @@ + + + + + Logout + + + +
+
+
+

You have been logged out

+

Thank you for using Poseidon Capital Solutions. We hope to see you again soon!

+
+
+
+ +
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/rating/add.html b/Poseiden-skeleton/target/classes/templates/rating/add.html new file mode 100644 index 0000000000..d05f0b66ed --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/rating/add.html @@ -0,0 +1,54 @@ + + + + + Add New Rating + + + +
+
+

Add New Rating

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/rating/list.html b/Poseiden-skeleton/target/classes/templates/rating/list.html new file mode 100644 index 0000000000..b8fbb177bb --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/rating/list.html @@ -0,0 +1,72 @@ + + + + + Rating List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+
+ Logged in user: +
+ +
+
+
+ +
+
+

Rating List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + + + +
IdMoodys RatingS&P RatingFitch RatingOrderAction
+ Edit | + Delete +
+
+
+ +
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/rating/update.html b/Poseiden-skeleton/target/classes/templates/rating/update.html new file mode 100644 index 0000000000..df39afe226 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/rating/update.html @@ -0,0 +1,55 @@ + + + + + Update Rating + + + +
+
+

Update Rating

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/ruleName/add.html b/Poseiden-skeleton/target/classes/templates/ruleName/add.html new file mode 100644 index 0000000000..d8fcb69933 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/ruleName/add.html @@ -0,0 +1,68 @@ + + + + + Add New Rule + + + +
+
+

Add New Rule

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/ruleName/list.html b/Poseiden-skeleton/target/classes/templates/ruleName/list.html new file mode 100644 index 0000000000..0aafee3af2 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/ruleName/list.html @@ -0,0 +1,77 @@ + + + + + Rule List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+ +
+ Logged in user: +
+ +
+
+
+ +
+
+

Rule List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + +
IdNameDescriptionJsonTemplateSQLSQL PartAction
+ Edit | + Delete +
+
+
+ +
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/ruleName/update.html b/Poseiden-skeleton/target/classes/templates/ruleName/update.html new file mode 100644 index 0000000000..eeeb96e081 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/ruleName/update.html @@ -0,0 +1,69 @@ + + + + + Update Rule + + + +
+
+

Update Rule

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/trade/add.html b/Poseiden-skeleton/target/classes/templates/trade/add.html new file mode 100644 index 0000000000..b5681f3adb --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/trade/add.html @@ -0,0 +1,47 @@ + + + + + Add New Trade + + + +
+
+

Add New Trade

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/trade/list.html b/Poseiden-skeleton/target/classes/templates/trade/list.html new file mode 100644 index 0000000000..c463394f44 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/trade/list.html @@ -0,0 +1,70 @@ + + + + + Trade List + + + +
+ +
+
+ Bid List | + Curve Points | + Ratings | + Trade | + Rule +
+
+ Logged in user: +
+ +
+
+
+ +
+
+

Trade List

+
+
+ +
+
+ Add New +
+
+ +
+
+ + + + + + + + + + + + + + + + + + + +
IdAccountTypeBuy QuantityAction
+ Edit | + Delete +
+
+
+ +
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/trade/update.html b/Poseiden-skeleton/target/classes/templates/trade/update.html new file mode 100644 index 0000000000..7de635788a --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/trade/update.html @@ -0,0 +1,48 @@ + + + + + Update Trade + + + +
+
+

Update Trade

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/user/add.html b/Poseiden-skeleton/target/classes/templates/user/add.html new file mode 100644 index 0000000000..a52c71c332 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/user/add.html @@ -0,0 +1,55 @@ + + + + + Add New User + + + +
+
+

Add New User

+
+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ + +

+
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/user/list.html b/Poseiden-skeleton/target/classes/templates/user/list.html new file mode 100644 index 0000000000..fc1076688c --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/user/list.html @@ -0,0 +1,42 @@ + + + + + User List + + + +
+
+

User List

+
+
+ Add New + + + + + + + + + + + + + + + + + + + +
IdFull NameUser NameRoleAction
+ Edit |  + Delete +
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/classes/templates/user/update.html b/Poseiden-skeleton/target/classes/templates/user/update.html new file mode 100644 index 0000000000..d81814e7c0 --- /dev/null +++ b/Poseiden-skeleton/target/classes/templates/user/update.html @@ -0,0 +1,55 @@ + + + + + Update User + + + +
+
+

Update User

+
+
+
+ +
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ +

+
+
+
+ +
+ + +
+
+
+
+ Cancel + +
+
+
+
+
+ + \ No newline at end of file diff --git a/Poseiden-skeleton/target/maven-archiver/pom.properties b/Poseiden-skeleton/target/maven-archiver/pom.properties new file mode 100644 index 0000000000..e7bbad132f --- /dev/null +++ b/Poseiden-skeleton/target/maven-archiver/pom.properties @@ -0,0 +1,3 @@ +artifactId=spring-boot-skeleton +groupId=net.guides.springboothelloworld +version=0.0.1-SNAPSHOT diff --git a/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/compile/default-compile/createdFiles.lst b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/compile/default-compile/createdFiles.lst new file mode 100644 index 0000000000..54940ba6ba --- /dev/null +++ b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/compile/default-compile/createdFiles.lst @@ -0,0 +1,30 @@ +com\nnk\springboot\repositories\RatingRepository.class +com\nnk\springboot\controllers\CurvePointController.class +com\nnk\springboot\Application.class +com\nnk\springboot\controllers\RatingController.class +com\nnk\springboot\services\TradeService.class +com\nnk\springboot\domain\BidList.class +com\nnk\springboot\controllers\LoginController.class +com\nnk\springboot\domain\Trade.class +com\nnk\springboot\repositories\UserRepository.class +com\nnk\springboot\controllers\UserController.class +com\nnk\springboot\controllers\TradeController.class +com\nnk\springboot\domain\RuleName.class +com\nnk\springboot\domain\CurvePoint.class +com\nnk\springboot\services\RatingService.class +com\nnk\springboot\controllers\HomeController.class +com\nnk\springboot\controllers\LogoutController.class +com\nnk\springboot\services\RuleNameService.class +com\nnk\springboot\domain\User.class +com\nnk\springboot\repositories\TradeRepository.class +com\nnk\springboot\repositories\BidListRepository.class +com\nnk\springboot\services\CustomUserDetailsService.class +com\nnk\springboot\services\UserService.class +com\nnk\springboot\services\BidListService.class +com\nnk\springboot\config\SecurityConfig.class +com\nnk\springboot\services\CurvePointService.class +com\nnk\springboot\domain\Rating.class +com\nnk\springboot\controllers\RuleNameController.class +com\nnk\springboot\controllers\BidListController.class +com\nnk\springboot\repositories\CurvePointRepository.class +com\nnk\springboot\repositories\RuleNameRepository.class diff --git a/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/compile/default-compile/inputFiles.lst b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/compile/default-compile/inputFiles.lst new file mode 100644 index 0000000000..7fdaf04ea2 --- /dev/null +++ b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/compile/default-compile/inputFiles.lst @@ -0,0 +1,30 @@ +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\repositories\BidListRepository.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\RuleNameController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\HomeController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\UserController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\TradeController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\domain\RuleName.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\repositories\RuleNameRepository.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\RatingService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\RuleNameService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\domain\CurvePoint.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\RatingController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\UserService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\CustomUserDetailsService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\CurvePointController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\repositories\CurvePointRepository.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\domain\Trade.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\CurvePointService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\BidListService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\domain\BidList.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\services\TradeService.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\repositories\RatingRepository.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\domain\Rating.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\config\SecurityConfig.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\repositories\UserRepository.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\LogoutController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\LoginController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\controllers\BidListController.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\domain\User.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\repositories\TradeRepository.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\main\java\com\nnk\springboot\Application.java diff --git a/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/testCompile/default-testCompile/createdFiles.lst b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/testCompile/default-testCompile/createdFiles.lst new file mode 100644 index 0000000000..fa359e71db --- /dev/null +++ b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/testCompile/default-testCompile/createdFiles.lst @@ -0,0 +1,12 @@ +com\nnk\springboot\services\TradeServiceTests.class +com\nnk\springboot\repositories\BidTests.class +com\nnk\springboot\repositories\CurvePointTests.class +com\nnk\springboot\repositories\RuleTests.class +com\nnk\springboot\services\RatingServiceTests.class +com\nnk\springboot\repositories\RatingTests.class +com\nnk\springboot\repositories\UserTests.class +com\nnk\springboot\services\BidListServiceTests.class +com\nnk\springboot\services\CurvePointServiceTests.class +com\nnk\springboot\repositories\TradeTests.class +com\nnk\springboot\services\RuleNameServiceTests.class +com\nnk\springboot\services\UserServiceTests.class diff --git a/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/testCompile/default-testCompile/inputFiles.lst b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/testCompile/default-testCompile/inputFiles.lst new file mode 100644 index 0000000000..68c419e091 --- /dev/null +++ b/Poseiden-skeleton/target/maven-status/maven-compiler-plugin/testCompile/default-testCompile/inputFiles.lst @@ -0,0 +1,12 @@ +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\repositories\RuleTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\repositories\CurvePointTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\repositories\UserTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\services\TradeServiceTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\services\UserServiceTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\services\RatingServiceTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\repositories\RatingTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\repositories\BidTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\services\CurvePointServiceTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\services\RuleNameServiceTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\services\BidListServiceTests.java +C:\Users\User\OneDrive\Desktop\Y.Ouicha\OpenClassrooms\Projet_N7\ouicha-hammou-yassine-poseidon-trading-app\Poseiden-skeleton\src\test\java\com\nnk\springboot\repositories\TradeTests.java diff --git a/Poseiden-skeleton/target/spring-boot-skeleton-0.0.1-SNAPSHOT.jar b/Poseiden-skeleton/target/spring-boot-skeleton-0.0.1-SNAPSHOT.jar new file mode 100644 index 0000000000..f7c35d7b84 Binary files /dev/null and b/Poseiden-skeleton/target/spring-boot-skeleton-0.0.1-SNAPSHOT.jar differ diff --git a/Poseiden-skeleton/target/spring-boot-skeleton-0.0.1-SNAPSHOT.jar.original b/Poseiden-skeleton/target/spring-boot-skeleton-0.0.1-SNAPSHOT.jar.original new file mode 100644 index 0000000000..2d6a4ff1b5 Binary files /dev/null and b/Poseiden-skeleton/target/spring-boot-skeleton-0.0.1-SNAPSHOT.jar.original differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/BidTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/BidTests.class new file mode 100644 index 0000000000..abf9ce828e Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/BidTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/CurvePointTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/CurvePointTests.class new file mode 100644 index 0000000000..e47859b45a Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/CurvePointTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/RatingTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/RatingTests.class new file mode 100644 index 0000000000..69ae1b76b5 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/RatingTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/RuleTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/RuleTests.class new file mode 100644 index 0000000000..67090a034d Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/RuleTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/TradeTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/TradeTests.class new file mode 100644 index 0000000000..32540baf73 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/TradeTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/UserTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/UserTests.class new file mode 100644 index 0000000000..75f0c6fa6e Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/repositories/UserTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/BidListServiceTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/BidListServiceTests.class new file mode 100644 index 0000000000..fc087bbe2a Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/BidListServiceTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/CurvePointServiceTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/CurvePointServiceTests.class new file mode 100644 index 0000000000..3f413b09a5 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/CurvePointServiceTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/RatingServiceTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/RatingServiceTests.class new file mode 100644 index 0000000000..9e33596689 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/RatingServiceTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/RuleNameServiceTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/RuleNameServiceTests.class new file mode 100644 index 0000000000..d7d3a62e55 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/RuleNameServiceTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/TradeServiceTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/TradeServiceTests.class new file mode 100644 index 0000000000..ac326884b3 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/TradeServiceTests.class differ diff --git a/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/UserServiceTests.class b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/UserServiceTests.class new file mode 100644 index 0000000000..f8ab502707 Binary files /dev/null and b/Poseiden-skeleton/target/test-classes/com/nnk/springboot/services/UserServiceTests.class differ