368 lines
16 KiB
OpenEdge ABL
368 lines
16 KiB
OpenEdge ABL
|
/*----------------------------------------------------------------------------*/
|
||
|
/* */
|
||
|
/* 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: "<strong>P</strong>retty<strong>P</strong>rint" ;) . */
|
||
|
::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
|
||
|
|