一、背景
使用smbj实现samba有个问题,不能先列出服务器的所有samba账户。当面对某些场景,例如先列出局域网所有的samba账户,再让用户选择其中一个,然后让用户输入密码,完成登录再浏览文件。samba代码交叉编译出android平台可以用的smbclient非常困难,很难实现像LINUX平台smbclient -L这样的列出结果。
二、使用jcifs与smbj联合解决
可以使用老版本jcifs库先匿名访问服务器。列出所有用户后,根据用户的操作(选择一个账户,并输入密码)。但是jcifs仅支持samba 1.0。之后再使用支持较高版本的smbj接口完成浏览。
jcifs匿名访问代码:
import jcifs.smb.NtlmPasswordAuthentication;
import jcifs.smb.SmbFile;
String remoteUrl= "smb://10.10.10.1";
SmbFile smbFile = new SmbFile(remoteUrl, NtlmPasswordAuthentication.ANONYMOUS);
for(SmbFile smbfile: smbFile.listFiles("*")){
String username = smbfile.getName();
System.out.println("user name: " + username);
}
上面的username可以存成一个列表,滤掉带$的打印机等用户名,每个username可以作为参数,加上密码,配合smbj代码完成单个账户的访问:
SmbConfig config = SmbConfig.builder().withTimeout(120, TimeUnit.SECONDS)
.withTimeout(120, TimeUnit.SECONDS) // 超时设置读,写和Transact超时(默认为60秒)
.withSoTimeout(180, TimeUnit.SECONDS) // Socket超时(默认为0秒)
.build();
SMBClient client = new SMBClient(config);
Connection connection = null;
try {
connection = client.connect("10.10.10.1");
} catch (IOException ioException) {
ioException.printStackTrace();
}
AuthenticationContext ac = new AuthenticationContext("username", "password".toCharArray(), "10.10.10.1");
assert connection != null;
Session session = connection.authenticate(ac);
DiskShare share = (DiskShare) session.connectShare("username");
//如果要访问用户username目录里面的仅share文件夹,则
// share.list(null, "*.*") 改成 share.list("share", "*.*")
//share.list(null, "*.*")表示该用户名下全部文件和文件夹
for (FileIdBothDirectoryInformation f : share.list(null, "*.*")) {
String fname = f.getFileName();
System.out.println("File : " + fname);
}
其他:
需要使用1.X版本的jcifs.jar.以下链接的jcifs亲测完美支持。下载jar后,把jcifs-1.3.18.3.jar放在代码build.gradle同级新建的文件夹libs中。模块的build.gradle里面加上包的导入:
implementation files('libs/jcifs-1.3.18.3.jar')
jcifs-1.3.18用于android访问samba服务器。支持匿名访问,实现smbclient-L列出服务器用户名-Android文档类资源-CSDN文库