/*----------------------------------------------------------------------------*/ /* */ /* Copyright (c) 2002-2022 Rony G. Flatscher. All rights reserved. */ /* Copyright (c) 2023 Rexx Language Association. All rights reserved. */ /* */ /* This program and the accompanying materials are made available under */ /* the terms of the Common Public License v1.0 which accompanies this */ /* distribution. A copy is also available at the following address: */ /* https://www.oorexx.org/license.html */ /* */ /* Redistribution and use in source and binary forms, with or */ /* without modification, are permitted provided that the following */ /* conditions are met: */ /* */ /* Redistributions of source code must retain the above copyright */ /* notice, this list of conditions and the following disclaimer. */ /* Redistributions in binary form must reproduce the above copyright */ /* notice, this list of conditions and the following disclaimer in */ /* the documentation and/or other materials provided with the distribution. */ /* */ /* Neither the name of Rexx Language Association nor the names */ /* of its contributors may be used to endorse or promote products */ /* derived from this software without specific prior written permission. */ /* */ /* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS */ /* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT */ /* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS */ /* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT */ /* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, */ /* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED */ /* TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, */ /* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY */ /* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING */ /* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS */ /* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /* */ /*----------------------------------------------------------------------------*/ /*********************************************************************** program: reg_classids4ole.rex purpose: analyzes the Windows registry for CLSID-->PROGIDs, defines a class and collections usage: use the "::requires" directive to incorporate classes and routines ***********************************************************************/ .local~rgf.debug=0 -- set debug level -- initialization .local ~ rgf.registry = .WindowsRegistry~new -- create a registry object save it in .local .local ~ rgf.hKey_classes_root = .rgf.registry~classes_root -- save handle to root of classes -- get handle to key "CLSID" located in top level key "HKEY_CLASSES_ROOT" .local~rgf.hKey_CLSID_root = .rgf.registry~open(.rgf.hkey_classes_root, "CLSID", "READ") ::requires winsystm.cls -- get access to the Windows registry /** Returns .true if argument (a PROGID or CLSID) exists in HKEY_CLASSES_ROOT or HKEY_CLASSES_ROOT\CLSID, .false else. */ ::routine reg_exists public parse arg reg_name -- query key (case insensitive lookup by Windows), if not available "0" will be returned as the handle hKey = .rgf.registry~open(.rgf.hkey_classes_root, reg_name, "READ") if hKey=0 then do hKey = .rgf.registry~open(.rgf.hkey_classes_root, "CLSID\" || reg_name, "READ") if .rgf.debug > 1 then say "reg_exists(): not found in root, looking in CLSID\ ..." end if .rgf.debug > 1 then say "reg_exists(): reg_name="pp(reg_name) "hKey="pp(hKey) if .rgf.debug > 2 then call dump_handle hKey .rgf.registry~close(hKey) -- close (free) handle return hKey <> 0 /** Dumps the values and subkey names of the key referred to by the received handle */ ::routine dump_handle use arg hKey say s. = .rgf.registry~query(hKey) -- query key via its handle tmp="class name="pp(s.class) "subkeys="pp(s.subkeys) "values="pp(s.values) , "last changed on:" pp(s.date) pp(s.time) say tmp ind=" " -- leadin for text-output if s.values>0 then -- show all values of the key, if any do say; say ind "listing all values for this key:" .rgf.registry~listvalues(hKey, stem.) -- get a list of all values for this key do i=1 to s.values tmp=ind ind "name:" pp(stem.i.name) "type:" pp(stem.i.type) "data:" pp(stem.i.data) say tmp end end if s.subkeys>0 then -- show names of all subkeys of the key, if any do say; say ind "listing all subkeys of this key:" .rgf.registry~list(hKey, stem.) -- get a list of all subkeys of this key do i=1 to s.subkeys tmp=ind ind "subkey name:" pp(stem.i) say tmp end end say copies("-", 30) return /** Returns string value of given argument, enclosed in square brackets: "PrettyPrint" ;) . */ ::routine pp return "[" || arg(1)~string || "]" /** Routine expands environment variables in the data. */ ::routine expand public use arg data tmp="" do while data <> "" parse var data before "%" v "%" data tmp = tmp || before || value(v, , "ENVIRONMENT") -- replace environment variable with its value end return tmp /** Routine to get the 'CLSID'-object from a given progid or clsid. */ ::routine get_clsid_object public use arg prog_or_clsid clsid2progid=.clsid~clsid2progid -- get access to clsid2porgid[clsid]=progid o=clsid2progid[prog_or_clsid] -- assume CLSID in hand if o=.nil then do o=clsid2progid~allat(prog_or_clsid)[1] -- oops, maybe a PROGID ? if o=.nil then o=create_clsid_object(prog_or_clsid) -- does not exist as of yet, create it ! end else return .clsid~all_clsid[o] return o /** Routine either gets a PROGID or a CLSID, analyzes it and returns an instance of class CLSID or .nil, if PROGID or CLSID not found. */ ::routine create_clsid_object public use arg clsid, bCLSID -- it's either a PROGID or a CLSID bClsid=(bClsid=.true) -- determine if already CLSID given if \bClsid then -- undetermined, could be PROGID or CLSID do -- a PROGID? hKey = .rgf.registry~open(.rgf.hkey_classes_root, clsid, "READ") if hKey\=0 then -- a PROGID, get CLSID do hKey2=.rgf.registry~open(.rgf.hkey_classes_root, clsid"\CLSID") -- get handle to subkey -- 2002-12-29, ---rgf if hkey2=0 then -- subkey "CLSID" not found, maybe "CurVer" pointing to actual PROGID ? do hKey3=.rgf.registry~open(.rgf.hkey_classes_root, clsid"\CurVer") -- get handle to subkey if hKey3 <> 0 then -- o.k. such a subkey was found, now use it to find CLSID do s. = .rgf.registry~getValue(hkey3, "") -- get default value = PROGID curVerPROGID=s.data -- get default value = PROGID .rgf.registry~close(hKey3) -- close hKey3 drop s. .rgf.registry~close(hkey2) -- close hkey2 hKey2=.rgf.registry~open(.rgf.hkey_classes_root, curVerPROGID"\CLSID") -- get CLSID of "CurVer"-PROGID end end s. = .rgf.registry~getValue(hKey2, "") -- get default value -- if .rgf.debug > 1 then say "create_c_o(): hkey="pp(hkey) "progid="pp(clsid) to "clssid="pp(s.data) "s.type="pp(s.type) clsid = s.data -- extract value from stem .rgf.registry~close(hKey2) -- close (free) handle .rgf.registry~close(hKey) -- close (free) handle end end -- get a handle to subkey hClsidKey = .rgf.registry~open(.rgf.hkey_classes_root, "CLSID\"clsid, "READ") if hClsidKey=0 then do msg=clsid": not found!" if window <> "WINDOW" then nop -- call alert msg else .error~say(msg) return .nil -- do return nothing end o=.clsid~new(clsid) -- create instance to store relevant information odir=o~keys -- get access to directory to contain the keys and values s. = .rgf.registry~query(hClsidKey) -- query infos on key (number of subkeys, values; value of: date, time, class (name)) o~datetime=changestr("/", s.date, "") s.time if s.values>0 then -- show all values of the key, if any do .rgf.registry~listvalues(hClsidKey, stem.) -- get a list of all values for this key -- show values of CLSID do i=1 to s.values if stem.i.name="" then -- default value in hand ? do o~description=stem.i.data -- save value with object leave i -- leave loop end end end -- process subkeys .rgf.registry~list(hClsidKey, list2.) -- get all subkeys keysDir=.clsid~keysDir -- get directory of interesting subkeys from class CLSID do idx2=1 to s.subkeys -- iterate over subkeys and their values if \ keysDir~hasentry(list2.idx2) then iterate -- if subkey-name is not of interest, iterate -- get a handle to the subkey in hand keyName=list2.idx2 hSubKey2=.rgf.registry~open(hClsidKey, keyName, "READ") if hSubKey2 \= 0 then do s2. = .rgf.registry~query(hSubKey2) -- query various infos if s2.values>0 then -- if values available, iterate over them do .rgf.registry~listvalues(hSubKey2, stem2.) -- get a list of all values for this key do i=1 to s2.values if stem2.i.name="" then -- default string value in hand, if so save do tmp="" if stem2.i.type="EXPAND" then tmp = expand(stem2.i.data) -- expand environment variable in string else tmp = stem2.i.data odir~setentry(keyName, tmp) -- save key with object leave i end end end .rgf.registry~close(hSubKey2) -- close (return) handle end end -- idx2 if o=.nil then return .nil -- iterate -- no entry found, iterate if \ odir~hasentry("PROGID") then return .nil -- iterate -- do not process a CLSID which has no value given for PROGID -- set up collections for administrating objects of this class .clsid~clsid_list~insert(clsid) -- save CLSID in list .clsid~all_clsid~setentry(clsid, o) -- save instance in collection -- set up relation between CLSID and PROGID .clsid~clsid2progid[clsid]=odir~progid -- all_progid~setentry(odir~progid, o) if odir~hasentry("VERSIONINDEPENDENTPROGID") then do .clsid~clsid2progid[clsid]=odir~versionIndependentProgid -- all_progid~setentry(odir~versionIndependentProgid, o) .clsid~all_progid[odir~versionIndependentProgid] = o -- say "versionIndependentProgId:" pp(odir~versionIndependentProgid) "-> o:" pp(o) "CLSID:" pp(o~clsid) end -- else do .clsid~all_progid[odir~progid] = o -- o.k., then use PROGID -- say " ProgId:" pp(odir~progId) "-> o:" pp(o) "CLSID:" pp(o~clsid) end if odir~hasentry("TREATAS") then do .clsid~clsid2progid[clsid]=odir~treatAs .clsid~treatAsSrc~put(clsid) -- save CLSID, which needs to be treated as defined in another CLSID .clsid~treatAsTgt~put(odir~treatAs) -- save target CLSID end return o /* ================================================================================= */ /* class to represent all interesting information about CLSID's */ ::class clsid public /* ------------------------- class methods ----------------------------------- */ ::method init class expose all_clsid all_progid clsid_list clsid2progid keysList keysDir treatAsSrc treatAsTgt all_clsid = .directory~new -- collects all instances, indexed by CLSID all_progid= .directory~new -- collects all instances, indexed by PROGID+VERSIONINDEPENDENTPROGID clsid_list= .list~new clsid2progid = .relation~new -- maps clsids to ProgID, VersionIndependentProgID and TreatAs treatAsSrc = .set~new treatAsTgt = .set~new keysList = .list~of( "VersionIndependentProgID", , "ProgID", , "Version", , "TreatAs", , -- points to CLSID which contains definitions "LocalServer", , -- binary program "LocalServer32", , -- binary program "ScriptletURL", , -- script-program (instead of LocalServer32) "InProcHandler", , -- handler being used "InProcHandler32", , -- handler being used "InProcServer", , -- server for handler being used "InProcServer32" ) -- server for handler being used keysDir=.directory~new do item over keysList keysDir~setentry(item, item) -- add name of key into directory end ::method all_clsid attribute class -- directory containing clsid as key to instance of this class ::method all_progid attribute class -- directory containing versionindependent/progid as key to instance of this class ::method clsid_list attribute class -- list: containing CLSIDs in order of appearence ::method clsid2progid attribute class -- relation: maps class-ids (idx) to progids, independent progids, treat-as progids ::method keysList attribute class -- list: denotes the key-names regarded to be interesting ::method keysDir attribute class -- directory: allow case-independent look-up of interesting keys ::method treatAsSrc attribute class -- set: contains CLSIDs which depend on other CLSID for operation ::method treatAsTgt attribute class -- set: contains CLSID pointed to -- analyze and build classes ::method analyze_and_build class expose all_clsid all_progid clsid_list clsid2progid keysList keysDir treatAsSrc treatAsTgt handle = .rgf.registry~open(.rgf.hKey_classes_root, "CLSID", "READ") clslist. = .rgf.registry~query(handle) -- query number of subkeys, values, value of: class (name), date, time .rgf.registry~list(handle, subKeyList.) -- get the subkeys (i.e. class-ids in UUID-format do idx =1 to clslist.subkeys -- iterate over subkeys clsid=subKeyList.idx o=create_clsid_object( clsid, .true ) -- indicate that CLSID-value in hand if .rgf.debug>0 then if idx > 200 then leave -- on debug, just process the first 50 entries end -- .rgf.registry~close(hSubKey) -- end .rgf.registry~close(handle) /* ------------------------- instance methods -------------------------------- */ ::method init expose clsid description keys use arg clsid keys=.directory~new ::method clsid attribute -- stores the CLSID as found in registry ::method description attribute -- stores the default value of CLSID (describing it) ::method keys attribute -- directory possessing the values of found keys ::method datetime attribute -- date and time of entry in registry -- this defines the default string value for this object ::method makestring expose clsid description datetime return clsid pp(description) pp(datetime) -- this dumps the keys ::method dumpkeys expose keys indent=copies(" ", 8) do item over self~class~keysList if keys~hasentry(item) then say indent item"="pp(keys~entry(item)) end