Skip to content

Commit 8eeaccb

Browse files
Enable username for new user registrations (#509)
* Add domain and data resource logic * Add user name to all layers except UI * Enable user name upon registration * Ensure empty content is highlighted * Fix test --------- Co-authored-by: Steffengreiner <[email protected]>
1 parent ff90309 commit 8eeaccb

File tree

18 files changed

+141
-46
lines changed

18 files changed

+141
-46
lines changed

identity-api/src/main/java/life/qbic/identity/api/UserInfo.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
*
88
* @since 1.0.0
99
*/
10-
public record UserInfo(String id, String fullName, String emailAddress, String encryptedPassword,
10+
public record UserInfo(String id, String fullName, String emailAddress, String userName, String encryptedPassword,
1111
boolean isActive) {
1212

1313
}

identity-api/src/main/java/life/qbic/identity/api/UserInformationService.java

+9
Original file line numberDiff line numberDiff line change
@@ -33,4 +33,13 @@ public interface UserInformationService {
3333
*/
3434
Optional<UserInfo> findById(String userId);
3535

36+
/**
37+
* Queries if a desired username is still available and not in use by another user already.
38+
*
39+
* @param userName the desired username
40+
* @return true, if the username is still available, false if not
41+
* @since 1.0.0
42+
*/
43+
boolean userNameAvailable(String userName);
44+
3645
}

identity-infrastructure/src/main/java/life/qbic/identity/infrastructure/QbicUserRepo.java

+2
Original file line numberDiff line numberDiff line change
@@ -44,4 +44,6 @@ public interface QbicUserRepo extends CrudRepository<User, UserId> {
4444
* @return a list of matching users which are set to active in the data manager application
4545
*/
4646
List<User> findUsersByActiveTrue();
47+
48+
User findUserByUserName(String userName);
4749
}

identity-infrastructure/src/main/java/life/qbic/identity/infrastructure/UserJpaRepository.java

+7-4
Original file line numberDiff line numberDiff line change
@@ -16,12 +16,10 @@
1616
* <p>Implementation for the {@link UserDataStorage} interface.
1717
*
1818
* <p>This class serves as an adapter and proxies requests to an JPA implementation to interact
19-
* with
20-
* persistent {@link User} data in the storage layer.
19+
* with persistent {@link User} data in the storage layer.
2120
*
2221
* <p>The actual JPA implementation is done by {@link QbicUserRepo}, which is injected as
23-
* dependency
24-
* upon creation.
22+
* dependency upon creation.
2523
*
2624
* @since 1.0.0
2725
*/
@@ -54,4 +52,9 @@ public Optional<User> findUserById(UserId id) {
5452
public List<User> findAllActiveUsers() {
5553
return userRepo.findUsersByActiveTrue();
5654
}
55+
56+
@Override
57+
public Optional<User> findUserByUserName(String userName) {
58+
return Optional.ofNullable(userRepo.findUserByUserName(userName));
59+
}
5760
}

identity/src/main/java/life/qbic/identity/application/service/BasicUserInformationService.java

+7-2
Original file line numberDiff line numberDiff line change
@@ -17,8 +17,7 @@
1717
* <b>Basic user information service</b>
1818
*
1919
* <p>Implementation of the {@link UserInformationService}, provides a OHS via a Java interface to
20-
* query
21-
* user information</p>
20+
* query user information</p>
2221
*
2322
* @since 1.0.0
2423
*/
@@ -54,8 +53,14 @@ public Optional<UserInfo> findById(String userId) {
5453
}
5554
}
5655

56+
@Override
57+
public boolean userNameAvailable(String userName) {
58+
return userRepository.findByUserName(userName).isEmpty();
59+
}
60+
5761
private UserInfo convert(User user) {
5862
return new UserInfo(user.id().get(), user.fullName().get(), user.emailAddress().get(),
63+
user.userName(),
5964
user.getEncryptedPassword().get(),
6065
user.isActive());
6166
}

identity/src/main/java/life/qbic/identity/application/user/IdentityService.java

+17-2
Original file line numberDiff line numberDiff line change
@@ -45,13 +45,14 @@ public IdentityService(UserRepository userRepository) {
4545
* client's responsibility to handle the raw password.
4646
*
4747
* @param fullName the full name of the user
48+
* @param userName
4849
* @param email the mail address of the user
4950
* @param rawPassword the raw password provided by the user
5051
* @return a registration response with information about if the registration was successful or
5152
* not.
5253
* @since 1.0.0
5354
*/
54-
public ApplicationResponse registerUser(final String fullName, final String email,
55+
public ApplicationResponse registerUser(final String fullName, String userName, final String email,
5556
final char[] rawPassword) {
5657

5758
var registrationResponse = validateInput(fullName, email, rawPassword);
@@ -72,8 +73,12 @@ public ApplicationResponse registerUser(final String fullName, final String emai
7273
return ApplicationResponse.failureResponse(new UserExistsException());
7374
}
7475

76+
if (userRepository.findByUserName(userName).isPresent()) {
77+
return ApplicationResponse.failureResponse(new UserNameNotAvailableException());
78+
}
79+
7580
// Trigger the user creation in the domain service
76-
userDomainService.get().createUser(userFullName, userEmail, userPassword);
81+
userDomainService.get().createUser(userFullName, userName, userEmail, userPassword);
7782

7883
// Overwrite the password
7984
Arrays.fill(rawPassword, '-');
@@ -190,6 +195,16 @@ public UserExistsException() {
190195
}
191196
}
192197

198+
public static class UserNameNotAvailableException extends ApplicationException {
199+
200+
201+
@Serial
202+
private static final long serialVersionUID = 4409722243047442583L;
203+
204+
public UserNameNotAvailableException() {
205+
}
206+
}
207+
193208
/**
194209
* Activates a user with the userId provided. If no user is matched then this method does
195210
* nothing.

identity/src/main/java/life/qbic/identity/application/user/registration/RegisterUserInput.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -15,9 +15,10 @@ public interface RegisterUserInput {
1515
* @param fullName the full name of the user
1616
* @param email the user's mail address
1717
* @param rawPassword the user selected raw password for authentication
18+
* @param userName
1819
* @since 1.0.0
1920
*/
20-
void register(String fullName, String email, char[] rawPassword);
21+
void register(String fullName, String email, char[] rawPassword, String userName);
2122

2223
/**
2324
* Set the output the use case shall call, when finished.

identity/src/main/java/life/qbic/identity/application/user/registration/Registration.java

+6-3
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
import life.qbic.application.commons.ApplicationResponse;
44
import life.qbic.identity.application.user.IdentityService;
55
import life.qbic.identity.application.user.IdentityService.UserExistsException;
6+
import life.qbic.identity.application.user.IdentityService.UserNameNotAvailableException;
67
import life.qbic.identity.domain.model.EmailAddress.EmailValidationException;
78
import life.qbic.identity.domain.model.EncryptedPassword.PasswordValidationException;
89
import life.qbic.identity.domain.model.FullName.FullNameValidationException;
@@ -35,7 +36,7 @@ public class Registration implements RegisterUserInput {
3536
*
3637
* <p>The default output implementation just prints to std out on success and std err on failure,
3738
* after the use case has been executed via
38-
* {@link Registration#register(String, String, char[])}.
39+
* {@link RegisterUserInput#register(String, String, char[], String)}.
3940
*
4041
* @param identityService the user registration service to save the new user to.
4142
* @since 1.0.0
@@ -59,13 +60,13 @@ public void setRegisterUserOutput(RegisterUserOutput registerUserOutput) {
5960
* @inheritDocs
6061
*/
6162
@Override
62-
public void register(String fullName, String email, char[] rawPassword) {
63+
public void register(String fullName, String email, char[] rawPassword, String userName) {
6364
if (registerUserOutput == null) {
6465
log.error("No use case output set.");
6566
return;
6667
}
6768
try {
68-
identityService.registerUser(fullName, email, rawPassword)
69+
identityService.registerUser(fullName, userName, email, rawPassword)
6970
.ifSuccessOrElse(this::reportSuccess,
7071
response -> registerUserOutput.onUnexpectedFailure(build(response)));
7172
} catch (Exception e) {
@@ -90,6 +91,8 @@ private UserRegistrationException build(ApplicationResponse applicationResponse)
9091
builder.withFullNameException((FullNameValidationException) e);
9192
} else if (e instanceof UserExistsException) {
9293
builder.withUserExistsException((UserExistsException) e);
94+
} else if (e instanceof UserNameNotAvailableException) {
95+
builder.withUserNameNotAvailableException((UserNameNotAvailableException) e);
9396
} else {
9497
builder.withUnexpectedException(e);
9598
}

identity/src/main/java/life/qbic/identity/application/user/registration/UserRegistrationException.java

+15
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,9 @@
33
import java.io.Serial;
44
import java.util.Optional;
55
import life.qbic.application.commons.ApplicationException;
6+
import life.qbic.identity.application.user.IdentityService;
67
import life.qbic.identity.application.user.IdentityService.UserExistsException;
8+
import life.qbic.identity.application.user.IdentityService.UserNameNotAvailableException;
79
import life.qbic.identity.domain.model.EmailAddress.EmailValidationException;
810
import life.qbic.identity.domain.model.EncryptedPassword.PasswordValidationException;
911
import life.qbic.identity.domain.model.FullName.FullNameValidationException;
@@ -30,6 +32,8 @@ public class UserRegistrationException extends ApplicationException {
3032
private final transient PasswordValidationException invalidPasswordException;
3133
private final transient FullNameValidationException fullNameException;
3234

35+
private final transient UserNameNotAvailableException userNameNotAvailableException;
36+
3337
private final transient UserExistsException userExistsException;
3438
private final transient RuntimeException unexpectedException;
3539

@@ -47,6 +51,7 @@ public static class Builder {
4751

4852
private UserExistsException userExistsException;
4953
private RuntimeException unexpectedException;
54+
private UserNameNotAvailableException userNameNotAvailableException;
5055

5156
protected Builder() {
5257

@@ -80,6 +85,11 @@ public Builder withUnexpectedException(RuntimeException e) {
8085
public UserRegistrationException build() {
8186
return new UserRegistrationException(this);
8287
}
88+
89+
public Builder withUserNameNotAvailableException(UserNameNotAvailableException e) {
90+
this.userNameNotAvailableException = e;
91+
return this;
92+
}
8393
}
8494

8595
private UserRegistrationException(Builder builder) {
@@ -88,6 +98,7 @@ private UserRegistrationException(Builder builder) {
8898
invalidPasswordException = builder.invalidPasswordException;
8999
userExistsException = builder.userExistsException;
90100
unexpectedException = builder.unexpectedException;
101+
userNameNotAvailableException = builder.userNameNotAvailableException;
91102
}
92103

93104
public Optional<EmailValidationException> emailFormatException() {
@@ -106,6 +117,10 @@ public Optional<UserExistsException> userExistsException() {
106117
return Optional.ofNullable(userExistsException);
107118
}
108119

120+
public Optional<UserNameNotAvailableException> userNameNotAvailableException() {
121+
return Optional.ofNullable(userNameNotAvailableException);
122+
}
123+
109124
public Optional<RuntimeException> unexpectedException() {
110125
return Optional.ofNullable(unexpectedException);
111126
}

identity/src/main/java/life/qbic/identity/domain/model/User.java

+23-14
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,9 @@ public class User implements Serializable {
3535
@Convert(converter = FullNameConverter.class)
3636
private FullName fullName;
3737

38+
@Column(name = "userName")
39+
private String userName;
40+
3841
@Column(name = "email")
3942
@Convert(converter = EmailConverter.class)
4043
private EmailAddress emailAddress;
@@ -47,6 +50,15 @@ public class User implements Serializable {
4750
protected User() {
4851
}
4952

53+
private User(UserId id, FullName fullName, EmailAddress emailAddress,
54+
String userName, EncryptedPassword encryptedPassword) {
55+
this.id = id;
56+
this.fullName = fullName;
57+
this.emailAddress = emailAddress;
58+
this.encryptedPassword = encryptedPassword;
59+
this.userName = userName;
60+
}
61+
5062
/**
5163
* Creates a new user account, with a unique identifier to unambiguously match the user within
5264
* QBiC's organisation.
@@ -59,27 +71,20 @@ protected User() {
5971
*
6072
* @param fullName the full name of the user
6173
* @param emailAddress the email address value of the user
74+
* @param userName the desired username
6275
* @param encryptedPassword the encrypted password of the new user
6376
* @return the new user
6477
* @since 1.0.0
6578
*/
6679
public static User create(FullName fullName, EmailAddress emailAddress,
67-
EncryptedPassword encryptedPassword) {
80+
String userName, EncryptedPassword encryptedPassword) {
6881
UserId id = UserId.create();
69-
var user = new User(id, fullName, emailAddress, encryptedPassword);
82+
var user = new User(id, fullName, emailAddress, userName, encryptedPassword);
7083
user.active = false;
7184

7285
return user;
7386
}
7487

75-
private User(UserId id, FullName fullName, EmailAddress emailAddress,
76-
EncryptedPassword encryptedPassword) {
77-
this.id = id;
78-
this.fullName = fullName;
79-
this.emailAddress = emailAddress;
80-
this.encryptedPassword = encryptedPassword;
81-
}
82-
8388
@Override
8489
public String toString() {
8590
return "User{" +
@@ -91,10 +96,6 @@ public String toString() {
9196
'}';
9297
}
9398

94-
private void setEncryptedPassword(EncryptedPassword encryptedPassword) {
95-
this.encryptedPassword = encryptedPassword;
96-
}
97-
9899
/**
99100
* Get access to the encrypted password
100101
*
@@ -105,6 +106,10 @@ public EncryptedPassword getEncryptedPassword() {
105106
return this.encryptedPassword;
106107
}
107108

109+
private void setEncryptedPassword(EncryptedPassword encryptedPassword) {
110+
this.encryptedPassword = encryptedPassword;
111+
}
112+
108113
public UserId id() {
109114
return this.id;
110115
}
@@ -113,6 +118,10 @@ public EmailAddress emailAddress() {
113118
return this.emailAddress;
114119
}
115120

121+
public String userName() {
122+
return userName;
123+
}
124+
116125
public FullName fullName() {
117126
return this.fullName;
118127
}

identity/src/main/java/life/qbic/identity/domain/repository/UserDataStorage.java

+1
Original file line numberDiff line numberDiff line change
@@ -55,5 +55,6 @@ public interface UserDataStorage {
5555
*/
5656
List<User> findAllActiveUsers();
5757

58+
Optional<User> findUserByUserName(String userName);
5859

5960
}

identity/src/main/java/life/qbic/identity/domain/repository/UserRepository.java

+8-4
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,10 @@ public class UserRepository implements Serializable {
2222

2323
private final UserDataStorage dataStorage;
2424

25+
protected UserRepository(UserDataStorage dataStorage) {
26+
this.dataStorage = dataStorage;
27+
}
28+
2529
/**
2630
* Retrieves a Singleton instance of a user {@link UserRepository}. In case this method is called
2731
* the first time, a new instance is created.
@@ -38,10 +42,6 @@ public static UserRepository getInstance(UserDataStorage dataStorage) {
3842
return instance;
3943
}
4044

41-
protected UserRepository(UserDataStorage dataStorage) {
42-
this.dataStorage = dataStorage;
43-
}
44-
4545
/**
4646
* Searches for a user with the provided mail address
4747
*
@@ -70,6 +70,10 @@ public Optional<User> findByEmail(EmailAddress emailAddress) throws RuntimeExcep
7070
}
7171
}
7272

73+
public Optional<User> findByUserName(String userName) {
74+
return dataStorage.findUserByUserName(userName);
75+
}
76+
7377
/**
7478
* Searches for a user matching a provided userId
7579
*

identity/src/main/java/life/qbic/identity/domain/service/UserDomainService.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -37,16 +37,17 @@ public UserDomainService(UserRepository userRepository) {
3737
* will be created.
3838
*
3939
* @param fullName the full name of the user
40+
* @param userName
4041
* @param emailAddress a valid mail address
4142
* @param password the password desired by the user
4243
* @since 1.0.0
4344
*/
44-
public void createUser(FullName fullName, EmailAddress emailAddress, EncryptedPassword password) {
45+
public void createUser(FullName fullName, String userName, EmailAddress emailAddress, EncryptedPassword password) {
4546
// Ensure idempotent behaviour of the service
4647
if (userRepository.findByEmail(emailAddress).isPresent()) {
4748
return;
4849
}
49-
var user = User.create(fullName, emailAddress, password);
50+
var user = User.create(fullName, emailAddress, userName, password);
5051
userRepository.addUser(user);
5152
var userCreatedEvent = UserRegistered.create(user.id().get(), user.fullName().get(),
5253
user.emailAddress().get());

0 commit comments

Comments
 (0)