Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Captcha configurable #310

Open
wants to merge 5 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
54 changes: 34 additions & 20 deletions src/main/java/com/nonononoki/alovoa/component/AuthFilter.java
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.security.authentication.AuthenticationManager;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.AuthenticationException;
Expand All @@ -12,24 +13,37 @@

public class AuthFilter extends UsernamePasswordAuthenticationFilter {

private static final String USERNAME = "username";
private static final String PASSWORD = "password";
private static final String CAPTCHA_ID = "captchaId";
private static final String CAPTCHA_TEXT = "captchaText";
public static final String REDIRECT_URL = "redirect-url";

@Override
public Authentication attemptAuthentication(HttpServletRequest request, HttpServletResponse response)
throws AuthenticationException {

String username = request.getParameter(USERNAME);
String password = request.getParameter(PASSWORD);
long captchaId = Long.parseLong(request.getParameter(CAPTCHA_ID));
String captchaText = request.getParameter(CAPTCHA_TEXT);
request.getSession().setAttribute(REDIRECT_URL, request.getParameter(REDIRECT_URL));

AuthToken auth = new AuthToken(username, password, captchaId, captchaText);
AuthenticationManager am = this.getAuthenticationManager();
return am.authenticate(auth);
}
private static final String USERNAME = "username";
private static final String PASSWORD = "password";
private static final String CAPTCHA_ID = "captchaId";
private static final String CAPTCHA_TEXT = "captchaText";
public static final String REDIRECT_URL = "redirect-url";

@Value("${app.captcha.login.enabled}")
private String captchaLoginEnabled;

@Override
public Authentication attemptAuthentication(HttpServletRequest request, HttpServletResponse response)
throws AuthenticationException {
long captchaId;
String captchaText;

String username = request.getParameter(USERNAME);
String password = request.getParameter(PASSWORD);

if (Boolean.parseBoolean(captchaLoginEnabled)) {
captchaId = Long.parseLong(request.getParameter(CAPTCHA_ID));
captchaText = request.getParameter(CAPTCHA_TEXT);
} else {
captchaId = -1;
captchaText = null;
}

request.getSession().setAttribute(REDIRECT_URL, request.getParameter(REDIRECT_URL));

AuthToken auth = new AuthToken(username, password, captchaId, captchaText);
AuthenticationManager am = this.getAuthenticationManager();
return am.authenticate(auth);

}
}
25 changes: 17 additions & 8 deletions src/main/java/com/nonononoki/alovoa/component/AuthProvider.java
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.security.authentication.AuthenticationProvider;
import org.springframework.security.authentication.BadCredentialsException;
import org.springframework.security.authentication.DisabledException;
Expand Down Expand Up @@ -37,7 +38,10 @@ public class AuthProvider implements AuthenticationProvider {

@Autowired
private PasswordEncoder passwordEncoder;


@Value("${app.captcha.login.enabled}")
private String captchaEnabled;

@SuppressWarnings("unused")
private static final Logger logger = LoggerFactory.getLogger(AuthProvider.class);

Expand All @@ -50,15 +54,20 @@ public Authentication authenticate(Authentication authentication) throws Authent
long captchaId = a.getCaptchaId();
String captchaText = a.getCaptchaText();

Captcha c = captchaRepo.findById(captchaId).orElse(null);
if (c == null) {
throw new BadCredentialsException("");
}
if (Boolean.parseBoolean(captchaEnabled)) {
logger.debug("Captcha enabled, so check captcha");
Captcha c = captchaRepo.findById(captchaId).orElse(null);
if (c == null) {
throw new BadCredentialsException("");
}

captchaRepo.delete(c);
captchaRepo.delete(c);

if (!c.getText().equalsIgnoreCase(captchaText)) {
throw new BadCredentialsException("");
if (!c.getText().equalsIgnoreCase(captchaText)) {
throw new BadCredentialsException("");
}
} else {
logger.debug("Captcha disabled, so we do not care about it");
}

User user = userRepo.findByEmail(email);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,9 @@ public class DeleteAccountResource {
@Value("${app.user.delete.duration.valid}")
private long accountDeleteDuration;

@Value("${app.captcha.delete.enabled}")
private String captchaDeleteEnabled;

@GetMapping("/delete-account/{tokenString}")
public ModelAndView deleteAccount(@PathVariable String tokenString) throws AlovoaException, InvalidKeyException,
IllegalBlockSizeException, BadPaddingException, NoSuchAlgorithmException, NoSuchPaddingException,
Expand All @@ -57,6 +60,7 @@ public ModelAndView deleteAccount(@PathVariable String tokenString) throws Alovo
active = true;
}
mav.addObject("active", active);
mav.addObject("captchaEnabled", Boolean.valueOf(captchaDeleteEnabled));

return mav;
}
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/com/nonononoki/alovoa/html/ImprintResource.java
Original file line number Diff line number Diff line change
Expand Up @@ -11,11 +11,16 @@ public class ImprintResource {
@Value("${app.company.name}")
private String companyName;

@Value("${app.captcha.imprint.enabled}")
private String captchaImprintEnabled;

@GetMapping("/imprint")
public ModelAndView imprint() {

ModelAndView mav = new ModelAndView("imprint");
mav.addObject("companyName", companyName);
mav.addObject("captchaEnabled", Boolean.valueOf(captchaImprintEnabled));

return mav;
}
}
6 changes: 5 additions & 1 deletion src/main/java/com/nonononoki/alovoa/html/LoginResource.java
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,10 @@ public class LoginResource {

@Value("${app.privacy.update-date}")
private String privacyDate;


@Value("${app.captcha.login.enabled}")
private String captchaEnabled;

public static final String URL = "/login";

@GetMapping(URL)
Expand All @@ -30,6 +33,7 @@ public ModelAndView login() throws AlovoaException {
}

ModelAndView mav = new ModelAndView("login");
mav.addObject("captchaEnabled", Boolean.valueOf(captchaEnabled));
return mav;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
import com.nonononoki.alovoa.service.AuthService;
import com.nonononoki.alovoa.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
Expand Down Expand Up @@ -34,13 +35,17 @@ public class PasswordResource {
@Autowired
private TextEncryptorConverter textEncryptor;

@Value("${app.captcha.password.enabled}")
private String captchaPasswordEnabled;

@GetMapping("/reset")
public ModelAndView passwordReset() throws AlovoaException, InvalidKeyException, IllegalBlockSizeException,
BadPaddingException, NoSuchAlgorithmException, NoSuchPaddingException, InvalidAlgorithmParameterException,
UnsupportedEncodingException {
ModelAndView mav = new ModelAndView("password-reset");
User user = authService.getCurrentUser();
mav.addObject("user", UserDto.userToUserDto(user, user, userService, textEncryptor, UserDto.NO_MEDIA));
mav.addObject("captchaEnabled", Boolean.valueOf(captchaPasswordEnabled));
return mav;
}

Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package com.nonononoki.alovoa.html;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
Expand All @@ -27,7 +28,10 @@ public class RegisterResource {

@Autowired
private AuthService authService;


@Value("${app.captcha.register.enabled}")
private String captchaRegisterEnabled;

public static final String URL = "/register";

@GetMapping(URL)
Expand All @@ -40,6 +44,7 @@ public ModelAndView register() throws AlovoaException {
ModelAndView mav = new ModelAndView("register");
mav.addObject("genders", genderRepo.findAll());
mav.addObject("intentions", userIntentionRepo.findAll());
mav.addObject("captchaRegisterEnabled", Boolean.parseBoolean(captchaRegisterEnabled));
return mav;
}

Expand Down
3 changes: 3 additions & 0 deletions src/main/java/com/nonononoki/alovoa/model/RegisterDto.java
Original file line number Diff line number Diff line change
Expand Up @@ -17,4 +17,7 @@ public class RegisterDto {

private boolean termsConditions;
private boolean privacy;

private String captchaText;
private long captchaId;
}
12 changes: 9 additions & 3 deletions src/main/java/com/nonononoki/alovoa/service/ImprintService.java
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
import java.util.Date;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;

import com.nonononoki.alovoa.entity.Contact;
Expand All @@ -24,11 +25,16 @@ public class ImprintService {
@Autowired
private ContactRepository contactRepo;

@Value("${app.captcha.imprint.enabled}")
private String captchaImprintEnabled;

public Contact contact(ContactDto dto)
throws UnsupportedEncodingException, NoSuchAlgorithmException, AlovoaException {
boolean isValid = captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText());
if (!isValid) {
throw new AlovoaException(publicService.text("backend.error.captcha.invalid"));
if (Boolean.parseBoolean(captchaImprintEnabled)) {
boolean isValid = captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText());
if (!isValid) {
throw new AlovoaException(publicService.text("backend.error.captcha.invalid"));
}
}

Contact c = new Contact();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,14 +52,19 @@ public class PasswordService {
@Value("${app.user.password-reset.duration.valid}")
private int userPasswordResetDuration;

@Value("${app.captcha.password.enabled}")
private String captchaPasswordEnabled;

public UserPasswordToken resetPassword(PasswordResetDto dto)
throws AlovoaException, NoSuchAlgorithmException, MessagingException, IOException {

User u = authService.getCurrentUser();

if (u == null) {
if (!captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText())) {
throw new AlovoaException("captcha_invalid");
if (Boolean.parseBoolean(captchaPasswordEnabled)) {
if (!captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText())) {
throw new AlovoaException("captcha_invalid");
}
}
u = userRepo.findByEmail(Tools.cleanEmail(dto.getEmail()));

Expand Down
10 changes: 10 additions & 0 deletions src/main/java/com/nonononoki/alovoa/service/RegisterService.java
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,9 @@ public class RegisterService {
@Autowired
private TextEncryptorConverter textEncryptor;

@Value("${app.captcha.register.enabled}")
private String captchaRegisterEnabled;

private static final int MIN_PASSWORD_SIZE = 7;

private static final Logger logger = LoggerFactory.getLogger(RegisterService.class);
Expand All @@ -121,6 +124,13 @@ public String register(RegisterDto dto)
throw new AlovoaException("email_invalid");
}

if (Boolean.parseBoolean(captchaRegisterEnabled)) {
boolean isValid = captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText());
if (!isValid) {
throw new AlovoaException(publicService.text("backend.error.captcha.invalid"));
}
}

if (!profile.equals(Tools.DEV)) {
dto.setEmail(Tools.cleanEmail(dto.getEmail()));
if (plusAddressing && dto.getEmail().contains("+")) {
Expand Down
13 changes: 9 additions & 4 deletions src/main/java/com/nonononoki/alovoa/service/UserService.java
Original file line number Diff line number Diff line change
Expand Up @@ -120,6 +120,9 @@ public class UserService {
@Value("${app.intention.delay}")
private long intentionDelay;

@Value("${app.captcha.delete.enabled}")
private String captchaDeleteEnabled;

public static void removeUserDataCascading(User user, UserDeleteParams userDeleteParam) {

UserRepository userRepo = userDeleteParam.getUserRepo();
Expand Down Expand Up @@ -329,8 +332,10 @@ public void deleteAccountConfirm(UserDeleteAccountDto dto)
throw new AlovoaException("deletion_wrong_email");
}

if (!captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText())) {
throw new AlovoaException("captcha_invalid");
if (Boolean.parseBoolean(captchaDeleteEnabled)) {
if (!captchaService.isValid(dto.getCaptchaId(), dto.getCaptchaText())) {
throw new AlovoaException("captcha_invalid");
}
}

UserDeleteParams userDeleteParam = UserDeleteParams.builder().conversationRepo(conversationRepo)
Expand Down Expand Up @@ -800,7 +805,7 @@ public boolean hasNewAlert(String lang) throws AlovoaException {
User user = authService.getCurrentUser(true);
// user always check their alerts periodically in the background, so just update
// it here
if (user != null) {
if (user != null && user.getDates()!=null) {
updateUserInfo(user, lang);
return user.getDates().getNotificationDate().after(user.getDates().getNotificationCheckedDate());
} else {
Expand All @@ -810,7 +815,7 @@ public boolean hasNewAlert(String lang) throws AlovoaException {

public boolean hasNewMessage() throws AlovoaException {
User user = authService.getCurrentUser(true);
if (user != null && user.getDates().getMessageDate() != null
if (user != null && user.getDates()!=null && user.getDates().getMessageDate() != null
&& user.getDates().getMessageCheckedDate() != null) {
return user.getDates().getMessageDate().after(user.getDates().getMessageCheckedDate());
} else {
Expand Down
9 changes: 9 additions & 0 deletions src/main/resources/application.properties
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,15 @@
spring.profiles.active=test

server.port=${PORT:8080}
server.forward-headers-strategy=native
server.tomcat.remote-ip-header=x-forwarded-for
server.tomcat.protocol-header=x-forwarded-proto

app.captcha.login.enabled=false
app.captcha.delete.enabled=false
app.captcha.imprint.enabled=true
app.captcha.password.enabled=false
app.captcha.register.enabled=true

#enable to force https for development
#server.ssl.key-store-type=PKCS12
Expand Down
6 changes: 4 additions & 2 deletions src/main/resources/templates/delete-account.html
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,9 @@ <h3 class="title" th:text="#{delete-account.title}"></h3>
<input class="input" name="email" required th:placeholder="#{email}" type="email">
</div>
<div style="padding-top: 32px;"></div>
<div th:replace="~{fragments.html::captcha}"></div>
<div th:if="${captchaEnabled}">
<div th:replace="~{fragments.html::captcha}"></div>
</div>
<div class="field">
<input class="switch" id="confirm" name="confirm" required style="position: absolute"
type="checkbox"> <label for="confirm"><span th:text="#{delete-account.confirm}"></span>
Expand Down Expand Up @@ -76,4 +78,4 @@ <h3 class="title" th:text="#{delete-account.title}"></h3>
<script src="/js/user-profile-lib.js"></script>
</body>

</html>
</html>
6 changes: 4 additions & 2 deletions src/main/resources/templates/imprint.html
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,9 @@ <h2 th:text="#{email} + ':'"></h2>
th:placeholder="#{message}" required></textarea>
</div>
<div style="padding-top: 12px;"></div>
<div th:replace="~{fragments.html::captcha}"></div>
<div th:if="${captchaEnabled}">
<div th:replace="~{fragments.html::captcha}"></div>
</div>
<button class="button colored is-primary" th:text="#{submit}"></button>
</form>
</div>
Expand Down Expand Up @@ -77,4 +79,4 @@ <h2 th:text="#{email} + ':'"></h2>
<script src="/js/alovoa.js"></script>
<script src="/js/imprint.js"></script>
</body>
</html>
</html>
Loading