go to post Dan Pasco · Jan 12, 2022 Interesting! I didn't see which version of InterSystems IRIS you are using but you might try a couple of things: First, don't return a status value from your method. Instead return a %Stream.GlobalBinary instance. If you still want a status, get it some other way. I recommend just throwing an exception - IRIS Native should handle that okay. The object value you get back will be a proxy object that should allow you to read the stream. In our documentation you might search for "reverse proxy objects". If get some time, I will try to build a sample of doing this.
go to post Dan Pasco · Jan 4, 2022 You can. Two different ways but both utilize the External Java Server with gateway connections. That is the same mechanism employed by LOAD DATA. If you have an example of what you want to do then I can provide you with a demo - using both options.
go to post Dan Pasco · Jan 4, 2022 Not quite - I just defined that alias this morning to test something out. I think that a pre-defined (shipped with Iris) sql alias should launch the shell - I didn't know anything about pre-installed/shipped with Iris aliases.
go to post Dan Pasco · Jan 4, 2022 That's because I defined that alias myself - on Linux it is the .iris_init file in my home directory. I don't know where that is on Windows...
go to post Dan Pasco · Jan 4, 2022 I like :sql as a way to launch the shell - so that means this alias should be renamed to something else!! USER>:sql "select top 2 * from person" do $system.SQL.Execute("select top 2 * from person").%Display() id name ssn dob home_street home_city home_state home_zip 1 Willeke,Thelma K. 934-46-2099 18948 308 Madison Blvd Bensonhurst HI 71501 2 Nathanson,Will P. 771-91-1008 37359 9664 Second Place Jackson VA 15837 2 Rows(s) Affected
go to post Dan Pasco · Dec 24, 2021 Use the IRISList class: IRIS iris = IRIS.createIRIS(connection); IRISList list = new IRISList(); list.add("this is a string"); list.add(100); iris.set(list, "test",1); USER>zw ^test^test(1)=$lb("this is a string",100)
go to post Dan Pasco · Dec 22, 2021 Hi Marcio, I am a developer at InterSystems and I work with Java, JSON, and SQL every day. Perhaps I can help. Can you provide an example of the Java code where you receive the JSON output from IRIS? I think I have a couple of interesting options for you. -Dan
go to post Dan Pasco · Nov 26, 2021 I just saw this on the community: https://community.intersystems.com/post/intersystems-objectscript-101-en. Perhaps it will help?
go to post Dan Pasco · Jul 8, 2021 I suspect that is the case but I cannot verify it without seeing more of your code. Someone from our HealthShare team probably knows this but I don't work on that team. Sorry.
go to post Dan Pasco · Jul 8, 2021 I didn't count the number of question marks but I assume you've verified that it matches the number of columns. Also, there are limitations on the number of command line arguments you can pass but that number is rather high. I don't recall exactly what it is but it is certainly limited. At one time we established 16 as the maximum number of arguments. As I recall, we did nothing to impose an arbitrary limit but ObjectScript itself does have a limit. My next questions are meant to establish where the error is reported. You can prepare a dynamic SQL statement and then execute it separately and multiple times. Is the error reported during prepare or during execute? PS: (edited): Sorry, I see now your code. You are using a HS Adapter to execute this statement. That leads me to a different question - can you prepare and execute this statement as a Dynamic SQL Statement? set statement = ##class(%SQL.Statement).%New() set status = statement.%Prepare(.tSQL) If status is okay (1) then execute it - set result = statement.%Execute( <your parameter values go here>) Then check result.%SQLCODE. In the meantime, I will take a look at the Adapter (not my area but I can still look) to see if there is some limitation.
go to post Dan Pasco · Jul 6, 2021 I agree with Herman - to a point. Leave the IDKEY index alone and just accept the system assigned idkey. In most cases this is the best policy. But - there are potentially several "candidate keys", all can be defined - or not, but one can be chosen to be the "primary key". In that case, define the key not as unique but as "primarykey". Some tooling works better with a defined primary key. It does not interfere with the IDKEY unless it is also defined as the "idkey".
go to post Dan Pasco · Jul 6, 2021 Hi Jonathan, I am very interested in this problem (I wrote the early version of dynamic statement parameter binding). My first question is simple - can you execute this statement using embedded SQL? If you can then we need to dig further to discover the correct error (some errors get masked as syntax when there is a different problem). Thanks, Dan PS: Of course, you will have to supply values for ? to test compile this as an embedded statement (:hv1, :hv2... can work) - or - can you try this using an SQL utility or as a ODBC/JDBC statement?
go to post Dan Pasco · Apr 28, 2021 set java = $system.external.getJavaGateway() do java.addToPath("/home/myhome/mariadb-java-client-2.7.2.jar") set mariads = java.new("org.mariadb.jdbc.MariaDbDataSource","jdbc:mariadb://myhost:3306/SAMPLES") set mariaconn = mariads.getConnection("myuser","secret") set pstmt = mariaconn.prepareStatement("select * from person limit 5") set jdbcresult = pstmt.executeQuery() set jdbcmetadata = jdbcresult.getMetaData() for i=1:1:jdbcmetadata.getColumnCount() { write jdbcmetadata.getColumnName(i),$char(9) } while jdbcresult.next() { write !,$increment(rowcnt),$char(9) for i=1:1:jdbcmetadata.getColumnCount() { write jdbcresult.getString(i),$c(9) } } write !!!,"OR I COULD JUST USE MY ALREADY IMPLEMENTED RESULT SET RENDERER FROM ANOTHER JAVA PROJECT",!!! do java.addToPath(..#EXTERNALLIBPATH) set ers = java.new("external.test.ExternalResult",pstmt.executeQuery()) do ers.renderTable() do mariaconn.close() do java.disconnect() And the results: USER>do ##class(external.test.MariaJDBC).communityDemo() name ssn dob home_street home_city home_state home_zip 1 Basile,Molly M. 452-57-8033 1994-06-02 1153 First Street Xavier SD 98033 2 Cooke,Howard F. 131-62-3894 2017-12-09 5172 Washington Place Zanesville NE 44980 3 Donaldson,Phil R. 480-79-5019 1990-01-23 4429 Elm Street Miami WA 67638 4 Eisenstien,Michael D. 655-11-6334 1948-08-14 4676 Elm Avenue Reston KY 52729 5 Faust,Mo E. 772-42-3921 2018-01-10 826 Maple Avenue Youngstown OH 37180 OR I COULD JUST USE MY ALREADY IMPLEMENTED RESULT SET RENDERER FROM ANOTHER JAVA PROJECT ┌─────────────────────┬───────────┬──────────┬─────────────────────┬──────────┬──────────┬────────┐ │name │ssn │dob │home_street │home_city │home_state│home_zip│ ├─────────────────────┼───────────┼──────────┼─────────────────────┼──────────┼──────────┼────────┤ │Basile,Molly M. │452-57-8033│1994-06-02│1153 First Street │Xavier │SD │98033 │ ├─────────────────────┼───────────┼──────────┼─────────────────────┼──────────┼──────────┼────────┤ │Cooke,Howard F. │131-62-3894│2017-12-09│5172 Washington Place│Zanesville│NE │44980 │ ├─────────────────────┼───────────┼──────────┼─────────────────────┼──────────┼──────────┼────────┤ │Donaldson,Phil R. │480-79-5019│1990-01-23│4429 Elm Street │Miami │WA │67638 │ ├─────────────────────┼───────────┼──────────┼─────────────────────┼──────────┼──────────┼────────┤ │Eisenstien,Michael D.│655-11-6334│1948-08-14│4676 Elm Avenue │Reston │KY │52729 │ ├─────────────────────┼───────────┼──────────┼─────────────────────┼──────────┼──────────┼────────┤ │Faust,Mo E. │772-42-3921│2018-01-10│826 Maple Avenue │Youngstown│OH │37180 │ └─────────────────────┴───────────┴──────────┴─────────────────────┴──────────┴──────────┴────────┘
go to post Dan Pasco · Apr 28, 2021 Exactly - you haven't used the Gateways before. Why? IMO, the Gateways (mostly Java and DotNet) were a bit cumbersome to deal with. With ELS, we have default servers that can be easily managed and discovered, they require little or no configuration to get started (some ELS's do require some configuration if the language platform support is not discoverable by IRIS), and the interface is simple and direct. One quick example, using Python, is to use the os module to get the current working directory: USER>set python = $system.external.getPythonGateway() USER>write python.invoke("os","getcwd") /opt/intersystems/iris/xdbc/mgr All that is required is a gateway connection to an external server, your code needs to be visible to that external server, either by direct placement into the default path for that language platform or by explicitly adding it by calling addToPath(), and public interfaces in your external code. By "external", I mean code that isn't written inside of the IRIS Server - ObjectScript. When your external code writes to the "system output" device, that output is redirected to the IRIS current device. In my above example, the renderTable() function simply constructs a formatted string using the AsciiTable library (got it from GitHub) and writes it using System.out.println(formattedstring). I simply copied the output that was displayed in my IRIS session terminal window and pasted it in the original post. No extra work involved. If your external code returns an object then you can indeed make use of that object as if it were a local IRIS Object - because it is. It is actually a network proxy object that communicates with the original external language object. That communication can actually be full duplex - your external code can talk to IRIS and IRIS can talk to your external code. One really simple example that I think is quite profound is using JDBC in IRIS. This isn't JDBC Gateway - that still exists and it is what it is. I am talking about using a Java JAR file that contains a JDBC driver and you want to use it. I'll try a simple demo here. I have MariaDB installed on my system so I'll just use it to query a table I have defined there. First - using mariaDB from the command line interface: ~ % mysql -u myusername -p Enter password: Welcome to the MariaDB monitor. Commands end with ; or \g. Your MariaDB connection id is 4 Server version: 10.5.8-MariaDB Homebrew Copyright (c) 2000, 2018, Oracle, MariaDB Corporation Ab and others. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement. MariaDB [(none)]> show databases -> ; +--------------------+ | Database | +--------------------+ | SAMPLES | | information_schema | | mysql | | performance_schema | | test | +--------------------+ 5 rows in set (0.010 sec) MariaDB [(none)]> use SAMPLES Reading table information for completion of table and column names You can turn off this feature to get a quicker startup with -A Database changed MariaDB [SAMPLES]> show tables; +-------------------+ | Tables_in_SAMPLES | +-------------------+ | person | +-------------------+ 1 row in set (0.000 sec) MariaDB [SAMPLES]> select * from person limit 5 -> ; +-----------------------+-------------+------------+-----------------------+------------+------------+----------+ | name | ssn | dob | home_street | home_city | home_state | home_zip | +-----------------------+-------------+------------+-----------------------+------------+------------+----------+ | Basile,Molly M. | 452-57-8033 | 1994-06-02 | 1153 First Street | Xavier | SD | 98033 | | Cooke,Howard F. | 131-62-3894 | 2017-12-09 | 5172 Washington Place | Zanesville | NE | 44980 | | Donaldson,Phil R. | 480-79-5019 | 1990-01-23 | 4429 Elm Street | Miami | WA | 67638 | | Eisenstien,Michael D. | 655-11-6334 | 1948-08-14 | 4676 Elm Avenue | Reston | KY | 52729 | | Faust,Mo E. | 772-42-3921 | 2018-01-10 | 826 Maple Avenue | Youngstown | OH | 37180 | +-----------------------+-------------+------------+-----------------------+------------+------------+----------+ 5 rows in set (0.016 sec) MariaDB [SAMPLES]> Next, I'll follow up with an example of doing this from within IRIS.
go to post Dan Pasco · Apr 28, 2021 You are right, of course. I was puzzled by the lack of documentation myself. I did some research and I've been told that there will be documentation available for the External Language Servers and $system.external. The $system.external.Help() feature is something that has been part of the product for many years. It is not a substitute for proper documentation but serves to aid the command line user when questions regarding function names and arguments need a quick answer. The ELS features are based on what is known as "Dynamic Object Gateways". There is extensive documentation available for them. That documentation is a bit dated in that it doesn't incorporate the new $system.external API's but still mostly accurate. Obtaining a gateway connection to an External Language Server by simply invoking $system.external.getGateway(<gatewayname>) is just one way we've simplified that interface. There are default ELS's defined for Java, Python, DotNet, and R. Each is supported not only by the getGateway() that accepts the name of the ELS but also by get<Language>Gateway() functions for Java, DotNet, Python and R that return gateway connections to the default language servers.
go to post Dan Pasco · Apr 24, 2021 In the 2021.1 preview, you can execute $system.external.Help() to get a description of the External Language Server interface.
go to post Dan Pasco · Mar 31, 2021 How are you purging this data? SQL? Or by calling a class method such as %DeleteExtent?
go to post Dan Pasco · Mar 8, 2021 Just to clarify - inheritance considers the primary super class hierarchy. %Persistent must be the first class some where in the primary super class hierarchy. Sample.Employee extends Sample.Person which extends %Library.Persistent.
go to post Dan Pasco · Mar 8, 2021 If there is no storage definition then a storage definition whose type is %Storage.Persistent is created when compiling the class. This is done before the storage definition is compiled (every class member is compiled by the class compiler). Then, when the storage definition is compiled and if the type is %Storage.Persistent then the %Storage.Persistent.STORAGECOMPILERCLASS will generate a full storage definition for the current class definition. What does that mean? Well - if this is the first time this storage definition has been compiled then it is most likely empty, other than the type class. The STORAGECOMPILERCLASS will generate a complete storage definition. But even if the storage definition is already defined, the STORAGECOMPILERCLASS still reviews that definition. If there are any changes detected to the class definition then the storage definition is updated to reflect those changes. That means that the user can manually edit the storage definition at any time and the storage compiler will simply make sure that the definition is "complete" - no missing properties, no missing indexes, and so on. Bottom line is that the user is free to make any desired changes, including deleting the storage definition completely. Keep in mind that some changes can make the compiled class incompatible with existing data.
go to post Dan Pasco · Nov 25, 2020 Class utility.StatementColumns Extends %SQL.CustomQuery { Parameter SQLNAME As String = "statement_columns"; Property columns As %Collection.ListOfObj [ Internal, Private ]; Property columnPtr As %String [ Internal, Private ]; Property atEnd As %Boolean [ Internal, Private ]; Property columnType As %String; Property colName As %String(MAXLEN = 255); Property ODBCType As %Integer; Property precision As %Integer; Property scale As %Integer; Property isNullable As %Boolean; Property label As %String(MAXLEN = 255); Property tableName As %String; Property schemaName As %String; Property qualifier As %String; Property isAutoIncrement As %Boolean; Property isCaseSensitive As %Boolean; Property isCurrency As %Boolean; Property isReadOnly As %Boolean; Property isRowVersion As %Boolean; Property isUnique As %Boolean; Property isAliased As %Boolean; Property isExpression As %Boolean; Property isHidden As %Boolean; Property isIdentity As %Boolean; Property isKeyColumn As %Boolean; Property isRowId As %Boolean; Property isList As %Boolean; Property property As %Dictionary.CompiledProperty; /// The objects type class Property typeClass As %Dictionary.CompiledClass; Property clientType As %Integer; Method %OpenCursor(statement As %String) [ Private ] { set ..columns = $system.SQL.Prepare(statement).%Metadata.columns set ..atEnd = 0 set ..columnPtr = "" } Method %FetchCursor() As %Library.Integer { set response = 0 if '..atEnd { set next = ..columnPtr set column = ..columns.GetNext(.next) if next '= "" { set response = 1 set ..columnPtr = next do ..mapColumnToRow(column) } else { set ..atEnd = 1 set ..columnPtr = "" do ..clearRow() } } return response } Method mapColumnToRow(column As %SQL.StatementColumn) { set ..columnType = "SQLRESULTCOL" set ..colName = column.colName set ..ODBCType = column.ODBCType set ..precision = column.precision set ..scale = column.scale set ..isNullable = column.isNullable set ..label = column.label set ..tableName = column.tableName set ..schemaName = column.schemaName set ..qualifier = column.qualifier set ..isAutoIncrement = column.isAutoIncrement set ..isCaseSensitive = column.isCaseSensitive set ..isCurrency = column.isCurrency set ..isReadOnly = column.isReadOnly set ..isRowVersion = column.isRowVersion set ..isUnique = column.isUnique set ..isAliased = column.isAliased set ..isExpression = column.isExpression set ..isHidden = column.isHidden set ..isIdentity = column.isIdentity set ..isKeyColumn = column.isKeyColumn set ..isRowId = column.isRowId set ..isList = column.isList } Method clearRow() { set ..columnType = "" set ..colName = "" set ..ODBCType = "" set ..precision = "" set ..scale = "" set ..isNullable = "" set ..label = "" set ..tableName = "" set ..schemaName = "" set ..qualifier = "" set ..isAutoIncrement = "" set ..isCaseSensitive = "" set ..isCurrency = "" set ..isReadOnly = "" set ..isRowVersion = "" set ..isUnique = "" set ..isAliased = "" set ..isExpression = "" set ..isHidden = "" set ..isIdentity = "" set ..isKeyColumn = "" set ..isRowId = "" set ..isList = "" } }