HTTPS的Socket实现代码

 

 

服务端

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.security.KeyStore;

import javax.net.ssl.KeyManagerFactory;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLServerSocket;
import javax.net.ssl.TrustManagerFactory;

/*********************************************************************************************************************** 
 * <ul> 
 * <li>1)生成服务端私钥</li> 
 * <li>keytool -genkey -alias serverkey -keystore kserver.keystore</li> 
 * <li>2)根据私钥,到处服务端证书</li> 
 * <li>keytool -exoport -alias serverkey -keystore kserver.keystore -file server.crt</li> 
 * <li>3)把证书加入到客户端受信任的keystore中</li> 
 * <li>keytool -import -alias serverkey -file server.crt -keystore tclient.keystore</li> 
 * </ul> 
 **********************************************************************************************************************/

/**
 * SSL Server 
 *
 */
public class SSLServer {

    private static final int    DEFAULT_PORT                    = 7777;

    private static final String SERVER_KEY_STORE_PASSWORD       = "123456";
    private static final String SERVER_TRUST_KEY_STORE_PASSWORD = "123456";

    private SSLServerSocket     serverSocket;

    /**
     * 启动程序 
     *
     * @param args
     */
    public static void main(String[] args) {
        SSLServer server = new SSLServer();
        server.init();
        server.start();
    }

    /**
     * <ul> 
     * <li>听SSL Server Socket</li> 
     * <li> 由于该程序不是演示Socket监听,所以简单采用单线程形式,并且仅仅接受客户端的消息,并且返回客户端指定消息</li> 
     * </ul> 
     */
    public void start() {
        if (serverSocket == null) {
            System.out.println("ERROR");
            return;
        }
        while (true) {
            try {
                Socket s = serverSocket.accept();
                InputStream input = s.getInputStream();
                OutputStream output = s.getOutputStream();

                BufferedInputStream bis = new BufferedInputStream(input);
                BufferedOutputStream bos = new BufferedOutputStream(output);

                byte[] buffer = new byte[20];
                bis.read(buffer);
                System.out.println(new String(buffer));

                bos.write("Server Echo".getBytes());
                bos.flush();

                s.close();
            } catch (Exception e) {
                System.out.println(e);
            }
        }
    }

    /**
     * <ul> 
     * <li>ssl连接的重点:</li> 
     * <li>初始化SSLServerSocket</li> 
     * <li>导入服务端私钥KeyStore,导入服务端受信任的KeyStore(客户端的证书)</li> 
     * </ul> 
     */
    public void init() {
        try {
            SSLContext ctx = SSLContext.getInstance("SSL");

            KeyManagerFactory kmf = KeyManagerFactory.getInstance("SunX509");
            TrustManagerFactory tmf = TrustManagerFactory.getInstance("SunX509");

            KeyStore ks = KeyStore.getInstance("JKS");
            KeyStore tks = KeyStore.getInstance("JKS");

            ks.load(new FileInputStream("E:/kserver.keystore"), SERVER_KEY_STORE_PASSWORD.toCharArray());
            tks.load(new FileInputStream("E:/tserver.keystore"), SERVER_TRUST_KEY_STORE_PASSWORD.toCharArray());

            kmf.init(ks, SERVER_KEY_STORE_PASSWORD.toCharArray());
            tmf.init(tks);

            ctx.init(kmf.getKeyManagers(), tmf.getTrustManagers(), null);

            serverSocket = (SSLServerSocket) ctx.getServerSocketFactory().createServerSocket(DEFAULT_PORT);
            serverSocket.setNeedClientAuth(true);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}  


客户端

 

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.security.KeyStore;

import javax.net.ssl.KeyManagerFactory;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSocket;
import javax.net.ssl.TrustManagerFactory;

/**
 * SSL Client
 *
 */
public class SSLClient {

    private static final String DEFAULT_HOST                    = "127.0.0.1";
    private static final int    DEFAULT_PORT                    = 7777;

    private static final String CLIENT_KEY_STORE_PASSWORD       = "123456";
    private static final String CLIENT_TRUST_KEY_STORE_PASSWORD = "123456";

    private SSLSocket           sslSocket;

    /**
     * 启动客户端程序
     *
     * @param args
     */
    public static void main(String[] args) {
        SSLClient client = new SSLClient();
        client.init();
        client.process();
    }

    /**
     * 通过ssl socket与服务端进行连接,并且发送一个消息
     */
    public void process() {
        if (sslSocket == null) {
            System.out.println("ERROR");
            return;
        }
        try {
            InputStream input = sslSocket.getInputStream();
            OutputStream output = sslSocket.getOutputStream();

            BufferedInputStream bis = new BufferedInputStream(input);
            BufferedOutputStream bos = new BufferedOutputStream(output);

            bos.write("Client Message".getBytes());
            bos.flush();

            byte[] buffer = new byte[20];
            bis.read(buffer);
            System.out.println(new String(buffer));

            sslSocket.close();
        } catch (IOException e) {
            System.out.println(e);
        }
    }

    /**
     * <ul>
     * <li>ssl连接的重点:</li>
     * <li>初始化SSLSocket</li>
     * <li>导入客户端私钥KeyStore,导入客户端受信任的KeyStore(服务端的证书)</li>
     * </ul>
     */
    public void init() {
        try {
            SSLContext ctx = SSLContext.getInstance("SSL");

            KeyManagerFactory kmf = KeyManagerFactory.getInstance("SunX509");
            TrustManagerFactory tmf = TrustManagerFactory.getInstance("SunX509");

            KeyStore ks = KeyStore.getInstance("JKS");
            KeyStore tks = KeyStore.getInstance("JKS");

            ks.load(new FileInputStream("E:/kclient.keystore"), CLIENT_KEY_STORE_PASSWORD.toCharArray());
            tks.load(new FileInputStream("E:/tclient.keystore"), CLIENT_TRUST_KEY_STORE_PASSWORD.toCharArray());

            kmf.init(ks, CLIENT_KEY_STORE_PASSWORD.toCharArray());
            tmf.init(tks);

            ctx.init(kmf.getKeyManagers(), tmf.getTrustManagers(), null);

            sslSocket = (SSLSocket) ctx.getSocketFactory().createSocket(DEFAULT_HOST, DEFAULT_PORT);
        } catch (Exception e) {
            System.out.println(e);
        }
    }

}


 

 

 http://blog.csdn.net/xxb2008


JAVA调用HTTPS 链接

import javax.net.ssl.*;
import java.io.*;
import java.net.URL;
import java.net.URLConnection;
import java.security.KeyStore;
import java.security.SecureRandom;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;

/**
 * Created with .
 * Date: 14-4-10
 * Time: 下午3:09
 * To change this template use File | Settings | File Templates.
 */
public class SSLHttpClient {
   
    public static void main(String[] args) {
        SSLHttpClient client = new SSLHttpClient();
        client.connect();
    }

    private void connect() {
   
        sendPost("https://localhost/index.jsp", "name1=value1&name2=value2");
  
    }

   
    /**
     * 向指定URL发送POST方法的请求
     * @param url    发送请求的URL
     * @param params 请求参数,请求参数应该是name1=value1&name2=value2的形式。
     * @return URL所代表远程资源的响应
     */
    public static String sendPost(String url, String params) {
        PrintWriter out = null;
        BufferedReader in = null;
        String result = "";
        try {

            URL realUrl = new URL(url);

            SSLContext context = SSLContext.getInstance("SSL");
            context.init(null, new TrustManager[]{new TrustAnyTrustManager()}, new SecureRandom());


            HttpsURLConnection conn = (HttpsURLConnection) realUrl.openConnection();

            conn = (HttpsURLConnection) realUrl.openConnection();  // 打开和URL之间的连接
            conn.setSSLSocketFactory(context.getSocketFactory());
            conn.setHostnameVerifier(new TrustAnyHostnameVerifier());


            // 设置通用的请求属性
            conn.setRequestProperty("accept", "*/*");
            conn.setRequestProperty("connection", "Keep-Alive");
            conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");


            // 发送POST请求必须设置如下两行
            conn.setDoOutput(true);
            conn.setDoInput(true);
            // 获取URLConnection对象对应的输出流
            out = new PrintWriter(conn.getOutputStream());
            // 发送请求参数
            out.print(params);
            // flush输出流的缓冲
            out.flush();

            // 定义BufferedReader输入流来读取URL的响应
            in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
            String line;
            while ((line = in.readLine()) != null) {
                result += "\n" + line;
            }

            System.out.println(result);
        } catch (Exception e) {
            System.out.println("发送POST请求出现异常!" + e);
            e.printStackTrace();
        }
        // 使用finally块来关闭输出流、输入流
        finally {
            try {
                if (out != null) {
                    out.close();
                }
                if (in != null) {
                    in.close();
                }
            } catch (IOException ex) {
                ex.printStackTrace();
            }
        }
        return result;
    }

}

class TrustAnyTrustManager implements X509TrustManager {
    public void checkClientTrusted(X509Certificate[] chain, String authType)
            throws CertificateException {
    }

    public void checkServerTrusted(X509Certificate[] chain, String authType)
            throws CertificateException {
    }

    public X509Certificate[] getAcceptedIssuers() {
        return new X509Certificate[]{};
    }
}


class TrustAnyHostnameVerifier implements HostnameVerifier {
    public boolean verify(String hostname, SSLSession session) {
        // 直接Pass,全部信任
        return true;
    }
}


 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: Django-Channels 是一个基于 Django 框架的 WebSocket 库,用于实现实时应用程序。要发送 WebSocket 消息,您可以按照以下步骤进行操作: 1. 在 Django-Channels 的 consumer 中定义一个 WebSocket 处理函数,例如: ```python from channels.generic.websocket import AsyncWebsocketConsumer import json class MyConsumer(AsyncWebsocketConsumer): async def connect(self): await self.accept() async def disconnect(self, close_code): pass async def receive(self, text_data): text_data_json = json.loads(text_data) message = text_data_json['message'] await self.send(text_data=json.dumps({ 'message': message })) ``` 2. 在前端代码中创建 WebSocket 连接并发送消息,例如: ```javascript const socket = new WebSocket('ws://localhost:8000/ws/my_consumer/'); socket.onopen = function(event) { console.log('WebSocket is connected.'); }; socket.onmessage = function(event) { console.log('Message received:', event.data); }; socket.onclose = function(event) { console.log('WebSocket is closed.'); }; const message = { 'message': 'Hello, world!' }; socket.send(JSON.stringify(message)); ``` 在这个例子中,我们创建了一个 WebSocket 连接,并将消息发送到 `MyConsumer` consumer 中。当 `MyConsumer` consumer 接收到消息时,它将发送一个响应消息回到前端,前端代码将在 `onmessage` 回调函数中接收到响应消息并在控制台中打印出来。 注意,在生产环境中,您应该将 WebSocket 连接配置为使用 HTTPS 而不是 HTTP,并使用适当的身份验证和授权方法来保护您的应用程序。 ### 回答2: Django-Channels是一个用于构建实时Web应用程序的库,它支持使用WebSocket进行实时通信。在Django-Channels中,我们可以使用`channels.layers`模块来发送socket消息。 首先,我们需要在Django的设置文件中配置Channels。我们需要添加Channels的应用,并设置`ASGI_APPLICATION`为我们的应用名称。如下所示: ```python INSTALLED_APPS = [ ... 'channels', ] ASGI_APPLICATION = 'myproject.routing.application' ``` 接下来,我们需要创建一个routing.py文件来配置Channels路由。在这个文件中,我们可以定义消息路由的处理函数。示例如下: ```python from channels.routing import ProtocolTypeRouter, URLRouter from myapp import consumers application = ProtocolTypeRouter({ 'websocket': URLRouter([ url(r'^ws/myapp/(?P<room_name>\w+)/$', consumers.ChatConsumer), ]), }) ``` 然后,我们可以创建一个消费者类来处理接收到的消息,并使用Channels的`channels.layers`模块发送socket消息。在这个类中,我们可以重写`connect`、`disconnect`、`receive`等方法来处理连接、断开连接和接收消息的逻辑。以下是一个简单的示例: ```python from channels.generic.websocket import AsyncWebsocketConsumer from channels.layers import get_channel_layer from asgiref.sync import async_to_sync class MyConsumer(AsyncWebsocketConsumer): async def connect(self): self.room_name = self.scope['url_route']['kwargs']['room_name'] self.room_group_name = 'chat_%s' % self.room_name await self.channel_layer.group_add( self.room_group_name, self.channel_name ) await self.accept() async def disconnect(self, close_code): await self.channel_layer.group_discard( self.room_group_name, self.channel_name ) async def receive(self, text_data): await self.channel_layer.group_send( self.room_group_name, { 'type': 'chat_message', 'message': text_data } ) async def chat_message(self, event): message = event['message'] await self.send(text_data=message) ``` 最后,我们可以在需要发送socket消息的地方,使用`channel_layer`模块获取channel层对象,并使用其`group_send`方法来发送消息。以下是一个简单的示例: ```python from channels.layers import get_channel_layer from asgiref.sync import async_to_sync channel_layer = get_channel_layer() async_to_sync(channel_layer.group_send)( 'chat_room', { 'type': 'chat_message', 'message': 'Hello, World!' } ) ``` 以上是使用Django-Channels发送socket消息的基本实现代码。当有客户端连接到我们定义的WebSocket路由时,将会触发对应的消费者类方法,并通过`channel_layer`发送消息给指定的群组或客户端。 ### 回答3: Django-Channels是一个基于Django的扩展,用于为Web应用程序添加实时功能。它可以通过WebSocket协议发送和接收消息。以下是一个使用Django-Channels发送socket代码示例: 首先,安装Django-Channels和其依赖项。可以使用pip命令来安装: ``` pip install channels ``` 然后,将Channels添加到您的Django项目的`INSTALLED_APPS`中。在您的项目的`settings.py`文件中进行如下配置: ```python INSTALLED_APPS = [ ... 'channels', ... ] ``` 接下来,创建一个名为`consumers.py`的文件,用于定义处理WebSockets连接的消费者: ```python from channels.generic.websocket import WebsocketConsumer class MyConsumer(WebsocketConsumer): def connect(self): # 连接成功时调用 self.accept() def disconnect(self, close_code): # 断开连接时调用 pass def receive(self, text_data): # 接收到来自客户端的消息时调用 pass def send_message(self, text_data): # 向客户端发送消息 self.send(text_data) ``` 在您的Django项目的`routing.py`文件中,配置将传入的WebSocket连接传递给相应的消费者: ```python from django.urls import path from . import consumers websocket_urlpatterns = [ path('ws/my-consumer/', consumers.MyConsumer.as_asgi()), ] ``` 最后,在您的Django项目的`settings.py`文件中启用Channels: ```python ASGI_APPLICATION = 'myapp.routing.application' ``` 现在,您可以在任何视图或其他地方使用以下代码来向客户端发送WebSocket消息: ```python from asgiref.sync import async_to_sync from channels.layers import get_channel_layer channel_layer = get_channel_layer() def send_socket_message(message): async_to_sync(channel_layer.group_send)('my-group', { 'type': 'send_message', 'text': message }) ``` 这将向名称为`my-group`的组中的所有客户端发送消息。 以上就是使用Django-Channels发送socket代码实现

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值