2014.8.12-AKKA和Actor model 分布式开发环境学习小结

作者分享了一年的AKKA学习经历,包括使用Java编写程序、官方文档阅读、书籍《AKKAEssentials》的学习、编译配置、版本升级、多服务器集群部署等方面的经验。介绍了两种编译方式:使用sbt命令行和maven插件,并提供了相关链接和教程。

学习使用AKKA 断断续续有一年了。目前还是习惯用java来写akka下面的程序。对于原生的scala还是没有时间和兴趣去学习它。 毕竟学习一门语言需要兴趣和时间的。 


AKKA学习资源还是不算丰富。 看过最多的就是官方的编程手册,还有就是AKKA Essentials 这两本。  自己动手写的程序还不算多,也放在github上面。 另外,在akka编译配置、升级版本上,以及部署多台服务器组建akka cluster 方面花费了不少时间。因为项目需要,上周重新在办公室用两台mac台式机和一台thinkpad t400组建了一个akka cluster 分布式开发环境。


akka项目编译上有两种方法:

第一、 用sbt ,这个是akka 官方推荐的方法,可以用sbt生成Eclipsse项目,然后导入到Eclipse里面,可以运行。 但是我的编译还是通过sbt 命令行的方式来做的。 如果大家有好的方法,请指正。 

sbt的安装方法参考:http://www.scala-sbt.org/download.html

how to write a scala file , build and run  with sbt.
http://www.scala-sbt.org/0.13/tutorial/Hello.html

akka:http://akka.io/

scala for eclipse IDE bundle : http://scala-ide.org/  (也推荐这个集成开发工具,内含编译所需要的akka actor 和 scala library)


第二、是用maven, 这是akka essentials 这本书所采用的。 我亲自实验过,可以编译akka 2.0.5, 2.1.2的旧有项目。 而且这本书的大多数例子,我都验证过。 所以,如果是想从头到尾、编译调试程序,不妨试试maven 的elipse 插件。

安装mvn  http://maven.apache.org/download.cgi 然后在readme安装步骤做。

Akka-Essentials 源代码 https://github.com/write2munish/Akka-Essentials

参考: 

Maven实战(三)Eclipse构建Maven项目
http://tangyanbo.iteye.com/blog/1503782



snailjob连接服务端报错2025-05-28 10:12:59 [snail-job-grpc-client-executor-127.0.0.1-1] ERROR c.a.s.c.c.handler.ClientRegister - heartbeat check requestId:[0] message:[INTERNAL: http2 exception]客户端报错2025-05-28 10:25:16 [NETTY_ACTOR_SYSTEM-akka.actor.netty-receive-request-dispatcher-14] ERROR c.a.s.s.c.r.s.RequestHandlerActor - http request error. [] com.aizuda.snailjob.server.common.exception.SnailJobServerException: Token authentication failed. [namespace:null groupName:null token:null] at com.aizuda.snailjob.server.common.rpc.server.RequestHandlerActor.doProcess(RequestHandlerActor.java:84) at com.aizuda.snailjob.server.common.rpc.server.RequestHandlerActor.lambda$createReceive$0(RequestHandlerActor.java:63) at akka.japi.pf.UnitCaseStatement.apply(CaseStatements.scala:24) at akka.japi.pf.UnitCaseStatement.apply(CaseStatements.scala:20) at scala.PartialFunction.applyOrElse(PartialFunction.scala:214) at scala.PartialFunction.applyOrElse$(PartialFunction.scala:213) at akka.japi.pf.UnitCaseStatement.applyOrElse(CaseStatements.scala:20) at scala.PartialFunction$OrElse.applyOrElse(PartialFunction.scala:269) at akka.actor.Actor.aroundReceive(Actor.scala:537) at akka.actor.Actor.aroundReceive$(Actor.scala:535) at akka.actor.AbstractActor.aroundReceive(AbstractActor.scala:220) at akka.actor.ActorCell.receiveMessage(ActorCell.scala:579) at akka.actor.ActorCell.invoke(ActorCell.scala:547) at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:270) at akka.dispatch.Mailbox.run(Mailbox.scala:231) at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1144) at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:642) at java.base/java.lang.Thread.run(Thread.java:1583) 2025-05-28 10:25:16 [NETTY_ACTOR_SYSTEM-akka.actor.default-dispatcher-10] ERROR akka.actor.OneForOneStrategy - Json转对象失败 参数: com.aizuda.snailjob.common.core.exception.SnailJobCommonException: Json转对象失败 参数: at com.aizuda.snailjob.common.core.util.JsonUtil$JsonMapper.toJavaObject(JsonUtil.java:285) at com.aizuda.snailjob.common.core.util.JsonUtil.parseObject(JsonUtil.java:53) at com.aizuda.snailjob.server.common.rpc.server.RequestHandlerActor.lambda$createReceive$0(RequestHandlerActor.java:66) at akka.japi.pf.UnitCaseStatement.apply(CaseStatements.scala:24) at akka.japi.pf.UnitCaseStatement.apply(CaseStatements.scala:20) at scala.PartialFunction.applyOrElse(PartialFunction.scala:214) at scala.PartialFunction.applyOrElse$(PartialFunction.scala:213) at akka.japi.pf.UnitCaseStatement.applyOrElse(CaseStatements.scala:20) at scala.PartialFunction$OrElse.applyOrElse(PartialFunction.scala:269) at akka.actor.Actor.aroundReceive(Actor.scala:537) at akka.actor.Actor.aroundReceive$(Actor.scala:535) at akka.actor.AbstractActor.aroundReceive(AbstractActor.scala:220) at akka.actor.ActorCell.receiveMessage(ActorCell.scala:579) at akka.actor.ActorCell.invoke(ActorCell.scala:547) at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:270) at akka.dispatch.Mailbox.run(Mailbox.scala:231) at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1144) at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:642) at java.base/java.lang.Thread.run(Thread.java:1583) Caused by: com.fasterxml.jackson.databind.exc.MismatchedInputException: No content to map due to end-of-input at [Source: REDACTED (`StreamReadFeature.INCLUDE_SOURCE_IN_LOCATION` disabled); line: 1] at com.fasterxml.jackson.databind.exc.MismatchedInputException.from(MismatchedInputException.java:59) at com.fasterxml.jackson.databind.ObjectMapper._initForReading(ObjectMapper.java:5008) at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:4910) at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3860) at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3828) at com.aizuda.snailjob.common.core.util.JsonUtil$JsonMapper.toJavaObject(JsonUtil.java:283) ... 18 common frames omitted 2025-05-28 10:25:16 [NETTY_ACTOR_SYSTEM-akka.actor.default-dispatcher-10] INFO akka.actor.RepointableActorRef - Message [akka.dispatch.sysmsg.Recreate] from Actor[akka://NETTY_ACTOR_SYSTEM/user/$e#607696578] to Actor[akka://NETTY_ACTOR_SYSTEM/user/$e#607696578] was not delivered. [5] dead letters encountered. If this is not an expected behavior then Actor[akka://NETTY_ACTOR_SYSTEM/user/$e#607696578] may have terminated unexpectedly. This logging can be turned off or adjusted with configuration settings 'akka.log-dead-letters' and 'akka.log-dead-letters-during-shutdown'. 2025-05-28 10:25:16 [NETTY_ACTOR_SYSTEM-akka.actor.default-dispatcher-10] ERROR akka.actor.OneForOneStrategy - Json转对象失败 参数: com.aizuda.snailjob.common.core.exception.SnailJobCommonException: Json转对象失败 参数: at com.aizuda.snailjob.common.core.util.JsonUtil$JsonMapper.toJavaObject(JsonUtil.java:285) at com.aizuda.snailjob.common.core.util.JsonUtil.parseObject(JsonUtil.java:53) at com.aizuda.snailjob.server.common.rpc.server.RequestHandlerActor.lambda$createReceive$0(RequestHandlerActor.java:66) at akka.japi.pf.UnitCaseStatement.apply(CaseStatements.scala:24) at akka.japi.pf.UnitCaseStatement.apply(CaseStatements.scala:20) at scala.PartialFunction.applyOrElse(PartialFunction.scala:214) at scala.PartialFunction.applyOrElse$(PartialFunction.scala:213) at akka.japi.pf.UnitCaseStatement.applyOrElse(CaseStatements.scala:20) at scala.PartialFunction$OrElse.applyOrElse(PartialFunction.scala:269) at akka.actor.Actor.aroundReceive(Actor.scala:537) at akka.actor.Actor.aroundReceive$(Actor.scala:535) at akka.actor.AbstractActor.aroundReceive(AbstractActor.scala:220) at akka.actor.ActorCell.receiveMessage(ActorCell.scala:579) at akka.actor.ActorCell.invoke(ActorCell.scala:547) at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:270) at akka.dispatch.Mailbox.run(Mailbox.scala:231) at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1144) at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:642) at java.base/java.lang.Thread.run(Thread.java:1583) Caused by: com.fasterxml.jackson.databind.exc.MismatchedInputException: No content to map due to end-of-input at [Source: REDACTED (`StreamReadFeature.INCLUDE_SOURCE_IN_LOCATION` disabled); line: 1] at com.fasterxml.jackson.databind.exc.MismatchedInputException.from(MismatchedInputException.java:59) at com.fasterxml.jackson.databind.ObjectMapper._initForReading(ObjectMapper.java:5008) at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:4910) at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3860) at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3828) at com.aizuda.snailjob.common.core.util.JsonUtil$JsonMapper.toJavaObject(JsonUtil.java:283) ... 18 common frames omitted
最新发布
05-29
<think>好的,用户遇到了SnailJob客户端连接服务端时的两个错误:Token认证失败JSON转换异常。我需要结合提供的引用信息常见解决方案来思考如何解决这些问题。首先,关于Token认证失败。引用[4]提到JWT的结构,可能客户端生成的Token格式不正确或者签名密钥不匹配。需要检查客户端生成Token的方式是否正确,比如使用的算法、密钥是否与服务端一致。另外,Token的有效期是否过期也是一个可能的原因。还要确认请求头中的Authorization是否正确携带了Token,格式是否为Bearer加上Token。然后是JSON转换异常,Jackson的MismatchedInputException。引用[1][3]涉及JSON转换问题,可能客户端接收的JSON数据与目标对象结构不匹配。比如服务端返回的JSON字段与客户端的Account类字段不一致,或者有嵌套结构未正确处理。需要检查反序列化的类是否与服务端的数据结构一致,特别是字段名数据类型。同时,日期格式可能需要使用@JsonFormat注解来指定,如引用[3]中的例子。另外,引用[2]中的ResponseProvider类处理响应时使用了JsonUtil.toJson,如果这里出现异常,可能涉及到JSON序列化的配置问题。需要确认是否所有需要序列化的字段都有正确的gettersetter方法,或者是否使用了正确的Jackson注解,比如@JsonProperty。 可能还需要检查网络通信问题,比如HTTPS证书问题或者代理设置导致的数据截断,从而引发JSON解析错误。日志分析是关键,开启SnailJob客户端的DEBUG日志,查看具体的错误堆栈,定位是哪个环节出了问题。 最后,建议用户分步骤排查,先解决Token认证问题,确保Token生成传递正确,再处理JSON转换异常,检查数据结构反序列化配置。同时,参考官方文档社区案例,确认是否有已知的兼容性问题或配置要求。</think>针对SnailJob客户端连接服务端时出现的Token认证失败JSON转换异常问题,请按以下步骤排查: ### 一、Token认证失败解决方案 1. **检查Token生成逻辑** - 确保客户端生成JWT时使用的**签名算法**(如HS256)**密钥**与服务端配置完全一致 - 验证Token包含的必要声明字段(如exp、iat、sub)是否符合服务端要求 - 示例Token结构验证: $$header.payload.signature$$ 需符合RFC 7519标准[^4] 2. **验证Token传输方式** ```java // 正确设置Authorization请求头示例 request.setHeader("Authorization", "Bearer " + jwtToken); ``` - 确认使用`Bearer`认证模式且没有多余的空格 - 通过Wireshark抓包验证Token是否被完整传输 3. **检查时间同步** - 确保客户端与服务端的系统时间误差在**30秒以内** - 在JWT中加入时间容差配置(如leeway参数) ### 二、JSON转换异常解决方案 1. **字段映射检查** - 对比服务端返回的JSON数据结构与客户端反序列化类(如Account.java[^3]) - 使用`@JsonProperty`注解强制字段映射: ```java @JsonProperty("user_name") private String username; ``` 2. **日期格式处理** - 对日期字段必须添加格式注解: ```java @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") private Date createTime; ``` 3. **空安全处理** - 在反序列化代码中添加空值检测: ```python // Java示例 if (StringUtils.isNotBlank(jsonStr)) { objectMapper.readValue(jsonStr, TargetClass.class); } ``` ### 三、综合排查步骤 1. **开启DEBUG日志** - 在客户端配置中增加日志级别配置: ```properties logging.level.com.snailjob=DEBUG logging.level.org.springframework.web=DEBUG ``` 2. **网络层验证** - 使用Postman直接调用服务端接口,排除客户端代码问题 - 检查HTTPS证书是否过期(特别是自签名证书场景) 3. **版本兼容性检查** - 确认客户端与服务端的SnailJob版本匹配 - 检查Jackson库版本是否一致(推荐2.12.x+)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值