go to post David Hockenbroch · Mar 5 It's useful in terminal, and in my opinion, it lets me write neater code too. I don't have to include lines like if $$$ISERR(sc) { $$$ThrowStatus(sc) } every time.
go to post David Hockenbroch · Mar 5 Yes, that's how it works. If you don't want it to reset the sc, then you can leave out the R flag when you call the New() method. Just be aware that the error handling only triggers when sc goes from OK to an error, so it won't trigger again unless you manually set sc back to $$$OK.
go to post David Hockenbroch · Mar 4 @Evgeny Shvarov, I think I have submitted it to Open Exchange correctly. This is my first time using IPM, GitHub AND Open Exchange, so bear with me if I messed any of it up!
go to post David Hockenbroch · Mar 4 I will gladly if I can figure out how to do that. Documentation I'm finding for IPM is all referring to ZPM and from 2019 or so.
go to post David Hockenbroch · Mar 3 I wrote my own class that does some fun, magical stuff. Here's the class: /// This class can be used to make a status code do some basic handling on its own when it becomes an error status. /// For example:<br /><br /> /// set status = ##class(DH.WatchedStatus).New("RW")<br /> /// set status.sc = (some method that returns a status here)<br /><br /> /// If the method returns an error status, it will immediately be written, and the status will change back to $$$OK.<br /> /// Note that if ..sc is not set back to an $$$OK status, either automatically or manually, error handling will not trigger again on the next error. Class DH.WatchedStatus Extends %RegisteredObject { /// This is the status code to be watched. Property sc As %Status [ InitialExpression = 1, SqlComputeCode = {set {*} = ##class(DH.WatchedStatus).Reset({sc},{resetSelf})}, SqlComputed, SqlComputeOnChange = isErr ]; /// Used to track if the status code is an error. This is necessary for some shenanigans with SQLComputeCode between this flag and the status code. Property isErr As %Boolean [ InitialExpression = 0, SqlComputeCode = {set {*} = ##class(DH.WatchedStatus).ErrCheck({sc},{writeSelf},{logSelf},{throwSelf})}, SqlComputed, SqlComputeOnChange = sc ]; /// If true, this will will throw ..sc as soon as it becomes an error. Property throwSelf As %Boolean [ InitialExpression = 0 ]; /// If true, this will log ..sc as an Exception as soon as it becomes an error. Property logSelf As %Boolean [ InitialExpression = 0 ]; /// If true, this will write the error text of ..sc as soon as it becomes an error. Property writeSelf As %Boolean [ InitialExpression = 0 ]; /// If true, after other error handling, ..sc will be reset to $$$OK. /// Note that if this is false, you will need to reset the status yourself for the automatic handling to trigger again. Property resetSelf As %Boolean [ InitialExpression = 0 ]; /// Handles status according to flags set, then sets isErr. ClassMethod ErrCheck(sc, writeSelf, logSelf, throwSelf) As %Boolean [ Internal ] { if $$$ISERR(sc){ if writeSelf{ write $SYSTEM.Status.GetErrorText(sc) } if logSelf = 1{ do ##class(%Exception.StatusException).CreateFromStatus(sc).Log() } if throwSelf = 1{ $$$ThrowStatus(sc) } quit 1 } else{ quit 0 } } /// If resetSelf is true, resets the status code after error handling occurs. ClassMethod Reset(sc, resetSelf) As %Status [ Internal ] { return:resetSelf $$$OK return sc } /// flags is a string which determines status behavior when an error occurs /// T = throw the status /// L = log the status as an exception /// W = write the status error text /// R = reset status after error handling; if set, isErr goes back to 0 and sc goes back to 1 ClassMethod New(flags As %String) As DH.WatchedStatus { set status = ##class(DH.WatchedStatus).%New() set flags = $ZCVT(flags,"U") set:(flags [ "T") status.throwSelf = 1 set:(flags [ "L") status.logSelf = 1 set:(flags [ "W") status.writeSelf = 1 set:(flags [ "R") status.resetSelf = 1 return status } } Here's how I can use it: set mystatus = ##class(WoodWare.Status).New("WR") set mystatus.sc = (something that returns a status) Having done that, since I used the write and reset flags in the constructor, it will write out the error text and then reset the status object back to $$$OK. I can use this to save myself from constantly having to check whether the status is an error and log or throw it in other code outside the terminal too. It essentially watches itself. Update: this is now available on IPM. install watchedstatus
go to post David Hockenbroch · Feb 21 If it's a simple get request, you can test from the terminal without setting up a %Net.HttpRequest. I usually do it like this, in a terminal (for more complicated ones with a request body, you have to initialize %request to a %CSP.Request and set it up too): set %response = ##class(%CSP.Response).%New() do ##class(Your.Class).YourMethod(yourarguments) zw %response
go to post David Hockenbroch · Feb 17 I'm not sure what Matheus' specific issue was, but I had to do this when using SAP's Crystal Reports Java API because when shared memory was enabled, it tended to leave connections hanging open when it shouldn't, and eventually the number of connections would exceed the maximum. With shared memory disabled, it worked as expected.
go to post David Hockenbroch · Feb 14 One way to do that is to use your server's LAN IP in the JDBC URL instead of using 127.0.0.1 or localhost, if that suits your needs.
go to post David Hockenbroch · Feb 6 How did you do the SSLDefs.ini file in the client, and where did you put it?
go to post David Hockenbroch · Jan 22 I can answer one easy part of this. The ODBC.ini is in the mgr directory of the IRIS installation and is named irisodbc.ini. Once you create your ODBC definitions there, you should be able to log into your management portal and see them in the drop-down when you create an ODBC gateway connection.
go to post David Hockenbroch · Jan 16 In your system management portal, you'd have to have your /gmmhhcdev/csp/healthshare/gmmhtie/ web application set to be usable unauthenticated, but are you sure you want to do that?
go to post David Hockenbroch · Jan 15 This is a case where you will need to use Dynamic SQL, not Embedded SQL.
go to post David Hockenbroch · Jan 8 I'm probably going to take this approach going forward, but for the moment the amount of rewriting I would have to do to switch everything over to that subclass is a bit more than I want to take on.
go to post David Hockenbroch · Jan 8 That was the issue. The IRISLIB database was mounted as read-only, so once I gave myself the right permissions and fixed that, I was able to edit it.
go to post David Hockenbroch · Nov 22, 2024 Thanks for pointing that out. I changed just the number when writing the article so I wasn't using real data, and I missed changing it in the first instance!
go to post David Hockenbroch · Nov 20, 2024 You'll use both %Library.DynamicObject and %Library.DynamicArray classes for this. When you do JSON.%Get("Practice") the resulting object is a dynamic array, not a dynamic object, so it works a little differently. Then when you get the first element of the array, you have another dynamic object. Try: //Get the array Practice set array = JSON.%Get("Practice") //Get item 0 from the array we just got set nodejson = array.%Get(0) //Get the value of Node set node = nodejson.%Get("Node") //Write the value out (or do whatever else you need) write node
go to post David Hockenbroch · Nov 20, 2024 Is there a write command happening somewhere in your method before you try to set the header? If so, the headers will be sent before the write command, and will mean that it's too late to modify them after that. You can check %response.InProgress to see if that has already happened.
go to post David Hockenbroch · Nov 20, 2024 @Sammy Lee @Marc Mundt I'm able to replicate some of this behavior if I use the developer tools in Edge, go to the network tab, and use Network Request Blocking to block *algolia*. The issue might be with the XHR requests there. @Scott Roth In Edge, if you bring up the developer tools, go to the network tab, and in the filter buttons, click on XHR, then start typing in the search box, do you see errors there? (GIF shows what some good requests look like, followed by what errors might look like.)
go to post David Hockenbroch · Nov 20, 2024 For what it's worth, I get the error about the MutationObserver too, but my results still display correctly. (Is Magneto messing with Cerebro again over there?)
go to post David Hockenbroch · Nov 18, 2024 The documentation for the Read method says "Some stream classes use this to optimize the amount of data returned to align this with the underlying storage of the stream." I take this to mean that for a file stream, it might be trying to read in a way that aligns with how the drive is formatted. Can you run the command below and see if the Bytes Per Cluster is 1024? C:\Windows\System32>fsutil fsinfo ntfsInfo C: NTFS Volume Serial Number : 0x0060ba3960ba356e NTFS Version : 3.1 LFS Version : 2.0 Total Sectors : 997,918,719 (475.8 GB) Total Clusters : 124,739,839 (475.8 GB) Free Clusters : 96,063,960 (366.5 GB) Total Reserved Clusters : 893,290 ( 3.4 GB) Reserved For Storage Reserve : 884,043 ( 3.4 GB) Bytes Per Sector : 512 Bytes Per Physical Sector : 512 Bytes Per Cluster : 4096 (4 KB) Bytes Per FileRecord Segment : 1024 Clusters Per FileRecord Segment : 0 Mft Valid Data Length : 512.50 MB Mft Start Lcn : 0x00000000000c0000 Mft2 Start Lcn : 0x0000000000000002 Mft Zone Start : 0x0000000001d59580 Mft Zone End : 0x0000000001d65da0 MFT Zone Size : 200.13 MB Max Device Trim Extent Count : 256 Max Device Trim Byte Count : 0xffffffff Max Volume Trim Extent Count : 62 Max Volume Trim Byte Count : 0x40000000 Resource Manager Identifier : BBA1AD65-C5EC-11EE-8ED5-D0AD0854D65E