mirror of
https://github.com/klmp200/PQT_Gestionnaire_vente_stock.git
synced 2024-11-22 08:13:20 +00:00
[VRAC] Corrections de bug #1/?
Bug détectés restants : - MainFrame pas mise à jour avec le compte utilisateur connecté lorsqu'on switch dessus depuis la StartFrame - Le serveur renvoie REFUSED_QUERY lorsqu'on tente de se déconnecter
This commit is contained in:
parent
d697c1b123
commit
aad90bb989
@ -58,9 +58,12 @@ public class Main extends Application{
|
|||||||
}
|
}
|
||||||
|
|
||||||
private IStartupFrameModelListener getStartupFrameListener(Stage primaryStage, Scene sceneToDisplay){
|
private IStartupFrameModelListener getStartupFrameListener(Stage primaryStage, Scene sceneToDisplay){
|
||||||
return () -> trySwitchScene(primaryStage, sceneToDisplay, true);
|
return () -> {
|
||||||
|
Platform.runLater(()->trySwitchScene(primaryStage, sceneToDisplay, true));
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private IMainFrameModelListener getMainFrameListener(Stage primaryStage, Scene sceneToDisplay){
|
private IMainFrameModelListener getMainFrameListener(Stage primaryStage, Scene sceneToDisplay){
|
||||||
return () -> trySwitchScene(primaryStage, sceneToDisplay, false);
|
return () -> trySwitchScene(primaryStage, sceneToDisplay, false);
|
||||||
}
|
}
|
||||||
|
@ -35,7 +35,7 @@ public class StartupFrameController implements IStartupFrameModelListener {
|
|||||||
view.getAccountUsernameTextFieldContent(),
|
view.getAccountUsernameTextFieldContent(),
|
||||||
view.getAccountPasswordTextFieldContent()
|
view.getAccountPasswordTextFieldContent()
|
||||||
);
|
);
|
||||||
}catch(NullPointerException | IllegalArgumentException e){
|
}catch(Exception e){
|
||||||
view.displayError(GUIStringTool.getExceptionFormatter().render(e));
|
view.displayError(GUIStringTool.getExceptionFormatter().render(e));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ package com.pqt.client.gui.startup_frame;
|
|||||||
|
|
||||||
import com.pqt.client.gui.startup_frame.listeners.procedure.IStartupProcedureEventFirerer;
|
import com.pqt.client.gui.startup_frame.listeners.procedure.IStartupProcedureEventFirerer;
|
||||||
import com.pqt.client.gui.startup_frame.listeners.procedure.IStartupProcedureListener;
|
import com.pqt.client.gui.startup_frame.listeners.procedure.IStartupProcedureListener;
|
||||||
|
import com.pqt.client.gui.startup_frame.listeners.procedure.SimpleStartupProcedureEventFirerer;
|
||||||
import com.pqt.client.module.account.AccountService;
|
import com.pqt.client.module.account.AccountService;
|
||||||
import com.pqt.client.module.account.listeners.AccountListenerAdapter;
|
import com.pqt.client.module.account.listeners.AccountListenerAdapter;
|
||||||
import com.pqt.client.module.account.listeners.IAccountListener;
|
import com.pqt.client.module.account.listeners.IAccountListener;
|
||||||
@ -22,6 +23,7 @@ class StartupProcedureHandler {
|
|||||||
StartupProcedureHandler(NetworkService networkService, AccountService accountService) {
|
StartupProcedureHandler(NetworkService networkService, AccountService accountService) {
|
||||||
this.networkService = networkService;
|
this.networkService = networkService;
|
||||||
this.accountService = accountService;
|
this.accountService = accountService;
|
||||||
|
firerer = new SimpleStartupProcedureEventFirerer();
|
||||||
}
|
}
|
||||||
|
|
||||||
StartupProcedureHandler init(String host, Integer port, String username, String password){
|
StartupProcedureHandler init(String host, Integer port, String username, String password){
|
||||||
@ -39,13 +41,13 @@ class StartupProcedureHandler {
|
|||||||
|
|
||||||
private void testConnection(){
|
private void testConnection(){
|
||||||
networkService.addListener(getPingListener());
|
networkService.addListener(getPingListener());
|
||||||
|
networkService.setActiveServer(host, port);
|
||||||
networkService.sendPQTPing(host, port);
|
networkService.sendPQTPing(host, port);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void useRequestedServer(){
|
private void useRequestedServer(){
|
||||||
//Server found
|
//Server found
|
||||||
firerer.fireServerFoundEvent(host, port);
|
firerer.fireServerFoundEvent(host, port);
|
||||||
networkService.setActiveServer(host, port);
|
|
||||||
accountService.addListener(getUpdateAccountListListener());
|
accountService.addListener(getUpdateAccountListListener());
|
||||||
accountService.refreshAccounts();
|
accountService.refreshAccounts();
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package com.pqt.client.module.connection.senders;
|
package com.pqt.client.module.connection.senders;
|
||||||
|
|
||||||
import com.pqt.client.module.connection.listeners.IConnectionListener;
|
import com.pqt.client.module.connection.listeners.IConnectionListener;
|
||||||
|
import com.sun.javafx.binding.StringFormatter;
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
import java.io.BufferedReader;
|
||||||
import java.io.DataOutputStream;
|
import java.io.DataOutputStream;
|
||||||
@ -14,7 +15,10 @@ public class HttpTextSender implements ITextSender{
|
|||||||
@Override
|
@Override
|
||||||
public void send(String url, String text, IConnectionListener listener) {
|
public void send(String url, String text, IConnectionListener listener) {
|
||||||
try {
|
try {
|
||||||
HttpURLConnection con = (HttpURLConnection) new URL(url).openConnection();
|
String trueURL = String.format("http://%s?message=%s", url, text);
|
||||||
|
|
||||||
|
|
||||||
|
HttpURLConnection con = (HttpURLConnection) new URL(trueURL).openConnection();
|
||||||
con.setRequestMethod("GET");
|
con.setRequestMethod("GET");
|
||||||
con.setRequestProperty("Content-Type", "application/json");
|
con.setRequestProperty("Content-Type", "application/json");
|
||||||
con.setConnectTimeout(5000);
|
con.setConnectTimeout(5000);
|
||||||
|
@ -73,7 +73,9 @@ public class NetworkService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void setActiveServer(String host, Integer port){
|
public void setActiveServer(String host, Integer port){
|
||||||
connectionService.setServerUrl(String.format("%s:%s", host, port));
|
//TODO changer le nom de context de la webapp
|
||||||
|
String webAppContext = "pqt-server";
|
||||||
|
connectionService.setServerUrl(String.format("%s:%s/%s", host, port, webAppContext));
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendConfigRequest(String host, Integer port){
|
private void sendConfigRequest(String host, Integer port){
|
||||||
|
@ -4,6 +4,7 @@ import com.google.gson.Gson;
|
|||||||
import com.google.gson.GsonBuilder;
|
import com.google.gson.GsonBuilder;
|
||||||
import com.google.gson.reflect.TypeToken;
|
import com.google.gson.reflect.TypeToken;
|
||||||
|
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
import java.lang.reflect.Type;
|
import java.lang.reflect.Type;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@ -20,13 +21,55 @@ public class GSonMessageToolFactory implements IMessageToolFactory {
|
|||||||
return (obj)->gson.toJson(obj);
|
return (obj)->gson.toJson(obj);
|
||||||
}
|
}
|
||||||
public <T> IObjectParser<T> getObjectParser(Class<T> clazz){
|
public <T> IObjectParser<T> getObjectParser(Class<T> clazz){
|
||||||
return (str)->gson.fromJson(str, clazz);
|
return (str)->gson.fromJson(str, new Element<>(clazz));
|
||||||
}
|
}
|
||||||
public <T> IObjectFormatter<List<T>> getListFormatter(Class<T> clazz){
|
public <T> IObjectFormatter<List<T>> getListFormatter(Class<T> clazz){
|
||||||
return (obj)->gson.toJson(obj);
|
return (obj)->gson.toJson(obj);
|
||||||
}
|
}
|
||||||
public <T> IObjectParser<List<T>> getListParser(Class<T> clazz){
|
public <T> IObjectParser<List<T>> getListParser(Class<T> clazz){
|
||||||
Type listType = new TypeToken<ArrayList<T>>(){}.getType();
|
//Type listType = new TypeToken<ArrayList<T>>(){}.getType();
|
||||||
return (str)->gson.fromJson(str, listType);
|
return (str)->gson.fromJson(str, new ListWithElements<>(clazz));
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Element<T> implements ParameterizedType {
|
||||||
|
|
||||||
|
private Class<T> cl;
|
||||||
|
|
||||||
|
public Element(Class<T> cl) {
|
||||||
|
this.cl = cl;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type[] getActualTypeArguments() {
|
||||||
|
return new Type[] {cl};
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type getRawType() {
|
||||||
|
return cl;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type getOwnerType() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class ListWithElements<T> implements ParameterizedType {
|
||||||
|
|
||||||
|
private Class<T> elementsClass;
|
||||||
|
|
||||||
|
public ListWithElements(Class<T> elementsClass) {
|
||||||
|
this.elementsClass = elementsClass;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type[] getActualTypeArguments() {
|
||||||
|
return new Type[] {elementsClass};
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type getRawType() {
|
||||||
|
return List.class;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type getOwnerType() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
11
Workspace/server/src/main/WEB-INF/web.xml
Normal file
11
Workspace/server/src/main/WEB-INF/web.xml
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
<?xml version="1.0" encoding="ISO-8859-1"?>
|
||||||
|
<!DOCTYPE web-app
|
||||||
|
PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.2//EN"
|
||||||
|
"http://java.sun.com/j2ee/dtds/web-app_2_2.dtd">
|
||||||
|
|
||||||
|
<web-app>
|
||||||
|
<context-param>
|
||||||
|
<param-name>res-file.location</param-name>
|
||||||
|
<param-value>G:\temp\</param-value>
|
||||||
|
</context-param>
|
||||||
|
</web-app>
|
@ -5,6 +5,7 @@ import com.pqt.core.entities.user_account.AccountLevel;
|
|||||||
import com.pqt.server.tools.io.ISerialFileManager;
|
import com.pqt.server.tools.io.ISerialFileManager;
|
||||||
import com.pqt.server.tools.io.SimpleSerialFileManagerFactory;
|
import com.pqt.server.tools.io.SimpleSerialFileManagerFactory;
|
||||||
import com.pqt.server.tools.security.IHashTool;
|
import com.pqt.server.tools.security.IHashTool;
|
||||||
|
import com.pqt.server.tools.security.RandomString;
|
||||||
import com.pqt.server.tools.security.SHA256HashTool;
|
import com.pqt.server.tools.security.SHA256HashTool;
|
||||||
|
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
@ -22,19 +23,22 @@ import java.util.stream.Collectors;
|
|||||||
* Cette classe manipule les mot de passe sous forme chiffrée via un système de hash (SHA-256) + salt, et ne fait pas
|
* Cette classe manipule les mot de passe sous forme chiffrée via un système de hash (SHA-256) + salt, et ne fait pas
|
||||||
* persister les mots de passes non-chiffrées. Les noms d'utilisateurs sont stockés sans chiffrage.
|
* persister les mots de passes non-chiffrées. Les noms d'utilisateurs sont stockés sans chiffrage.
|
||||||
*/
|
*/
|
||||||
class FileAccountDao implements IAccountDao {
|
public class FileAccountDao implements IAccountDao {
|
||||||
|
|
||||||
private static final String ACCOUNT_FILE_NAME = "acc.pqt";
|
//TODO to modify
|
||||||
|
private static final String ACCOUNT_FILE_NAME = "G:\\temp\\acc.pqt";
|
||||||
|
|
||||||
private Set<AccountEntry> accountEntries;
|
private Set<AccountEntry> accountEntries;
|
||||||
private Set<AccountEntry> connectedAccount;
|
private Set<AccountEntry> connectedAccount;
|
||||||
private IHashTool hashTool;
|
private IHashTool hashTool;
|
||||||
|
private RandomString randomString;
|
||||||
private ISerialFileManager<AccountEntry> fileManager;
|
private ISerialFileManager<AccountEntry> fileManager;
|
||||||
|
|
||||||
FileAccountDao() {
|
public FileAccountDao() {
|
||||||
accountEntries = new HashSet<>();
|
accountEntries = new HashSet<>();
|
||||||
connectedAccount = new HashSet<>();
|
connectedAccount = new HashSet<>();
|
||||||
hashTool = new SHA256HashTool();
|
hashTool = new SHA256HashTool();
|
||||||
|
randomString = new RandomString(10);
|
||||||
fileManager = SimpleSerialFileManagerFactory.getFileManager(AccountEntry.class, ACCOUNT_FILE_NAME);
|
fileManager = SimpleSerialFileManagerFactory.getFileManager(AccountEntry.class, ACCOUNT_FILE_NAME);
|
||||||
loadFromFile();
|
loadFromFile();
|
||||||
}
|
}
|
||||||
@ -96,7 +100,7 @@ class FileAccountDao implements IAccountDao {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Passe un comtpe connecté dans l'état déconnecté. N'effectue le changement que si un compte connecté correspond
|
* Passe un compte connecté dans l'état déconnecté. N'effectue le changement que si un compte connecté correspond
|
||||||
* aux données fournies.
|
* aux données fournies.
|
||||||
* @param account données à utiliser pour efffectuer la correspondance avec un compte
|
* @param account données à utiliser pour efffectuer la correspondance avec un compte
|
||||||
* @return {@code true} si le changement d'état a eu lieu, {@code false sinon}
|
* @return {@code true} si le changement d'état a eu lieu, {@code false sinon}
|
||||||
@ -134,6 +138,18 @@ class FileAccountDao implements IAccountDao {
|
|||||||
fileManager.saveSetToFile(accountEntries);
|
fileManager.saveSetToFile(accountEntries);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean addAccount(Account account){
|
||||||
|
if(accountEntries.stream().filter(accountEntry -> accountEntry.getUsername().equals(account.getUsername())).count()==0) {
|
||||||
|
String salt = randomString.nextString();
|
||||||
|
String passwordHash = hashTool.hashAndSalt(account.getPassword(), salt);
|
||||||
|
accountEntries.add(new AccountEntry(account.getUsername(), passwordHash, salt, account.getPermissionLevel()));
|
||||||
|
saveToFile();
|
||||||
|
return true;
|
||||||
|
}else{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Charge les données des comptes depuis le fichier de sauvegarde.
|
* Charge les données des comptes depuis le fichier de sauvegarde.
|
||||||
* <p/>
|
* <p/>
|
||||||
|
@ -16,7 +16,7 @@ import java.util.List;
|
|||||||
*
|
*
|
||||||
* @author Guillaume "Cess" Prost
|
* @author Guillaume "Cess" Prost
|
||||||
*/
|
*/
|
||||||
interface IAccountDao {
|
public interface IAccountDao {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see AccountService#isAccountConnected(Account)
|
* @see AccountService#isAccountConnected(Account)
|
||||||
@ -59,4 +59,14 @@ interface IAccountDao {
|
|||||||
* de données.
|
* de données.
|
||||||
*/
|
*/
|
||||||
List<Account> getAccountList();
|
List<Account> getAccountList();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ajoute un objet {@link Account} dans la collection de comptes utilisateurs.
|
||||||
|
* <p/>
|
||||||
|
* Les implémentations doivent en outre effectuer une vérification pour s'assurer que le compte à rajouter ne pas
|
||||||
|
* un nom d'utilisateur qui existe déjà. Dans ce cas de figure, l'ajout du nouveau compte doit échouer.
|
||||||
|
* @param account
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
boolean addAccount(Account account);
|
||||||
}
|
}
|
||||||
|
@ -19,14 +19,15 @@ import java.util.*;
|
|||||||
*/
|
*/
|
||||||
public class FileStockDao implements IStockDao {
|
public class FileStockDao implements IStockDao {
|
||||||
|
|
||||||
private static final String STOCK_FILE_NAME = "stock.pqt";
|
//TODO to modify
|
||||||
|
private static final String STOCK_FILE_NAME = "G:\\temp\\stock.pqt";
|
||||||
private ISerialFileManager<Product> fileManager;
|
private ISerialFileManager<Product> fileManager;
|
||||||
private long nextProductId;
|
private long nextProductId;
|
||||||
private Random random;
|
private Random random;
|
||||||
|
|
||||||
private Map<Long, Product> products;
|
private Map<Long, Product> products;
|
||||||
|
|
||||||
FileStockDao() {
|
public FileStockDao() {
|
||||||
random = new Random();
|
random = new Random();
|
||||||
fileManager = SimpleSerialFileManagerFactory.getFileManager(Product.class, STOCK_FILE_NAME);
|
fileManager = SimpleSerialFileManagerFactory.getFileManager(Product.class, STOCK_FILE_NAME);
|
||||||
loadFromFile();
|
loadFromFile();
|
||||||
|
@ -28,10 +28,16 @@ public class QueryServlet extends HttpServlet {
|
|||||||
IMessageToolFactory messageToolFactory = new GSonMessageToolFactory();
|
IMessageToolFactory messageToolFactory = new GSonMessageToolFactory();
|
||||||
IMessageHandler msgHandler = new SimpleMessageHandler();
|
IMessageHandler msgHandler = new SimpleMessageHandler();
|
||||||
|
|
||||||
if (request.getParameter("message") != null) {
|
if (request.getQueryString() != null && !request.getQueryString().isEmpty() && request.getParameter("message")!=null) {
|
||||||
Message resp = msgHandler.handleMessage(messageToolFactory.getObjectParser(Message.class).parse(request.getParameter("message")));
|
try {
|
||||||
|
Message resp = msgHandler.handleMessage(messageToolFactory.getObjectParser(Message.class).parse(request.getParameter("message")));
|
||||||
|
|
||||||
response.getWriter().write(messageToolFactory.getObjectFormatter(Message.class).format(resp));
|
response.getWriter().write(messageToolFactory.getObjectFormatter(Message.class).format(resp));
|
||||||
|
}catch(Exception e){
|
||||||
|
response.getWriter().write(String.format("%s : %s", e.getClass().getName(), e.getMessage()));
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
response.getWriter().write("Query message was not correctly made : "+request.getQueryString());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -23,7 +23,7 @@ public class FileUtil {
|
|||||||
* @throws IOException if any IOException happend during this method's execution.
|
* @throws IOException if any IOException happend during this method's execution.
|
||||||
*/
|
*/
|
||||||
public static boolean createFileIfNotExist(Path filePath) throws IOException {
|
public static boolean createFileIfNotExist(Path filePath) throws IOException {
|
||||||
if(FileUtil.exist(filePath)){
|
if(!FileUtil.exist(filePath)){
|
||||||
Files.createFile(filePath);
|
Files.createFile(filePath);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -38,7 +38,7 @@ public class SimpleSerialFileManager<T> implements ISerialFileManager<T> {
|
|||||||
}catch(IOException | ClassNotFoundException e){
|
}catch(IOException | ClassNotFoundException e){
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
return null;
|
return new ArrayList<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -52,7 +52,7 @@ public class SimpleSerialFileManager<T> implements ISerialFileManager<T> {
|
|||||||
}catch(IOException | ClassNotFoundException e){
|
}catch(IOException | ClassNotFoundException e){
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
return null;
|
return new HashSet<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void fillCollection(Collection<T> collection) throws IOException, ClassNotFoundException {
|
private void fillCollection(Collection<T> collection) throws IOException, ClassNotFoundException {
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
package com.pqt.server.tools.io;
|
package com.pqt.server.tools.io;
|
||||||
|
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
|
||||||
public class SimpleSerialFileManagerFactory {
|
public class SimpleSerialFileManagerFactory {
|
||||||
protected SimpleSerialFileManagerFactory(){}
|
protected SimpleSerialFileManagerFactory(){}
|
||||||
|
|
||||||
public static <T> ISerialFileManager<T> getFileManager(Class<T> clazz, String filePath){
|
public static <T> ISerialFileManager<T> getFileManager(Class<T> clazz, String filePath){
|
||||||
return new SimpleSerialFileManager<>(filePath, clazz);
|
return SimpleSerialFileManagerFactory.getFileManager(clazz, Paths.get(filePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static <T> ISerialFileManager<T> getFileManager(Class<T> clazz, Path filePath){
|
public static <T> ISerialFileManager<T> getFileManager(Class<T> clazz, Path filePath){
|
||||||
|
@ -0,0 +1,62 @@
|
|||||||
|
package com.pqt.server.tools.security;
|
||||||
|
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.Random;
|
||||||
|
|
||||||
|
public class RandomString {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate a random string.
|
||||||
|
*/
|
||||||
|
public String nextString() {
|
||||||
|
for (int idx = 0; idx < buf.length; ++idx)
|
||||||
|
buf[idx] = symbols[random.nextInt(symbols.length)];
|
||||||
|
return new String(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final String upper = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
||||||
|
|
||||||
|
public static final String lower = upper.toLowerCase(Locale.ROOT);
|
||||||
|
|
||||||
|
public static final String digits = "0123456789";
|
||||||
|
|
||||||
|
public static final String alphanum = upper + lower + digits;
|
||||||
|
|
||||||
|
private final Random random;
|
||||||
|
|
||||||
|
private final char[] symbols;
|
||||||
|
|
||||||
|
private final char[] buf;
|
||||||
|
|
||||||
|
public RandomString(int length, Random random, String symbols) {
|
||||||
|
if (length < 1) throw new IllegalArgumentException();
|
||||||
|
if (symbols.length() < 2) throw new IllegalArgumentException();
|
||||||
|
this.random = Objects.requireNonNull(random);
|
||||||
|
this.symbols = symbols.toCharArray();
|
||||||
|
this.buf = new char[length];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an alphanumeric string generator.
|
||||||
|
*/
|
||||||
|
public RandomString(int length, Random random) {
|
||||||
|
this(length, random, alphanum);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an alphanumeric strings from a secure generator.
|
||||||
|
*/
|
||||||
|
public RandomString(int length) {
|
||||||
|
this(length, new SecureRandom());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create session identifiers.
|
||||||
|
*/
|
||||||
|
public RandomString() {
|
||||||
|
this(21);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user