go to post Dan Pasco · Feb 18 Dynamic SQL doesn't have to build the query solution every time you run the same statement. Once you initially prepare an SQL Statement, the implementation of that statement is cached and reused. The only overhead should be limited to a hopefully very brief cache resolve. There are ways that you can write a statement that makes the hashing algorithm we use very inefficient (prepare:execute ratio p:n, the larger 'n' is, the better - 1:1 being the worst). Since we now use a unified model, even embedded SQL will produce a cached implementation. Embedded reduces the cache resolve time.
go to post Dan Pasco · Feb 13 @Andre Larsen Barbosa - I am the author of the intiial dynamic SQL 'feature'. I no longer work in this area and many significant enhancements have been made since my initial work. That said, I applaud your article. It is quite nice. There is one feature that not many people know about - there is an Object mode that you can use with Dynamic Statement. Object mode is interesting when your query selects column values whose underlying type is a IRIS ObjectScript Class that is an object type (Persistent, Serial, Stream). Here is a trivial example of how it might be used. LATEST:USER>set statement = ##class(%SQL.Statement).%New() LATEST:USER>set statement.%ObjectSelectMode = 1 LATEST:USER>do statement.prepare("select name,address from person") LATEST:USER>set result = statement.execute() LATEST:USER>write result.%Next() 1 LATEST:USER>write result.name po1 LATEST:USER>write result.address 7@User.Address LATEST:USER>write result.address.city Boston LATEST:USER>write result.address.state MA
go to post Dan Pasco · Jan 15 set stream = ##class(%Stream.FileCharacter).%OpenId("/Users/.../data/continents-en.json") Did you know that you can just open a file as a stream? And for iterating - this has been in the product for a while now. This is the code for my :pp alias. ClassMethod pp(set As %AbstractSet, offset As %Integer = 0) { #define QUOTE(%val) $zu(144,1,%val) try { set isLabeled = set."_isLabeled"() if (isLabeled) { write "{" set close = "}" } else { write "[" set close = "]" } set it = set.iterator() while it.hasNext() { set next = it.next() if $isobject(next.value) { write !,?(offset+2) write:isLabeled $$$QUOTE(next.key),": " do ..pp(next.value, offset + 2) } else { write !,?(offset+2),$select(isLabeled:$$$QUOTE(next.key)_": ",1:""),$$$QUOTE(next.value) } if it.hasNext() { write "," } } write !,?offset,close } catch exc { write !,"Exception caught: ",exc.AsSQLMessage() } return } This is in %ASQ.SetUtils. My alias, pp - pretty print, is this: pp do ##class(%ASQ.SetUtils).pp($*)
go to post Dan Pasco · Nov 8, 2024 Basically the same as Enrico's: LATEST:USER>set result = $system.SQL.Execute("select * from %SYS.Namespace_List()") LATEST:USER>do result.%Display()Nsp Status Remote%SYS 1 0B360 1 0USER 1 0 3 Rows(s) Affected
go to post Dan Pasco · Sep 5, 2024 The PackageDefinition class contains only packages where we need to keep metadata about a package. Most packages do not have any extra metadata. The SQL solution is the correct solution IMO. If I were a developer on the Object team I would advocate for a method in $system.OBJ - PackageExists(<package_name>) perhaps. :)
go to post Dan Pasco · Jul 3, 2024 I was tempted to criticize the Mixtral response. I did implement the original versions of both of these items so I know them both well. The Mixtral code example for %SQL.Statement is incorrect in a few ways but I'm surprised that it was as close as it is. Once that temptation was put aside, I realized that Mixtral can only analyze what it finds. If it can't find the solution then perhaps our class documentation is not going to provide the answer to a user. Hmmm...
go to post Dan Pasco · Jul 1, 2024 %SQL.Statement is part of the IRIS implementation of Dynamic SQL and it is based on the SQL Standard's Call Level Interface (CLI). It allows the user to prepare and execute any SQL statement, including DDL. The result of executing a dynamic statement is an instance of %SQL.StatementResult. That result contains at least %SQLCODE and %Message, indicating success/failure along with some minimal information about the failure. Part of Dynamic SQL is also a result set interface - %SQL.IResultSet. There is at least one extention of %SQL.IResultSet, %SQL.ISelectResult. When a statement result is a result set or includes one or more result sets then those result sets are likely instances of %SQL.IResultSet. The other result set classes (some deprecated) are not necessarily instances of %SQL.IResultSet but do implement the most common members of that interface. There is another class, %SQL.CustomQuery, that can be used to implement custom queries that are more intuitive than class queries. The documentation for %SQL.CustomQuery contains information on how to implement your own custom query along with an example.
go to post Dan Pasco · Jul 1, 2024 Actually, the problem is that %Exception.AbstractException already has five parameters. The error message you received is because the fifth argument has a type of %Exception.AbstractException and your fifth argument has a type of %String. It is entirely possible to override the implementation of a member inherited from a super class so long as the inherited member is not defined as final and the signature of the local override is compatible with the inherited signature. The class compiler does perform a signature check to ensure the class does not violate the prime directive: Every instance of a class is implicitly a valid instance of its primary super class. Rewording that slightly - the interface of a class must be compatible with the interface of its primary super class. You can add new parameters to an inherited method but the existing parameters, those inherited, must declare a compatible type. A compatible type can be more specific (a subclass of the inherited type) but it cannot be a competely different type. I tried adding a sixth parameter and also corrected the type of the fifth: Class User.MyException Extends %Exception.AbstractException { Method %OnNew(pName As %String = "", pCode As %String = "", pLocation As %String = "", pData As %String = "", pInnerException As %Exception.AbstractException = {$$$NULLOREF}, myArg As %String) As %Status [ Private ] { } } This class compiled cleanly. ObjectScript does not allow method overloading which is a different thing.
go to post Dan Pasco · Jun 27, 2024 I like @Norman W. Freeman 's answer. That is the factory pattern and I use it regularly. %OnNew is a single method that is implemented in each class. There is a little known feature of %OnNew - it can return an oref whose type class is different from the class that implements %OnNew. In other words, %OnNew can be that factory method. Norman's Create() solution is cleaner IMO. The sole advantage (again, IMO) of a polymorphic %OnNew() is that it works with %New. I can provide a demo is anyone is interested.
go to post Dan Pasco · Jun 17, 2024 I am a huge fan of try/catch. The try is virtually cost-free with overhead only encountered when an exception is caught. I do agree with Jon Willeke - when I am in older code I normally do not refactor $ZTRAP unless the changes I have to make are more than a simple edit. I also use RETURN when exiting a function/method instead of QUIT. It is possible to simluate a finally but it isn't as nice as having a real finally block would be.
go to post Dan Pasco · Jun 17, 2024 And the first two lines can be combined: set str = ##class(%Stream.FileCharacter).%OpenId("c:\myfolder\file.csv")
go to post Dan Pasco · May 28, 2024 %KillExtent should not be used in production except in very controlled circumstances. It was meant for use in a development environment. It's only advantage is that it is fast.
go to post Dan Pasco · May 27, 2024 The simplest answer is that %KillExtent performs a physical delete of the extent (data and indexes) while %DeleteExtent performs a logical delete. The physical delete does not enforce any constraints and does not invoke referential actions. The logical delete does enforce all constraints and performs referential actions.
go to post Dan Pasco · Mar 20, 2024 Thank you for that information. I researched this and was told that we support Java 8+, not just 8 and 11. We submitted a request to correct the documentation.
go to post Dan Pasco · Mar 19, 2024 You should be able to use Java 17 now for your projects. Are you specifically asking for when we will support Java 17 (and perhaps later versions) when using External Java Server/Java Gateway? If that is the case then I believe you can do that in 2024.1.0.
go to post Dan Pasco · Feb 29, 2024 I can provide a solution for this but it comes with warnings - SQL doesn't do well with polymorphic embedded objects. If that isn't an issue then consider using a factory pattern. You must define a base class that extends %SerialObject and it must be instantiable. Then, override %OnNew() to dispatch to the concrete class that you wish to instantiate. Perhaps pass in a type argument. That %OnNew() can return an OREF (or a %Status if an error occurs). If you are interested in this model then I can provide a small demo. -Dan
go to post Dan Pasco · Feb 23, 2024 Yes, of course. JSON_TABLE in the FROM clause is just another table and can be used as such. If you have wish to use indexes on JSON values stored in columns then you can always define indexes on computed values. That has been possible even before JSON_TABLE. Keep in mind that dynamic data does not always follow expectations as a field in a dynamic object can be a literal, another object or an array of values. I previously posted about ASQ. You can use ASQ to derive a computed value in your class definition.
go to post Dan Pasco · Feb 23, 2024 The ISO Standard syntax for JSON_TABLE includes a FORMAT keyword. We are working on a binary compressed format for dynamic arrays and object but it isn't available yet. In addition to the binary format, we hope support other formats in the future. The default FORMAT, as defined by the Standard, is JSON.
go to post Dan Pasco · Jan 24, 2024 Perhaps this will help: https://docs.intersystems.com/irislatest/csp/docbook/platforms/DocBook.U...