|
Interesting thought, Daniel - in order to call procedures directly, we'd
need to know what the program is behind it - sometimes we might know it,
but most of the time we don't even go looking for it. And we would need
to know the calling style - which we can derive with a generate_sql kind
of thing, I suppose. But there is no reason to do any of this, seems to me.
So this is something SQL does __for__ us, making things easier, perhaps,
than what we might do otherwise.
Regards
Vern
On 2/4/2022 3:37 PM, Daniel Gross wrote:
Greg,examples Birgitta and me posted.
you can only use these SQL functions inside an EXEC SQL, like in der
does not provide prototypes, so you would have to write them yourself. But
Of course you can try to call the procedures directly, but AFAIK IBM
it is so much easier to use these function from SQL.
gwilburn@xxxxxxxxxxxxxxxxxxxxxxx>:
HTH
Daniel
Am 04.02.2022 um 22:07 schrieb Greg Wilburn <
RPG?
Daniel,
Thank you for the response.
I do have IFS functions in qsys2… do these need prototyped for use in
more files to limit number of arrays.SPECIFIC_NAME
ROUTINE_SCHEMA
ROUTINE_NAME
ROUTINE_TYPE
QSQIFSJOBI
QSYS2
IFS_JOB_INFO
FUNCTION
QSQIFSOBJL
QSYS2
IFS_OBJECT_LOCK_INFO
FUNCTION
QSQIFSOBJP
QSYS2
IFS_OBJECT_PRIVILEGES
FUNCTION
QSQIFSOBJI
QSYS2
IFS_OBJECT_REFERENCES_INFO
FUNCTION
QSQIFSOBJS
QSYS2
IFS_OBJECT_STATISTICS
FUNCTION
QSQIFSREAD
QSYS2
IFS_READ
FUNCTION
QSQIFSREADBINARY
QSYS2
IFS_READ_BINARY
FUNCTION
QSQIFSREADUTF8
QSYS2
IFS_READ_UTF8
FUNCTION
QSQIFSWRITE
QSYS2
IFS_WRITE
PROCEDURE
QSQIFSWRITEBINARY
QSYS2
IFS_WRITE_BINARY
PROCEDURE
QSQIFSWRITEUTF8
QSYS2
IFS_WRITE_UTF8
PROCEDURE
As for the “batches”… typically, I like to “chunk” the data into one or
creating a file for each set:
In my example, I was using an SQL cursor to fetch 50 rows at a time –
were to run the process for a month (instead of a day) it may be 5-10 files.
IFSfile_0.json
IFSfile_1.json
Etc.
Most of the time, it creates a single file (json doc) but if someone
services do the same using paging).Maybe I’m off base… but I like to limit the file size (many web
Daniel Gross
Thanks,
Greg
From: RPG400-L <rpg400-l-bounces@xxxxxxxxxxxxxxxxxx> On Behalf Of
have written, that you have to output the JSON for “various batches” - soSent: Friday, February 4, 2022 3:18 PM
To: RPG programming on IBM i <rpg400-l@xxxxxxxxxxxxxxxxxx>
Subject: Re: YAJLDTAGEN with SQL into DS
Hi Greg,
AFAIK the IFS_ functions were part of TR9 on 7.3.
You can try this:
select * from qsys2.sysroutines where routine_name like 'IFS_%';
The cust_group_id was somehow a placeholder for a grouping because you
“cust_group_id” stands for your various batches - maybe a customer id, or
something you construct using a case-expression.
what your output should look like. As you didn't wrote how the "batches"
When designing a SQL solution, you should start thinking backwards from
are built, I "invented" a "cust_group_id" field, to group the data for the
batches.
(which runs 7.4). To make and say more, I would need much more information.
I only tested it with some synthetic data which I set up on pub400.com
gwilburn@xxxxxxxxxxxxxxxxxxxxxxx<mailto:gwilburn@xxxxxxxxxxxxxxxxxxxxxxx
Kind regards,
Daniel
Am 04.02.2022 um 19:49 schrieb Greg Wilburn <
have the latest PTF groups loaded on v7r3:
Daniel,
I was trying to experiment with your code below, but I have two issues.
The compiler says object IFS_WRITE in QSYS2 not found. Pretty sure I
from... I'm assuming it is an incrementing number.Second, I don't quite understand where "cust_group_id" is coming
rpg400-l-bounces@xxxxxxxxxxxxxxxxxx>> On Behalf Of Daniel Gross
Thx,
Greg
-----Original Message-----
From: RPG400-L <rpg400-l-bounces@xxxxxxxxxxxxxxxxxx<mailto:
rpg400-l@xxxxxxxxxxxxxxxxxx>>Sent: Thursday, February 3, 2022 1:18 AM
To: RPG programming on IBM i <rpg400-l@xxxxxxxxxxxxxxxxxx<mailto:
even a 2GB BLOB field if needed) and write that out using RPG - very easy.Subject: Re: YAJLDTAGEN with SQL into DS
Hi Jon,
no - I didn’t miss these parts - and both are easily solved using SQL.
As Birgitta mentions one can select the JSON string INTO a string (or
function. That way you could also write multiple batches into multiple IFS
Or you can wrap an QSYS2.IFS_WRITE(….) around the JSON_OBJECT
files using a GROUP BY clause and JOINING a table with the output file
names or deriving the output file names somehow on the result set level.
|| char(cust_group_id) || '.json',
exec sql select qsys2.ifs_write(:ifs_file_name,
json_object(
'Customers' : json_arrayagg(
json_object(
'addr_line1' : addr_line1,
'addr_line2' : addr_line2,
'city' : city,
'state' : state,
'zip' : zip,
'phone' : phone,
'email' : email
)
)
)
)
from mytable
where ….;
Or even somehow more dynamically like that:
exec sql select qsys2.ifs_write('/path/to/your/file/output_file_‘
solution is because SQL on IBM i is definitely the future. And using SQL asjson_object(
'Customers' : json_arrayagg(
json_object(
'addr_line1' : addr_line1,
'addr_line2' : addr_line2,
'city' : city,
'state' : state,
'zip' : zip,
'phone' : phone,
'email' : email
)
)
)
)
from mytable
where ….
group by cust_group_id;
But I think that this is not the point. Why I have mentioned a SQL
a READ/CHAIN replacement is not how SQL should be used.
understood the problem” - even if I wouldn’t go so far, it’s clear, that
I once read somewhere “if you have to use a SQL cursor, you haven’t
you have to lay off "procedural thinking" when using SQL, because of SQL's
“anti-procedural” and “result-descriptive” nature.
after doing SQL in RPG programs - even a simple "if sqlcode <> *zero; …“ is
Kind regards,
Daniel
P.S.: Of course Birgitta is 100% right about always checking SQLCODE
better than nothing.
<Hauser@xxxxxxxxxxxxxxx<mailto:Hauser@xxxxxxxxxxxxxxx>>:
Am 03.02.2022 um 05:42 schrieb Birgitta Hauser
IFS which your code doesn't do.plus I think he has need to use different output file names in the
Statement and write the data into the IFS.Writing to the IFS is just an SELECT ... INTO into the IFS File:
Here is the complete code that is necessary to execute the SQL
Name
DCL-S YourIFSFile SQLTYPE(CLOB_FILE);
YourIFSFile_Name = '/YourDir/YourDir2/YourIFSFile.json'; //IFS File
IFS - Replace if existsYourIFSFile_NL = %Len(%Trim(YourIFSFile_Name)); //IFS File Name Length
YourIFSFile_FO = SQFOVR; //File Operation - Write the data to the
IFS_WRITE, IFS_WRITE_UTF8, IFS_WRITE_BINARY
Exec SQL select json_object(
'Customers' : json_arrayagg(
json_object(
'addr_line1' : addr_line1,
'addr_line2' : addr_line2,
'city' : city,
'state' : state,
'zip' : zip,
'phone' : phone,
'email' : email
)
)
)
Into :YourIFSFile
from mytable;
//Check SQL Code or SQL State here ... and then continue
... and done!
... there are also SQL Stored for writing directly into the IFS:
piece of open source code.
... and for those who do not need more than this, I've even written a
(Les Brown)https://github.com/BirgittaHauser/Generate-XML-and-JSON
Mit freundlichen Grüßen / Best regards
Birgitta Hauser
"Shoot for the moon, even if you miss, you'll land among the stars."
them and keeping them!""If you think education is expensive, try ignorance." (Derek Bok)
"What is worse than training your staff and losing them? Not training
so they don't want to.“ (Richard Branson)„Train people well enough so they can leave, treat them well enough
rpg400-l-bounces@xxxxxxxxxxxxxxxxxx>> On Behalf Of Jon Paris
-----Original Message-----
From: RPG400-L <rpg400-l-bounces@xxxxxxxxxxxxxxxxxx<mailto:
rpg400-l@xxxxxxxxxxxxxxxxxx>>Sent: Donnerstag, 3. Februar 2022 00:43
To: RPG programming on IBM i <rpg400-l@xxxxxxxxxxxxxxxxxx<mailto:
points/batches Daniel - plus I think he has need to use different outputSubject: Re: YAJLDTAGEN with SQL into DS
I think you missed the bits about starting and stopping at arbitrary
file names in the IFS which your code doesn't do.
daniel@xxxxxxxx>> wrote:
Jon P.
On Feb 2, 2022, at 4:55 PM, Daniel Gross <daniel@xxxxxxxx<mailto:
statement:Hi Greg,
after fiddling around for 10 minutes or so, I came up with this SQL
better readable):
select json_object(
'Customers' : json_arrayagg(
json_object(
'addr_line1' : addr_line1,
'addr_line2' : addr_line2,
'city' : city,
'state' : state,
'zip' : zip,
'phone' : phone,
'email' : email
)
)
)
from mytable;
Which results in the following JSON string (which I formatted to be
marvinstoys@xxxxxxxxx>"
{
"Customers":[
{
"addr_line1":"64a N Williams St",
"addr_line2":"",
"city":"Crystal Lake",
"state":"IL",
"zip":"60014",
"phone":"7792204179",
"email":"marvinstoys@xxxxxxxxx<mailto:
changes to existing code - but IMHO modernization starts exactly there.},
{
"addr_line1":"267 Main St",
"addr_line2":"",
"city":"Huntington",
"state":"NY",
"zip":"11743",
"phone":"6315498743",
"email":"lsdolls@xxxxxxx<mailto:lsdolls@xxxxxxx>"
}]
}
I hope I understood your problem correctly.
Of course I understand, that it is always a "problem" to make big
gwilburn@xxxxxxxxxxxxxxxxxxxxxxx<mailto:gwilburn@xxxxxxxxxxxxxxxxxxxxxxx
HTH
Daniel
Am 02.02.2022 um 22:06 schrieb Greg Wilburn <
"INTO" a qualified data structure. But it worked perfectly.:
Chris,
Thank you for your response. I did not realize that I could SELECT
//-------------------------------------------------------------------
I set up my SQL Cursor in a procedure called OpenCursor() then:
OpenCursor();
Dow FetchCursor(AllCustomers.Customers);
WriteJson(Stmf);
jsonCount += 1;
enddo;
CloseCursor();
...in WriteJson()
DATA-GEN AllCustomers
%DATA(inStmf: 'doc=file output=clear countprefix=num_')
%GEN('YAJLDTAGEN');
//------------------------------------------------------------------------------
// Fetch the Next Record
//
rpg400-l-bounces@xxxxxxxxxxxxxxxxxx>> On Behalf Of-----------
dcl-proc FetchCursor;
dcl-pi FetchCursor ind;
rec likeds(AllCustomers.Customers) dim(50);
end-pi;
sqlRows = 0;
Exec Sql
Fetch Next From sqlcsr for :maxrows rows
Into :rec ;
if SqlStt <> sqlOK;
return *off;
endif;
Exec Sql Get Diagnostics :sqlRows = ROW_COUNT;
AllCustomers.num_Customers = sqlRows;
return *on;
end-proc;
-----Original Message-----
From: RPG400-L <rpg400-l-bounces@xxxxxxxxxxxxxxxxxx<mailto:
rpg400-l@xxxxxxxxxxxxxxxxxx>>Hiebert, Chris
Sent: Wednesday, February 2, 2022 1:19 PM
To: RPG programming on IBM i <rpg400-l@xxxxxxxxxxxxxxxxxx<mailto:
array and continue to load that with sql and then:Subject: RE: YAJLDTAGEN with SQL into DS
Try:
dcl-ds AllCustomers qualified;
num_Customers uns(5);
dcl-ds Customers qualified dim(50) inz;
addr_line1 varchar(40) inz('');
addr_line2 varchar(40) inz('');
city varchar(40) inz('');
state varchar(3) inz('');
zip varchar(10) inz('');
phone varchar(22) inz('');
email varchar(50) inz('');
end-ds;
End-ds;
AllCustomers.num_Customers = sqlRows;
DATA-GEN AllCustomers
%DATA(inStmf: 'doc=file output=clear countprefix=num_')
%GEN('YAJLDTAGEN');
If that doesn't work with the SQL you can keep the other customers
author and do not necessarily represent those of the company.
%Subarr( AllCustomers.Customers:1:SqlRows ) =
%SUBARR(customers:1:sqlRows);
Then run the DATA-GEN.
Chris Hiebert
Senior Programmer/Analyst
Disclaimer: Any views or opinions presented are solely those of the
rpg400-l-bounces@xxxxxxxxxxxxxxxxxx>> On Behalf Of
From: RPG400-L <rpg400-l-bounces@xxxxxxxxxxxxxxxxxx<mailto:
rpg400-l@xxxxxxxxxxxxxxxxxx>>Greg Wilburn
Sent: Tuesday, February 1, 2022 3:03 PM
To: RPG programming on IBM i <rpg400-l@xxxxxxxxxxxxxxxxxx<mailto:
load a Data structure.Subject: YAJLDTAGEN with SQL into DS
So I have a working application where I've used SQL data access to
SQL INTO that DS.
dcl-ds Customers qualified dim(50) inz;
addr_line1 varchar(40) inz('');
addr_line2 varchar(40) inz('');
city varchar(40) inz('');
state varchar(3) inz('');
zip varchar(10) inz('');
phone varchar(22) inz('');
email varchar(50) inz('');
end-ds;
I then use YAJLDTAGEN to create the JSON file:
DATA-GEN %SUBARR(customers:1:sqlRows)
%DATA(inStmf: 'doc=file output=clear')
%GEN('YAJLDTAGEN');
Output looks like this:
[
{
"addr_line1": "64a N Williams St",
"addr_line2": "",
"city": "Crystal Lake",
"state": "IL",
"zip": "60014",
"phone": "7792204179",
"email": "marvinstoys@xxxxxxxxx<mailto:marvinstoys@xxxxxxxxx>"
},
{
"addr_line1": "267 Main St",
"addr_line2": "",
"city": "Huntington",
"state": "NY",
"zip": "11743",
"phone": "6315498743",
"email": "lsdolls@xxxxxxx<mailto:lsdolls@xxxxxxx>"
}
]
The endpoint has asked for this:
{
"Customers": [
{
"addr_line1": "64a N Williams St",
"addr_line2": "",
"city": "Crystal Lake",
"state": "IL",
"zip": "60014",
"phone": "7792204179",
"email": "marvinstoys@xxxxxxxxx<mailto:marvinstoys@xxxxxxxxx>"
},
{
"addr_line1": "267 Main St",
"addr_line2": "",
"city": "Huntington",
"state": "NY",
"zip": "11743",
"phone": "6315498743",
"email": "lsdolls@xxxxxxx<mailto:lsdolls@xxxxxxx>"
}
]
}
I know how to create the DS... but then I'm not sure how to do Exec
RPG400-L@xxxxxxxxxxxxxxxxxx> To subscribe,
I'm pretty sure the "into variable" cannot be "json.Customers".
What am I missing?
TIA,
Greg
--
This is the RPG programming on IBM i (RPG400-L) mailing list To post
a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives at
https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
RPG400-L@xxxxxxxxxxxxxxxxxx> To subscribe,
Help support midrange.com by shopping at amazon.com with our
affiliate link: https://amazon.midrange.com
--
This is the RPG programming on IBM i (RPG400-L) mailing list To post
a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives at
https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
a--
Help support midrange.com by shopping at amazon.com with our
affiliate link: https://amazon.midrange.com
This is the RPG programming on IBM i (RPG400-L) mailing list To post
RPG400-L@xxxxxxxxxxxxxxxxxx> To subscribe, unsubscribe,message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives at
https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
affiliate
Help support midrange.com by shopping at amazon.com with our
a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:link: https://amazon.midrange.com--
This is the RPG programming on IBM i (RPG400-L) mailing list To post
RPG400-L@xxxxxxxxxxxxxxxxxx> To subscribe, unsubscribe, or change list
options,
RPG400-L-request@xxxxxxxxxxxxxxxxxx>visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
https://archive.midrange.com/rpg400-l.Before posting, please take a moment to review the archives at
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
affiliate link: https://amazon.midrange.com
Help support midrange.com by shopping at amazon.com with our
RPG400-L@xxxxxxxxxxxxxxxxxx>
--
This is the RPG programming on IBM i (RPG400-L) mailing list
To post a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>To subscribe, unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives
at https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
affiliate link: https://amazon.midrange.com
Help support midrange.com by shopping at amazon.com with our
RPG400-L@xxxxxxxxxxxxxxxxxx>--
This is the RPG programming on IBM i (RPG400-L) mailing list
To post a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>To subscribe, unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives
at https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
affiliate link: https://amazon.midrange.com
Help support midrange.com by shopping at amazon.com with our
RPG400-L@xxxxxxxxxxxxxxxxxx>--
This is the RPG programming on IBM i (RPG400-L) mailing list
To post a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>To subscribe, unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives
at https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
affiliate link: https://amazon.midrange.com
Help support midrange.com by shopping at amazon.com with our
RPG400-L@xxxxxxxxxxxxxxxxxx>--
This is the RPG programming on IBM i (RPG400-L) mailing list
To post a message email: RPG400-L@xxxxxxxxxxxxxxxxxx<mailto:
RPG400-L-request@xxxxxxxxxxxxxxxxxx>To subscribe, unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx<mailto:
support@xxxxxxxxxxxxxxxxxxxx> for any subscription related questions.Before posting, please take a moment to review the archives
at https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx<mailto:
link: https://amazon.midrange.com
Help support midrange.com by shopping at amazon.com with our affiliate
related questions.--
This is the RPG programming on IBM i (RPG400-L) mailing list
To post a message email: RPG400-L@xxxxxxxxxxxxxxxxxx
To subscribe, unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx
Before posting, please take a moment to review the archives
at https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx for any subscription
link: https://amazon.midrange.com
Help support midrange.com by shopping at amazon.com with our affiliate
--
This is the RPG programming on IBM i (RPG400-L) mailing list
To post a message email: RPG400-L@xxxxxxxxxxxxxxxxxx
To subscribe, unsubscribe, or change list options,
visit: https://lists.midrange.com/mailman/listinfo/rpg400-l
or email: RPG400-L-request@xxxxxxxxxxxxxxxxxx
Before posting, please take a moment to review the archives
at https://archive.midrange.com/rpg400-l.
Please contact support@xxxxxxxxxxxxxxxxxxxx for any subscription related
questions.
Help support midrange.com by shopping at amazon.com with our affiliate
link: https://amazon.midrange.com
As an Amazon Associate we earn from qualifying purchases.
This mailing list archive is Copyright 1997-2025 by midrange.com and David Gibbs as a compilation work. Use of the archive is restricted to research of a business or technical nature. Any other uses are prohibited. Full details are available on our policy page. If you have questions about this, please contact [javascript protected email address].
Operating expenses for this site are earned using the Amazon Associate program and Google Adsense.