- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
大家好!我正在创建一个 AES 加密聊天程序,但遇到了一个问题;每当消息超过 47 个字符时,就会显示“javax.crypto.IllegalBlockSizeException:使用填充密码解密时,输入长度必须是 16 的倍数”。我理解该错误消息,但我似乎不知道如何修复它。这是我的完整代码:聊天客户端.java 封装Chat.Application;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import javax.swing.JFrame;
import javax.swing.JOptionPane;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField;
import javax.swing.text.*;
import java.security.*;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.crypto.*;
import javax.crypto.spec.SecretKeySpec;
import sun.misc.*;
import java.net.InetAddress;
/*
* Known Bugs:
* Message size is limited to 47 characters
*/
public class ChatClient {
BufferedReader in;
PrintWriter out;
JFrame frame = new JFrame("ELECTRON Chatroom");
JTextField textField = new JTextField(32);
JTextArea messageArea = new JTextArea(8, 40);
byte[] keyValue = new byte[]{'5', '7', '3', '4', '5', '6', '3', '4', '9', '8', '5', '6', 'l', '9', '3', '4'};
final String ALGO = "AES";
String name;
String myName;
InetAddress ip = null;
String errorType;
/*
* Constructs the client by laying out the GUI and registering a listener
* with the textfield so that pressing Return in the listener sends the
* textfield contents to the server. Note however that the textfield is
* initially NOT editable, and only becomes editable AFTER the client
* receives the NAMEACCEPTED message from the server.
*/
public class JTextFieldLimit extends PlainDocument {
private int limit;
// optional uppercase conversion
private boolean toUppercase = false;
JTextFieldLimit(int limit) {
super();
this.limit = limit;
}
JTextFieldLimit(int limit, boolean upper) {
super();
this.limit = limit;
toUppercase = upper;
}
public void insertString
(int offset, String str, AttributeSet attr)
throws BadLocationException {
if (str == null) return;
if ((getLength() + str.length()) <= limit) {
if (toUppercase) str = str.toUpperCase();
super.insertString(offset, str, attr);
}
}
}
public ChatClient() {
// Layout GUI
textField.setEditable(false);
messageArea.setEditable(false);
messageArea.setWrapStyleWord(true);
messageArea.setLineWrap(true);
frame.getContentPane().add(textField, "North");
frame.getContentPane().add(new JScrollPane(messageArea), "Center");
frame.pack();
// Add Listeners
textField.addActionListener(new ActionListener() {
/**
* Responds to pressing the enter key in the textfield by sending
* the contents of the text field to the server. Then clear the text
* area in preparation for the next message.
*/
@Override
public void actionPerformed(ActionEvent e) {
try {
if ((textField.getText()).startsWith("/")) {
if ("Electron".equals(myName)) {
String command = (textField.getText());
out.println(command);
textField.setText("");
}
} else {
//ENCRYPTION
String encrypt = textField.getText();
Key key = generateKey();
Cipher c = Cipher.getInstance(ALGO);
c.init(Cipher.ENCRYPT_MODE, key);
byte[] encVal = c.doFinal(encrypt.getBytes());
String input = new BASE64Encoder().encode(encVal);
out.println(input);
textField.setText("");
}
} catch (Exception ex) {
Logger.getLogger(ChatClient.class.getName()).log(Level.SEVERE, null, ex);
}
}
});
}
/**
* Prompt for and return the address of the server.
*/
private String getServerAddress() {
return JOptionPane.showInputDialog(
frame,
"Enter IP Address of the Server:",
"ELECTRON Chatroom",
JOptionPane.QUESTION_MESSAGE);
}
/**
* Prompt for and return the desired screen name.
*/
private String getName() {
return JOptionPane.showInputDialog(
frame,
"Choose a screen name:",
"Screen name selection",
JOptionPane.PLAIN_MESSAGE);
}
/**
* Connects to the server then enters the processing loop.
*/
private void run() throws IOException {
// Make connection and initialize streams
String serverAddress = getServerAddress();
Socket socket = new Socket(serverAddress, 9001);
in = new BufferedReader(new InputStreamReader(
socket.getInputStream()));
out = new PrintWriter(socket.getOutputStream(), true);
// Process all messages from server, according to the protocol.
while (true) {
String line = in.readLine();
if (line.startsWith("SUBMITNAME")) {
if (line.length() == 18) {
line = line.substring(11);
System.out.println(line);
switch (line) {
case "ERROR 1":
errorType = "[ERROR: Name cannot be blank]";
JOptionPane.showMessageDialog(frame, errorType);
System.exit(0);
break;
case "ERROR 2":
errorType = "[ERROR: Your names 'Admin'? Seems legit...]";
JOptionPane.showMessageDialog(frame, errorType);
System.exit(0);
break;
case "ERROR 3":
errorType = "[ERROR: You have been banned]";
JOptionPane.showMessageDialog(frame, errorType);
System.exit(0);
break;
}
} else if (line.length() == 10) {
out.println(getName());
}
} else if (line.startsWith("NAMEACCEPTED")) {
myName = line.substring(13);
if (myName == "Admin") {
errorType = "[ERROR: Your names 'Admin'? Seems legit...]";
JOptionPane.showMessageDialog(frame, errorType);
System.exit(0);
} else if (myName == "ADMIN") {
errorType = "[ERROR: Your names 'Admin'? Seems legit...]";
JOptionPane.showMessageDialog(frame, errorType);
System.exit(0);
} else if (myName == "admin") {
errorType = "[ERROR: Your names 'Admin'? Seems legit...]";
JOptionPane.showMessageDialog(frame, errorType);
System.exit(0);
}
JOptionPane.showMessageDialog(frame, "Welcome " + myName + " (" + ip + ")");
ip = InetAddress.getLocalHost();
textField.setEditable(true);
//Limits message length to 47 characters (Disabled for debugging purposes)
//textField.setDocument(new JTextFieldLimit(47));
out.println(ip);
} else if (line.startsWith("SERVERMESSAGE")) {
line = line.substring(14);
messageArea.append(line + "\n");
messageArea.setCaretPosition(messageArea.getDocument().getLength());
} else if (line.startsWith("SERVERCOMMAND")) {
line = line.substring(14);
if (line.startsWith("kick " + ip)) {
System.exit(0);
}
} else if (line.startsWith("FINGERPRINT")) {
ip = InetAddress.getLocalHost();
out.println(ip);
} else if (line.startsWith("NAME")) {
name = line.substring(5);
} else if (line.startsWith("MESSAGE")) {
try {
//DECRYPTION
System.out.println(line);
line = line.substring(8);
String encryptedData = line;
System.out.println(line.length());
Key key = generateKey();
Cipher c = Cipher.getInstance(ALGO);
c.init(Cipher.DECRYPT_MODE, key);
byte[] decordedValue = new BASE64Decoder().decodeBuffer(encryptedData);
byte[] decValue = c.doFinal(decordedValue);
String decryptedValue = new String(decValue);
messageArea.append(name + ": " + decryptedValue + "\n");
messageArea.setCaretPosition(messageArea.getDocument().getLength());
} catch (Exception ex) {
Logger.getLogger(ChatClient.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
}
private Key generateKey() throws Exception {
Key key = new SecretKeySpec(keyValue, ALGO);
return key;
}
/**
* Runs the client as an application with a closeable frame.
*/
public static void main(String[] args) throws Exception {
ChatClient client = new ChatClient();
client.frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
client.frame.setVisible(true);
client.run();
}
}
ChatServer.java
package Chat.Application;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.HashSet;
/**
* A multi-threaded chat room server. When a client connects the server requests
* a screen name by sending the client the text "SUBMITNAME", and keeps
* requesting a name until a unique one is received. After a client submits a
* unique name, the server acknowledges with "NAMEACCEPTED". Then all messages
* from that client will be broadcast to all other clients that have submitted a
* unique screen name. The broadcast messages are prefixed with "MESSAGE ".
*
* Because this is just a teaching example to illustrate a simple chat server,
* there are a few features that have been left out. Two are very useful and
* belong in production code:
*
* 1. The protocol should be enhanced so that the client can send clean
* disconnect messages to the server.
*
* 2. The server should do some logging.
*/
public class ChatServer {
/**
* The port that the server listens on.
*/
private static final int PORT = 9001;
/**
* The set of all names of clients in the chat room. Maintained so that we
* can check that new clients are not registering name already in use.
*/
private static HashSet<String> names = new HashSet<>();
/**
* The set of all the print writers for all the clients. This set is kept so
* we can easily broadcast messages.
*/
private static HashSet<PrintWriter> writers = new HashSet<>();
/**
* The set of all the print writers for all the clients. This set is kept so
* we can easily broadcast messages.
*/
private static HashSet<String> bans = new HashSet<>();
private static String[] Names = new String[50];
private static String[] Fingerprints = new String[50];
static int array = 0;
static int index;
/**
* The application main method, which just listens on a port and spawns
* handler threads.
*/
public static void main(String[] args) throws Exception {
System.out.println("Chat Server Activated");
ServerSocket listener = new ServerSocket(PORT);
try {
while (true) {
new Handler(listener.accept()).start();
}
} finally {
listener.close();
}
}
/**
* A handler thread class. Handlers are spawned from the listening loop and
* are responsible for a dealing with a single client and broadcasting its
* messages.
*/
private static class Handler extends Thread {
private String name;
private String ban;
private Socket socket;
private BufferedReader in;
private PrintWriter out;
private Integer length;
private String fingerprint;
/**
* Constructs a handler thread, squirreling away the socket. All the
* interesting work is done in the run method.
*/
public Handler(Socket socket) {
this.socket = socket;
}
/**
* Services this thread's client by repeatedly requesting a screen name
* until a unique one has been submitted, then acknowledges the name and
* registers the output stream for the client in a global set, then
* repeatedly gets inputs and broadcasts them.
*/
@Override
public void run() {
try {
// Create character streams for the socket.
in = new BufferedReader(new InputStreamReader(
socket.getInputStream()));
out = new PrintWriter(socket.getOutputStream(), true);
// Request a name from this client. Keep requesting until
// a name is submitted that is not already used. Note that
// checking for the existence of a name and adding the name
// must be done while locking the set of names.
while (true) {
out.println("SUBMITNAME");
name = in.readLine();
length = name.length();
if (length == 0) {
out.println("SUBMITNAME ERROR 1");
return;
}
if (name == "null") {
out.println("SUBMITNAME ERROR 1");
return;
}
if (name == "Admin") {
out.println("SUBMITNAME ERROR 2");
return;
}
if (name == "admin") {
out.println("SUBMITNAME ERROR 2");
return;
}
if (name == "ADMIN") {
out.println("SUBMITNAME ERROR 2");
return;
}
synchronized (bans) {
out.println("FINGERPRINT");
fingerprint = in.readLine();
if (bans.contains(fingerprint)) {
out.println("SUBMITNAME ERROR 3");
return;
}
}
synchronized (names) {
if (!names.contains(name)) {
names.add(name);
break;
}
}
}
// Now that a successful name has been chosen, add the
// socket's print writer to the set of all writers so
// this client can receive broadcast messages.
out.println("NAMEACCEPTED " + name);
Names[array] = name;
System.out.println(Names[array]);
Fingerprints[array] = in.readLine();
System.out.println(Fingerprints[array]);
array = array + 1;
//Announces that user is Online
for (PrintWriter writer : writers) {
writer.println("SERVERMESSAGE " + name + " is now Online");
}
System.out.println(Names);
writers.add(out);
// Accept messages from this client and broadcast them.
// Ignore other clients that cannot be broadcasted to.
while (true) {
String input = in.readLine();
System.out.println(input);
if (input == null) {
return;
}
for (PrintWriter writer : writers) {
if (input.startsWith("/")) {
if ("Electron".equals(name)) {
//Tracks the hostname and IP address of the user
if (input.startsWith("/track")) {
input = input.substring(7);
index = java.util.Arrays.asList(Names).indexOf(input);
out.println("SERVERMESSAGE " + input + ": " + Fingerprints[index]);
//Bans the user from returning until Server is deactivated
} else if (input.startsWith("/ban")) {
input = input.substring(5);
index = java.util.Arrays.asList(Names).indexOf(input);
synchronized (bans) {
bans.add(Fingerprints[index]);
}
out.println("SERVERMESSAGE " + input + "(" + Fingerprints[index] + ")" + " is now banned");
} else if (input.startsWith("/kick")) {
input = input.substring(6);
index = java.util.Arrays.asList(Names).indexOf(input);
out.println("SERVERCOMMAND kick " + Fingerprints[index]);
} else if (input.startsWith("/deactivate")) {
writer.println("SERVERMESSAGE The server is now Offline");
System.exit(0);
}
}
} else {
writer.println("NAME " + name);
writer.println("MESSAGE " + input);
}
}
}
} catch (IOException e) {
System.out.println(e);
} finally {
// This client is going down! Remove its name and its print
// writer from the sets, and close its socket.
for (PrintWriter writer : writers) {
writer.println("SERVERMESSAGE " + name + " is now Offline");
}
if (name != null) {
names.remove(name);
}
if (out != null) {
writers.remove(out);
}
try {
socket.close();
} catch (IOException e) {
}
}
}
}
}
我尝试过改变发送的文本量,添加更多填充,甚至尝试 1 - 1000 之间的每个输入,以查看是否存在模式(没有......)。我在互联网上进行了各种研究,但没有结果。
感谢您帮助菜鸟程序员完成他的第二个程序(真的。)!祝你度过愉快的一周!
最佳答案
这是真的。 AES 是一种分组密码,这意味着它一次只能加密 16 个字节。
这意味着您必须将消息填充为 16 的倍数,并多次调用它,每个 block 调用一次。为了获得良好的加密效果,您还需要添加 IV,并将其应用于所有 block 。这可能很难自己实现,因此您应该让您的库来实现。 javax.crypto 库已经可以做到这一点。
解决方案是选择另一种算法,如 the Javadocs 中所示。 (或查阅完整列表here)。
试试这个算法:AES/CBC/PKCS5Padding
关于java - 无法发送超过 47 个字符的 AES 加密消息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18306474/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!