客户端开发
本节内容介绍 VSOA 客户端使用 VSOA Datagram 的方法。
开发须知
客户端和服务端建立连接后,双方均可发送和接收 VSOA Datagram。发送使用直接调用函数方式;而接收使用事件机制,需要绑定一个接收回调函数。
常用接口
void vsoa_client_on_datagram(vsoa_client_t *client, vsoa_client_dat_func_t callback,
void *arg);
bool vsoa_client_datagram(vsoa_client_t *client, const vsoa_url_t *url,
const vsoa_payload_t *payload);
bool vsoa_client_quick_datagram(vsoa_client_t *client, const vsoa_url_t *url, const vsoa_payload_t *payload);
client.datagram(url, payload[, quick]);
client.on('datagram', function(url, payload, quick));
void Client.onDatagram(String url, Payload payload, boolean quick);
boolean client.datagram(String url, Payload payload, boolean quick);
func (client *Client) Call(URL string, mt protocol.MessageType, flags any, req *protocol.Message) (*protocol.Message, error)
client.datagram(url: str, payload: vsoa.Payload | dict = None, quick: bool = False) -> bool
client.ondata(cli: Client, url: str, payload: vsoa.Payload | dict = None, quick: bool = False)
说明:
VSOA Datagram 客户端的详细接口说明可参考以下手册:
- C 版本可参考 VSOA C 语言编程手册。
- JavaScript 版本可参考 VSOA JavaScript 编程手册。
- Java 版本可参考 VSOA Java 编程手册。
- Golang 版本可参考 VSOA Golang 编程手册。
- Python 版本可参考 VSOA Pyhon 编程手册。
开发示例
在本示例中,客户端执行以下流程:
- 连接 VSOA 服务器
vsoa_datagram_server
。 - 绑定一个 Datagram 接收事件的回调函数
datagram_callback
。 - 在回调函数中,打印收到的数据,并将数据原封不动发送给服务端。
运行成功时,客户端有如下输出:
- 在成功连接到 VSOA 服务器后,打印
Connected to VSOA datagram server
。 - 在成功接收服务器发送的数据后,打印
Datagram received from URL /axis with payload:
,后面带有收到的数据。
普通发送方式
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#ifdef SYLIXOS
#include <sys/vproc.h>
#endif
#include <sys/select.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <errno.h>
#include "vsoa_client.h"
#include "vsoa_cliauto.h"
#include "vsoa_position.h"
static void datagram_callback(void *arg, vsoa_client_t *client, vsoa_url_t *url,
vsoa_payload_t *payload, bool quick)
{
printf("Datagram received from URL %.*s with payload: %.*s\n",
(int)url->url_len, url->url, (int)payload->param_len, payload->param);
// send datagram to server (echo)
vsoa_client_datagram(client, url, payload);
}
static void connect_callback(void *arg, vsoa_client_auto_t *cliauto,
bool connect, const char *info)
{
vsoa_client_t *client;
vsoa_url_t url;
if (!connect) {
fprintf(stderr, "Cannot connect to VSOA stream server.\n");
return;
}
printf("Connected to VSOA datagram server.\n");
// get client handle
client = vsoa_client_auto_handle(cliauto);
// register datagram callback
vsoa_client_on_datagram(client, datagram_callback, NULL);
}
int main(int argc, char *argv[])
{
vsoa_client_auto_t *cliauto;
#ifdef SYLIXOS
vprocExitModeSet(getpid(), LW_VPROC_EXIT_FORCE);
#endif
/*
* Create client auto robot
*/
cliauto = vsoa_client_auto_create(NULL, NULL);
/*
* Add a connection callback
* The callback is called automatically when client is connected to the server.
*/
if (!vsoa_client_auto_setup(cliauto, connect_callback, NULL)) {
vsoa_client_auto_delete(cliauto);
fprintf(stderr, "Cannot register connect callback: %s (%d)\n", strerror(errno), errno);
return -1;
}
/*
* Client auto robot start
* The robot will automatically connect to the specified server and maintain the connection.
* At this time, the developer only needs to focus on the business.
*/
vsoa_client_auto_start(cliauto, "vsoa://vsoa_datagram_server", NULL, NULL, 0, 1000, 1000, 1000);
while (true) {
sleep(1);
}
return 0;
}
var vsoa = require('vsoa');
/* Server name to connect */
var SERVER_NAME = "vsoa_datagram_server"
/* Client */
var client = new vsoa.Client();
/* Receive datagram */
client.on('datagram', function(url, payload, quick) {
console.log('Datagram received from URL ' + url +
' with payload: ' + JSON.stringify(payload.param))
client.datagram(url, payload)
})
client.connect(`vsoa://${SERVER_NAME}`, error => {
if (error) {
console.error('Cannot connect to VSOA stream server.');
} else {
console.log('Connected to VSOA datagram server.')
}
});
/*
* Event loop
*/
require('iosched').forever();
import java.net.InetSocketAddress;
import com.acoinfo.vsoa.*;
import com.acoinfo.vsoa.Error;
public class datagram_client {
private static boolean POS_MANUALLY = true;
private static String SERVER_NAME = "vsoa_datagram_server";
private static String POS_ADDRESS = "0.0.0.0";
private static int POS_PORT = 3012;
public static Client client;
public static void main(String[] args) {
/*
* Initialize client
*/
client = new Client(new ClientOption(null, 6000, 4000, 3, false)) {
@Override
public void onError(Error error) {
System.out.println("Client error:" + error.message);
}
@Override
public void onConnected(String info) {
System.out.println("Connected to VSOA datagram server.");
}
@Override
protected void onDatagram(String url, Payload payload, boolean quick) {
System.out.println("Datagram received from URL " + url +
" with payload: " + payload.param);
this.datagram(url, payload);
}
};
if (!client.connect("vsoa://" + SERVER_NAME, null, Constant.VSOA_DEF_CONN_TIMEOUT)) {
System.out.println("Connected with server failed");
return;
}
while (true) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
package main
import (
"encoding/json"
"fmt"
"strconv"
"gitee.com/sylixos/go-vsoa/client"
"gitee.com/sylixos/go-vsoa/protocol"
)
var roll, pitch, yaw int = 1, 1, 1
// VsoaDatagramCall is a function that sends a datagram call using the VSOA protocol.
//
// It connects to the VSOA server at "localhost:3001" and sends a datagram message
// with the parameter `{"Test Datagarm": "Normal Channel"}`. The function prints an
// error if there was a problem connecting to the server or sending the datagram
// message. Otherwise, it prints "Datagram send done" to indicate a successful send.
func VsoaDatagramCall() {
// Set client options
clientOption := client.Option{
Password: "123456",
}
// Create a new client instance
c := client.NewClient(clientOption)
// Connect to the VSOA server
_, err := c.Connect("vsoa", "localhost:3001")
if err != nil {
fmt.Println(err)
return
}
// Close the connection when the function exits
defer c.Close()
// Create a new message
req := protocol.NewMessage()
// Set the message parameter
req.Param, _ = json.RawMessage(`{"roll":` + strconv.Itoa(roll) +
`, "pitch":` + strconv.Itoa(pitch) +
`, "yaw":` + strconv.Itoa(yaw) + `}`).MarshalJSON()
// Send the datagram call
_, err = c.Call("/axis", protocol.TypeDatagram, protocol.ChannelNormal, req)
if err != nil {
fmt.Println(err)
} else {
fmt.Println("Datagram /axis send done")
}
}
func main() {
VsoaDatagramCall()
}
from vsoa.client import Client
import vsoa
# Server name to connect
SERVER_NAME = 'vsoa_datagram_server'
client = Client()
# Receive datagram
def ondata(cli: Client, url: str, payload: vsoa.Payload | dict = None, quick: bool = False):
print('Datagram received from URL {} with payload: {}'.format(url, str(payload.param)))
cli.datagram(url, payload)
client.ondata = ondata
# Client connect
if err := client.connect('vsoa://{}'.format(SERVER_NAME)):
print('Connect error: {}'.format(err))
sys.exit(-1)
# Event loop
client.run()
说明:
- 在 Node.js 环境中,不需要最后的事件循环操作,需要去除
require('iosched').forever();
。- 在该范例中,因为使用了独立的位置服务或 ECSM 集成的位置服务,所以客户端可以自动定位到指定的服务位置,不需要指定任何 IP 和端口信息。
快速发送方式
C 语言环境
对于 C 语言环境,将如下接口进行替换即可:
// send datagram to server (echo)
vsoa_client_datagram(client, url, payload);
修改后:
// send datagram to server (echo)
vsoa_client_quick_datagram(client, url, payload);
JavaScript 语言环境
对于 JavaScript 语言环境 将 datagram
的最后一个参数设置为 true
即可:
client.datagram(url, payload);
修改后:
client.datagram(url, payload, true);
Java 语言环境
对于 Java 语言环境,将 datagram
的最后一个参数设置为 true
即可:
this.datagram(url, payload);
修改后:
this.datagram(url, payload, true);
Go 语言环境
对于 Go 语言环境,将参数 protocol.ChannelNormal
修改为 protocol.ChannelQuick
即可:
_, err = c.Call("/axis", protocol.TypeDatagram, protocol.ChannelNormal, req)
修改后:
_, err = c.Call("/axis", protocol.TypeDatagram, protocol.ChannelQuick, req)
Python 语言环境
对于 Python 语言环境,将 datagram
的最后一个参数设置为 true
即可:
cli.datagram(url, payload)
修改后:
cli.datagram(url, payload, true)