如何在 Informix 中过滤 dbaccess 输出?
Posted
技术标签:
【中文标题】如何在 Informix 中过滤 dbaccess 输出?【英文标题】:How to filter dbaccess output in Informix? 【发布时间】:2016-05-12 12:45:05 【问题描述】:我想运行dbaccess <dbname> <sqlfile.sql>
,并将输出存储到一个shell 变量。我知道有两种方法可以(i)输出到管道和(ii)卸载到文件。我想使用方法 (i) 的方法将查询输出存储到 shell 变量中,但是随着查询输出,我得到了不需要的东西(连接到数据库、列标题、断开连接)——参见附图。我不想使用方法 (ii),因为我需要将查询输出存储到 shell 变量,而不是文件。请帮我解决这个问题。
【问题讨论】:
【参考方案1】:一种方法,在某些情况下不是最好的方法是将 stderr 发送到/dev/null
。
让我们创建一个表来测试它:
[infx1210@tardis ~]$ dbaccess demo -
Database selected.
> CREATE TABLE starc (col1 INT, col2 INT);
Table created.
> INSERT INTO starc VALUES (1,1);
1 row(s) inserted.
> INSERT INTO starc VALUES (2,2);
1 row(s) inserted.
>
Database closed.
[infx1210@tardis ~]$
对于一列一行,或者更多,这就足够了:
[infx1210@tardis ~]$ out_1r1c=`echo "select col1 FROM starc WHERE col1 = 1" | dbaccess demo 2>/dev/null | uniq`
[infx1210@tardis ~]$ echo $out_1r1c
col1 1
[infx1210@tardis ~]$ out_2r1c=`echo "select col1 FROM starc" | dbaccess demo 2>/dev/null | uniq`
[infx1210@tardis ~]$ echo $out_2r1c
col1 1 2
[infx1210@tardis ~]$
对于不止一列,可能不是最佳选择:
[infx1210@tardis ~]$ out_1r2c=`echo "select * FROM starc WHERE col1 = 1" | dbaccess demo 2>/dev/null | uniq`
[infx1210@tardis ~]$ echo $out_1r2c
col1 col2 1 1
[infx1210@tardis ~]$ out_2r2c=`echo "select * FROM starc" | dbaccess demo 2>/dev/null | uniq`
[infx1210@tardis ~]$ echo $out_2r2c
col1 col2 1 1 2 2
[infx1210@tardis ~]$
跟进问题
对于您正在做的事情,只需将eco
命令的信息传递给 SQL 文件脚本并执行它。
例如:
[infx1210@tardis ~]$ echo "CONNECT TO 'sysmaster@infx1210' USER 'starc' USING '$PASSWD'; SELECT USER FROM sysdual;" | dbaccess -
32412: USING clause unsupported. DB-Access will prompt you for a password.
Error in line 1
Near character position 45
[infx1210@tardis ~]$ finderr 32412
-32412 USING clause unsupported. DB-Access will prompt you for a password.
DB-Access does not support the USING password clause in a CONNECT ...
USER statement when it violates security. For example, do not type a
password on the screen where it can be seen or include it in a command
file that someone other than the user can read. To maintain security,
DB-Access prompts you to enter the password on the screen and uses echo
suppression to hide it from view.
[infx1210@tardis ~]$ echo "CONNECT TO 'sysmaster@infx1210' USER 'starc' USING '$PASSWD'; SELECT USER FROM sysdual;" > file.sql
[infx1210@tardis ~]$ dbaccess - file.sql 2>> test.log
(expression)
starc
[infx1210@tardis ~]$
我不喜欢这种方法。您应该考虑使用SET SESSION AUTHORIZATION 语句。
现在要让用户使用它 DBA 必须授予数据库级权限,并且还需要 SETSESSIONAUTH 访问权限,并且只有拥有@987654322 的用户@角色可以授予SETSESSIONAUTH权限,只有DBSA可以授予用户DBSECADM角色。
通常,拥有 $INFORMXIDR/etc 的 OS 组的成员是 DBSA,在这种情况下:
[infx1210@tardis ~]$ ls -ld $INFORMIXDIR/etc
drwxrwxr-x. 5 informix informix 4096 May 18 13:33 /opt/IBM/informix/V12.1/etc
[infx1210@tardis ~]$ grep informix /etc/group
informix:x:501:ricardo
[infx1210@tardis ~]$
所以,除了 informix 用户之外,只有 ricardo 是 DBSA 的成员。让我们坚持使用 informix简单。
下一步是GRANT the DBSECADM角色为informix,这是一个特殊的角色,将分布在所有数据库中,你不必一个一个地去做:
[infx1210@tardis ~]$ echo "GRANT DBSECADM TO 'informix'" | dbaccess sysmaster
Database selected.
DBSECADM granted.
Database closed.
[infx1210@tardis ~]$
现在,SETSESSIONAUTH 不能给用户自己,所以让我们给 ricardo:
[infx1210@tardis ~]$ echo "GRANT SETSESSIONAUTH ON 'starc' TO 'ricardo'" | dbaccess demo
Database selected.
SETSESSIONAUTH privilege granted.
Database closed.
[infx1210@tardis ~]$
切换到用户ricardo,记住它应该有DBA权限,我们现在可以:
[infx1210@tardis ~]$ echo "SET SESSION AUTHORIZATION TO 'starc'; SELECT USER FROM systables WHERE tabid = 1;" | dbaccess demo 2>>/dev/null
(expression)
starc
[infx1210@tardis ~]$
【讨论】:
非常感谢@henriques,你一如既往的好,真的很有用。 嗨,在(out_2r2c=echo "select * FROM starc" | dbaccess demo 2>/dev/null | uniq
)为什么我们使用“2>”并且当我包含时我没有得到输出(使用'pwd'连接到db@instance用户'un'; ) 在查询中通过从 dbaccess dbname 2> 中删除“demo”...即我想知道我们是否可以以用户身份连接并执行查询?请澄清我的疑问。
2>/dev/null
会将 stdout 重定向到位桶,这意味着如果您遇到错误,您将看不到它。将其更改为 2>>query.err
以将其重定向到实际文件并查看其错误。更改后显示您传递的实际命令。以及文件上的错误。
@henriques ,发现错误 "32412: USING 子句不受支持。DB-Access 会提示您输入密码。第 1 行错误" ,有没有办法以上述方式执行查询说并包括使用密码'$password'连接到'db@server'用户'$username',我的意思是作为用户连接到数据库并代表该用户执行查询?
@starc 检查跟进问题部分。【参考方案2】:
正如Ricardo Henriques 在他的answer 中所指出的,您可以通过重定向标准错误来做一定的事情。
同时考虑OUTPUT 语句:
OUTPUT TO "/dev/stdout" WITHOUT HEADINGS
SELECT * FROM YourTable WHERE …
或UNLOAD 声明:
UNLOAD TO "/dev/stdout"
SELECT * FROM YourTable WHERE …
使用"/dev/stdout"
是一种技巧——有时很有用。您可以在那里指定任何文件名。您可能仍希望重定向错误。请注意,DB-Access 在出错后会出错 — 您可以通过在环境中设置 DBACCNOIGN=1
来阻止它这样做。
另外,请考虑查看我写的 SQLCMD,因为它在 shell 脚本上下文中运行,而 DB-Access 没有。它可以追溯到 1986 年(之前还有 dbaccess
;在那些日子里,您使用的是 isql
— DB-Access 是在一个晚上从 isql
中雕刻出来的)。当前版本是 SQLCMD 90.00 (2015-11-08)。它与 Microsoft 的同名 johnny-come-lately 程序没有任何关系——除了名称。
【讨论】:
尝试了您和@henriques 建议的两种方法都有效,您总是激励我并推动我学习informix。【参考方案3】:我要补充前面所说的,首先将 STDERR 输出为 null,然后 grep -v for $^。
IE,从数据库中选择当前日期和时间:
$> echo "output to /dev/stdout without headings select first 1 current from systables;" > query.sql
$> FOO=`dbaccess mydb query.sql 2>/dev/null | grep -v "^$"`
$> echo $FOO
2017-07-27 14:25:30.000
$>
您只需要确保您的查询只返回一行、一列。
【讨论】:
以上是关于如何在 Informix 中过滤 dbaccess 输出?的主要内容,如果未能解决你的问题,请参考以下文章