/
gmniMigration -- Deprecated
gmniMigration -- Deprecated
Mark Juras
Owned by Mark Juras
Last updated: Apr 28, 2020
The Collections upgrade described below is a deprecated implementation. A similar feature is implemented using the GenericCollection.dll
Using Standard Migrations
No migration can expect to use all or even most of the original COM components that were used by VB6, and no serious migration can be considered complete before all COM components have been replaced by native .NET code. Finally, since these migrations must be upgraded as they are performed, no two migration teams are likely to agree on the best upgrade path to be taken. The default migrations performed by gmBasic are generic and designed to be compilable even in situations where the target code is not fully mature. Their intent is not to produce final functionally equivalent .NET code, but rather to provide a structured, well-formed starting point.<LoadRuntime dllName="Migration.dll" />
The Collections Command
The Collections command supplies the specifications needed to transform VBA.Collection and Scripting.Dictionary into strong collection classes. The two classes have separate specification subcommands.The StrongCollections Subcommand
The goal of the StrongCollections subcommand is to transform references to the weakly typed VB6.Collection class into the strongly typed .NET List<T> or .NET Dictionary<Key,TValue> class. This transformation is implemented within the FinishAnalyser event. It begins with symbol table scan and then requires a series of code scans. Simply stated the command takes original VB6 code likePrivate col1 As VBA.Collection Private col2 As VBA.Collection Private col3 As VBA.Collection ... Private Sub AddItem(ByVal i_Item As String) col1.Add(i_Item) col2.Add(i_Item, i_Item) End Sub Private Sub AddItemWithKey(ByVal i_Key As String, ByVal i_Item As String) col3.Add(i_Item,i_Key) End Sub Private Sub Class_Initialize() col1 = New Collection col2 = New Collection col3 = New Collection End Sub
private List<string> col1; private List<string> col2; private Dictionary <string, string> col3; ... private void AddItem(string i_Item) { col1.Add(i_Item); col2.Add(i_Item); } private void AddItemWithKey(string i_Key, string i_Item) { col3.Add(i_Key, i_Item); } public Class1() { col1 = new List<string>(); col2 = new List<string>(); col3 = new Dictionary<string, string>(); }
<StrongCollections> <ItemOnly> ... </ItemOnly> <KeyItem> .. </KeyItem> </StrongCollections>
<subcode id="ForEach"> <vbn role="command" narg="3" code="For Each %1d As %3d In %2d\p"/> <csh role="command" narg="3" code="foreach(%3d %1d in %2d)\n{\p"/> </subcode>
<ItemOnly> <NetName><![CDATA[List<$1d>
<KeyItem> <NetName><![CDATA[Dictionary<$1d,$2d>
The StrongDictionaries Subcommand
The goal of the StrongDictionaries subcommand is to transform references to the weakly typed Scripting.Dictionary class into the strongly typed .NET HashSet<T> or .NET Dictionary<Key,TValue> class. This transformation is implemented within the FinishAnalyser event. It begins with symbol table scan and then requires a series of code scans. The Scripting.Dictionary is described in the interface description file scrrun.dll.xml This transformation assumes that the translation script used a simplified version of the this file that removed the unneeded ByRef parameters from the methods. This allowed the tool to generate simpler code patterns. In particular the class interface IDictionary must be described as shown here.<class id="IDictionary" parent="IDispatch" default="Item"> <property id="Count" type="Integer" status="Out"/> <property id="CompareMode" type="CompareMethod" status="InOut"/> <accessor id="Item" type="Variant"> <argument id="Key" type="Variant" status="ByVal"/> </accessor> <method id="Add" type="Void"> <argument id="Key" type="Variant" status="ByVal"/> <argument id="Item" type="Variant" status="ByVal"/> </method> <method id="Exists" type="Boolean"> <argument id="Key" type="Variant" status="ByVal"/> </method> <method id="Items" type="Variant"/> <method id="Key" type="Void"> <argument id="Key" type="Variant" status="ByVal"/> </method> <method id="Keys" type="Variant"/> <method id="Remove" type="Void"> <argument id="Key" type="Variant" status="ByVal"/> </method> <method id="RemoveAll" type="Void"/> <method id="_NewEnum" type="Object"/> <accessor id="HashVal" type="Variant"> <argument id="Key" type="Variant" status="ByVal"/> </accessor> </class>
Private dic1 As Scripting.Dictionary Private dic2 As Scripting.Dictionary ... Private Sub AddItem(ByVal i_Item As String) dic1.Add(i_Item,i_Item) End Sub Private Sub AddItemWithKey(ByVal i_Key As String, ByVal i_Item As String) dic2.Add(i_Key,i_Item) End Sub Private Sub Class_Initialize() dic1 = New Dictionary dic2 = New Dictionary End Sub
private HashSet<string> dic1; private Dictionary <string, string> dic2; ... private void AddItem(string i_Item) { dic1.Add(i_Item); } private void AddItemWithKey(string i_Key, string i_Item) { dic2.Add(i_Key, i_Item); } public Class1() { dic1 = new HashSet<string>(); col2 = new Dictionary<string, string>(); }
<StrongDictionaries> <ItemOnly> ... </ItemOnly> <KeyItem> .. </KeyItem> </StrongDictionaries>
<ItemOnly> <NetName><![CDATA[HashSet<$1d>
<KeyItem> <NetName><![CDATA[Dictionary<$1d,$2d>
LDA,LDA,TYP,IFS.ForEach LDA,LDA,LLP,MEM.Child,CUF.Args0,REF,TYP,COL,CNV.CastType,LIC,TYP,IFS.ForVariant
usekeys: LDA.%dictionary%,LLP.Scripting.IDictionary.Keys,MEM.Child,CUF.Args0,argument,ANA.Indexer passKeys: LDA.%dictionary%,LLP.Scripting.IDictionary.Keys,MEM.Child,COF.Args0,REF
LDA%dictionary%,LLP.Scripting.IDictionary.Exists,MEM.Child
LDA%dictionary%,LLP.Scripting.IDictionary.RemoveAll,MEM.Child
LEV.Nest0,LLP,ARG,LDA.%dictionary%,LLP.Scripting.IDictionary.CompareMode,MEM.Child,STR.AssignValue
The NotUpgraded Subcommand
Both the StrongCollections and StrongDictionaries migrations depend upon the presence of Add method calls to determine the item and key types for the individual collections. If such method calls can not be found for a given collection or dictionary then it is not upgraded. The NotUpgraded subcommand supplies the format of a message to be displayed and triggers a search for components that have not been upgraded. When there are such present in a given code project, messages are sent to the log file. The command itself is a singleton content only command. It might be as follows.<NotUpgraded>UPGRADE_TODO: $1d $2d cannot be upgraded to generic.</NotUpgraded>
UPGRADE_TODO: Scripting.Dictionary SQLDiff.clsIndices.mdicIndices cannot be upgraded to generic. UPGRADE_TODO: Scripting.Dictionary SQLDiff.clsFKeys.m_FKeyDic cannot be upgraded to generic.
Scripting Commands
The migrations of particular components within particular source code projects do not always succeed, because the information needed to perform them is either not available of not in the expected form. The tool supports an ExecuteCommand event which is triggered whenever the tool encounters a script command at the primary level of a command script that it does not recognize. This event can be used to supply additional information to the migrations of particular code projects within the translation scripts for those projects.The DictionaryAdd Scripting Command
The DictionaryAdd scripting command supports the Collections.StrongDictionaries migration. That migration depends upon finding calls to this method in the code.<method id="Add" type="Void"> <argument id="Key" type="Variant" status="ByVal"/> <argument id="Item" type="Variant" status="ByVal"/> </method>
Attribute | Description |
Identifier | Specifies the fully qualified name of a dictionary component in the code project. |
Key | Specifies the key type for the dictionary. |
Item | If present, specifies the item type for the dictionary. |
If the identifier cannot be located, then the command is simply ignored. As an example assume that a translation script whose Collections specification contained a NotUpgraded subcommand, sent this message to the log
UPGRADE_TODO: Scripting.Dictionary SQLDiff.GenerateScript.dicGenerateIndex cannot be upgraded to generic.
</Compile> <DictionaryAdd identifier="SQLDiff.GenerateScript.dicGenerateIndex" Key="String" Item="Boolean" /> <Analyse />
Comparing files ..\SQLDiff.bnd and ..\SQLDIFF.SAV ***** ..\SQLDiff.bnd Dictionary<string, bool> generateIndex; generateIndex = new Dictionary<string, bool>(); generateIndex[strIndexName] = true; ***** ..\SQLDIFF.SAV Scripting.Dictionary generateIndex; generateIndex = new Scripting.Dictionary(); generateIndex.set_Item(strIndexName, true);
The CollectionAdd Scripting Command
The CollectionAdd scripting command supports the Collections.StrongCollections migration. That migration depends upon finding calls to this method in the code.<method id="Add" type="String" opcode="COL.1"> <argument id="Item" type="Variant" status="ByVal"/> <argument id="Key" type="String" status="ByVal" optional="DEF.NullObject"/> <argument id="Before" type="Variant" status="ByVal" optional="%15%"/> <argument id="After" type="Variant" status="ByVal" optional="%15%"/> </method>
Attribute | Description |
Identifier | Specifies the fully qualified name of a collection component in the code project. |
Item | Specifies the item type for the collection. |
Key | If present, specifies the key type for the collection. |
If the identifier cannot be located, then the command is simply ignored. Note that the actual CollectionAdd commands must follow the Compile and precede the Analyse statements.
Table of Contents
, multiple selections available,
Related content
Support Statement: MigrationSupport (gmRTL) Source Code
Support Statement: MigrationSupport (gmRTL) Source Code
More like this
Support Statement: Deprecating MigrationSupport namespaces
Support Statement: Deprecating MigrationSupport namespaces
More like this
gmStudio Release News: Tuesday, 23-March-2021
gmStudio Release News: Tuesday, 23-March-2021
More like this
gmStudio Release News: Wednesday, 01-Dec-2021
gmStudio Release News: Wednesday, 01-Dec-2021
More like this
gmStudio Release News: Sunday, 11-December-2022
gmStudio Release News: Sunday, 11-December-2022
More like this
gmStudio Release News: Monday, 17-June-2019
gmStudio Release News: Monday, 17-June-2019
More like this