当前位置: 首页 > 知识库问答 >
问题:

如何使uCanAccess使用Samba身份验证,并在用户名或密码中添加特殊字符?

丘友樵
2023-03-14

TL;DR:在创建数据库对象时,对于MS Access 2000-2003数据库,应该使用什么database.fileformat常量?

我还有一个应用程序,它使用UCANAccess/JackCess连接到网络共享上的MDB。然而(据我所知),它使用登录用户的凭据,其中许多用户具有只读访问权限。只有系统/网络管理员才有写权限。

有问题的数据库没有密码保护。(打开时我不需要输入密码。)

我的目的是让应用程序在向DB写入之前使用uCanAccess连接中的Samba凭据询问管理员的Samba凭据,这样它就不会抛出java.nio.channels.nonwritableChannelException,如下所示:

java.nio.channels.NonWritableChannelException
  at sun.nio.ch.FileChannelImpl.write(FileChannelImpl.java:747)
  at com.healthmarketscience.jackcess.impl.PageChannel.writePage(PageChannel.java:310)
  at com.healthmarketscience.jackcess.impl.PageChannel.writePage(PageChannel.java:247)
  at com.healthmarketscience.jackcess.impl.TableImpl.writeDataPage(TableImpl.java:1980)
  at com.healthmarketscience.jackcess.impl.TableImpl.addRows(TableImpl.java:2229)
  at com.healthmarketscience.jackcess.impl.TableImpl.addRow(TableImpl.java:2067)
  at net.ucanaccess.converters.UcanaccessTable.addRow(UcanaccessTable.java:44)
  at net.ucanaccess.commands.InsertCommand.insertRow(InsertCommand.java:101)
  at net.ucanaccess.commands.InsertCommand.persist(InsertCommand.java:148)
  at net.ucanaccess.jdbc.UcanaccessConnection.flushIO(UcanaccessConnection.java:315)
  at net.ucanaccess.jdbc.UcanaccessConnection.commit(UcanaccessConnection.java:205)
  at net.ucanaccess.jdbc.AbstractExecute.executeBase(AbstractExecute.java:217)
  at net.ucanaccess.jdbc.Execute.execute(Execute.java:46)
  at net.ucanaccess.jdbc.UcanaccessPreparedStatement.execute(UcanaccessPreparedStatement.java:228)
  at myapp.db.Digger.addTransaction(Digger.java:993)
  at myapp.tasks.TransactionRunnable.run(TransactionRunnable.java:42)
  at java.lang.Thread.run(Thread.java:745)
// Ask the user for login credentials and the path to the database
String smbURL = (chosenDir.endsWith("/") ? chosenDir : chosenDir + '/') 
  + dbName;
System.out.println("DB Path to use for URL: " + smbURL);
URL u = new URL(smbURL);
try (
  // construct the SMB DB URL
  SmbFileChannel sfc = new SmbFileChannel(smbURL);
  Database db = new DatabaseBuilder().setChannel(sfc)
    .setFileFormat(Database.FileFormat.GENERIC_JET4).create();
) {
  // Model the table
  Table tbl = new TableBuilder("Transactions")
    .addColumn(new ColumnBuilder("TransactionID", DataType.LONG).setAutoNumber(true))
    .addColumn(new ColumnBuilder("ControllerID", DataType.LONG).setAutoNumber(false))
    .addColumn(new ColumnBuilder("ReaderID", DataType.LONG).setAutoNumber(false))
    .addColumn(new ColumnBuilder("Event", DataType.LONG).setAutoNumber(false))
    .addColumn(new ColumnBuilder("Timestamp", DataType.SHORT_DATE_TIME).setAutoNumber(false))
    .addColumn(new ColumnBuilder("Number", DataType.LONG).setAutoNumber(false))
    .addIndex(new IndexBuilder(IndexBuilder.PRIMARY_KEY_NAME).addColumns("TransactionID").setPrimaryKey())
    .toTable(db);
  // Add the row
  Map<String, Object> values = new HashMap<>();
  values.put("ControllerID", cid);
  values.put("ReaderID", rid);
  values.put("Event", evtNum);
  values.put("Timestamp", ts); // Long; must be converted to DataType.SHORT_DATE_TIME
  values.put("Number", accNum);
  tbl.addRowFromMap(values);
} catch (IOException IOEx) {
  System.err.println(
    "Failed to write record to Transactions table in database: " 
    + IOEx.getMessage()
   );
   IOEx.printStackTrace(System.err);
  } catch (Exception ex) {
    System.err.println(
      '[' + ex.getClass().getSimpleName() + "]: Failed to write record to "
      + "Transactions table in database: " + ex.getMessage()
    );
    ex.printStackTrace(System.err);
  }
DB Path to use for URL: smb://machine.vpnName/Storage/me/dbs/DBName.mdb
Failed to write record to Transactions table in database: Logon failure: account currently disabled.
jcifs.smb.SmbAuthException: Logon failure: account currently disabled.
   at jcifs.smb.SmbTransport.checkStatus(SmbTransport.java:546)
    at jcifs.smb.SmbTransport.send(SmbTransport.java:663)
    at jcifs.smb.SmbSession.sessionSetup(SmbSession.java:390)
    at jcifs.smb.SmbSession.send(SmbSession.java:218)
    at jcifs.smb.SmbTree.treeConnect(SmbTree.java:176)
    at jcifs.smb.SmbFile.doConnect(SmbFile.java:911)
    at jcifs.smb.SmbFile.connect(SmbFile.java:957)
    at jcifs.smb.SmbFile.connect0(SmbFile.java:880)
    at jcifs.smb.SmbFile.open0(SmbFile.java:975)
    at jcifs.smb.SmbFile.open(SmbFile.java:1009)
    at jcifs.smb.SmbRandomAccessFile.<init>(SmbRandomAccessFile.java:57)
    at jcifs.smb.SmbRandomAccessFile.<init>(SmbRandomAccessFile.java:42)
    at samba.SmbFileChannel.<init>(SmbFileChannel.java:30)
    at samba.SambaLanWriteTest.writeTest(SambaLanWriteTest.java:130)
    at samba.SambaLanWriteTest.main(SambaLanWriteTest.java:181)

在使用Windows文件资源管理器时,我对有问题的数据库文件的测试副本具有写权限。当有人提示时,我选择了那一个。

更新2:我意识到我忽略了将用户名和密码添加到smb://URL,如Thompson的示例所示。我将代码改为:

String smbCred = "smb://" + auth.getUsername() + ":" + auth.getPassword() + "@",
  fixer = chosenDir.replace("\\", "/").replace("smb://", smbCred),
  smbURL = fixer + dbName;
System.out.println("DB Path to use for URL: " + smbURL);
// URL u = new URL(smbURL);

我遇到的下一个问题是我的密码包含 特殊的非法字符(例如'@'、':'、';'、'='和‘?')。我通过在auth.getusername()auth.getpassword()上使用java.net.urlencoder.encode()对这些代码进行了转义,以便在创建SMBChannel时代码不会抛出malformedurlexception。然而,我遇到的下一个异常如下:

Failed to write record to Transactions table in database: File format GENERIC_JET4 [VERSION_4] does not support file creation for null
java.io.IOException: File format GENERIC_JET4 [VERSION_4] does not support file creation for null
    at com.healthmarketscience.jackcess.impl.DatabaseImpl.create(DatabaseImpl.java:444)

在创建database对象时,应该为MS Access 2000-2003数据库使用什么database.fileformat常量?

共有1个答案

潘自强
2023-03-14

原来我需要使用database.fileformat.v2000

之后,一切都一帆风顺(尽管我仍然需要找出如何获得长时间戳以正确转换)。

 类似资料:
  • 客户已创建密钥存储库并存储凭据。为了验证密钥存储库,我已经在节点中创建了应用程序,并使用客户端id和客户端机密,我能够读取机密。但是现在客户不想使用客户id和客户机密,而是使用AZURE的用户名和密码来访问程序中的keyvault。它是一个没有MFA的keyvault访问的专用用户。 我不确定我们是否可以从节点JS使用用户名和密码访问keyvault。敬请建议。 谢谢

  • 我是Java的初学者,目前我有这样的代码。 规则是:没有特殊字符,只有小写或0到9之间的数字。 这段代码工作正常,但也需要特殊字符。有没有其他方法可以阻止这种情况发生?请不要正则表达式。

  • 是否可以在Flatter中使用用户名和密码(而不是电子邮件和密码)实现Firebase身份验证?有没有办法用Firebase Auth插件来实现这一点?

  • 我有一个用户名和密码,我怎么用他们来验证库贝特尔? 我要运行哪个命令? 我已经通读了https://kubernetes.io/docs/reference/access-authn-authz/authorization/和https://kubernetes.io/docs/reference/access-authn-authz/authorization/,但在其中找不到任何与本例相关的信

  • 我已经安装了OpenLDAP服务器。 如何在LDAP服务器中添加用户(条目)?以及如何打开该服务器的命令窗口,以便在其上运行ldap命令: :我的要求是:在我的应用程序中,我想对Openldap服务器中添加的用户进行身份验证,但我只能在windows os上安装Openldap服务器,但如何添加用户我无法获得方法。在Windows7操作系统上安装openLDAP whic时添加用户的方法是什么?这

  • 我想在Go中使用AES-256加密一个字符串,无需任何GCM处理,以与MQL4进行比较。当我试图加密特殊字符或数字时,我会遇到问题。我应该以某种方式预处理我的明文吗?我是新来的,所以任何帮助都将不胜感激;我的代码在这个解释下面。 如果我加密明文“这是一个秘密”,然后解密密文(编码为十六进制),我会得到相同的结果(即“这是一个秘密”)。pt是下面代码中明文的变量名。 如果我试图加密“这是一个秘密;1