如何过滤 Informix 中的 dbaccess 输出?

How to filter dbaccess output in Informix?

我想 运行 dbaccess <dbname> <sqlfile.sql>,并将输出存储到 shell 变量。我知道有两种方法可以 (i) 输出到管道和 (ii) 卸载到文件。我想使用方法 (i) 种方法将查询输出存储到 shell 变量,但是随着查询输出我得到了不需要的东西(连接到数据库、列标题、断开连接)——参见图像随附的。我不想使用方法 (ii),因为我需要将查询输出存储到 shell 变量,而不是文件。请帮我解决这个问题。

一种方式,在某些情况下不是最好的方式是将 stderr 发送到 /dev/null

让我们创建一个 table 来测试它:

[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 访问权限,并且只有拥有 DBSECADM role can grant the SETSESSIONAUTH privilege, and only a 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 用户之外,只有 ricardoDBSA 的成员。 让我们为简单起见,坚持使用 informix

下一步是informixGRANT the DBSECADM角色,这是一个特殊的角色,会分布在所有数据库中,你不必做一个一个:

[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 ~]$

Ricardo Henriques in his 所述,您可以通过重定向标准错误来完成一定数量的工作。

还要考虑 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 程序无关——除了名称。

我要补充前面所说的,首先将 STDERR 输出为 null,然后 grep -v for $^。

IE,select 来自数据库的当前日期和时间:

$> 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
$>

您只需要确保您的查询 returns 只有一行,一列。