通过jdbc执行的存储过程

时间:2011-10-24 13:32:26

标签: mysql stored-procedures jdbc

我在mysql数据库上创建了一些存储过程,但是当我尝试执行它们时,我得到:

User does not have access to metadata required to determine stored procedure parameter types. If rights can not be granted, configure connection with "noAccessToProcedureBodies=true" to have driver generate parameters that represent INOUT strings irregardless of actual parameter types.

我添加到我的连接字符串noAccessToProcedureBodies = true,但我仍然收到此消息。我使用DriverManager.getConnection(url,user,password)方法,我使用的URL是

jdbc:mysql://[server-url]/[database-name]?noAccessToProcedureBodies=true 

编辑:我的意思是我有一个名为creautenti的用户,我在此方法中用来创建新用户:

public static boolean creazioneUtente(String user, String password) throws EccezioneDaMostrare{
    if(apriConnessione(CREA_UTENTI_USER, CREA_UTENTI_PW, false)){
        try{
            conn.setAutoCommit(false);
            String sqlCommand="CREATE USER ? @'%' IDENTIFIED BY PASSWORD ?";
            String sqlCommandGrant="GRANT EXECUTE ON salvataggi.* TO ? REQUIRE SSL;";
            String sqlCommandGrantEx="GRANT SELECT ON `mysql`.`proc` TO ? @'%';";
            PreparedStatement s=conn.prepareStatement(sqlCommand);
            PreparedStatement sGrant=conn.prepareStatement(sqlCommandGrant);
            PreparedStatement sGrantEx=conn.prepareStatement(sqlCommandGrantEx);
            s.setString(1, user);
            sGrant.setString(1, user);
            sGrantEx.setString(1, user);
            s.setString(2, mySQLPASSWORD(password));
            s.execute();
            sGrant.executeUpdate();
            sGrantEx.executeUpdate();
            conn.commit();
            conn.setAutoCommit(true);
            chiudiConnessione();
            return true;
        }
        catch(SQLException e){
            try {
                conn.rollback();
            } catch (SQLException e1) {
                String msg=String.format("Errore durante la connessione al server MySQL:\n Messaggio:%s \n Stato SQL:%s \n Codice errore:%s", e.getMessage(), e.getSQLState(), e.getErrorCode());
                throw new EccezioneDaMostrare(msg);
            }
            chiudiConnessione();
            return false;
        }
    }
    else
        return false;
}

(第一行只是打开与服务器的连接为creautenti,conn是Connection对象)。所以我在服务器上以root身份登录,并调用

GRANT SELECT ON `mysql`.`proc` TO 'creautenti'@'%' WITH GRANT OPTION;

相应更新了creautenti的特权,但是当我打电话时

GRANT SELECT ON `mysql`.`proc` TO '<username>'@'%';

以任何其他用户身份登录为creautenti,不会修改权限。所有例程都在salvataggi中,我给每个用户赋予EXECUTE权限,所以我认为这也不是问题所在。

3 个答案:

答案 0 :(得分:3)

您可能最好在mysql.proc表上向您的应用程序用户授予访问权限。因此,以root身份连接到MySQL数据库并运行以下命令:

GRANT SELECT ON `mysql`.`proc` TO '<username>'@'%';

您的Java应用程序应该能够查看正确的元数据,而无需指定noAccessToProcedureBodies=true

还要确保连接到数据库的用户对相关过程具有执行权限。同样,作为root用户或具有授予权限的用户:

GRANT EXECUTE ON PROCEDURE db.storedproc TO '<username>'@'%';
祝你好运!

答案 1 :(得分:3)

TLDR;将您的Callable语句更改为按索引而不是名称引用参数。

遇到类似问题后,我更新了我的JDBC驱动程序mysql-connector-java-5.1.26-bin.jar。 然后错误从

更改
  

用户无权访问确定存储所需的元数据   过程参数类型。如果无法授予权限,请配置   与“noAccessToProcedureBodies = true”连接以获得驱动程序   生成表示INOUT字符串的参数,无论如何   实际参数类型。

  

当连接配置为不访问过程主体时,无法按名称访问参数

我将我的Callable语句更改为通过索引而不是名称引用参数,并且嘿presto它可以工作。

当您没有元数据访问权限或例行正文访问时,可能不需要更新驱动程序,只需知道使用索引而不是名称。

祝你好运

答案 2 :(得分:1)

以下步骤在mysql中为我工作

第1步:添加/编辑以下

Connection con = DriverManager.getConnection(“jdbc:mysql:// ipaddress:3306 / logparser?noAccessToProcedureBodies = true”,“root”,“”);

第2步:执行以下查询 GRANT ALL ON logparser.proc TO root @'%';

其中logparser是DB名称,proc是过程名称。