<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Dwarfsoft [GPA] &#187; Batch</title>
	<atom:link href="http://www.dwarfsoft.com/blog/tag/batch/feed/" rel="self" type="application/rss+xml" />
	<link>http://www.dwarfsoft.com/blog</link>
	<description>Great Programming Artistry</description>
	<lastBuildDate>Tue, 22 Jun 2010 21:01:59 +0000</lastBuildDate>
	<generator>http://wordpress.org/?v=2.9.2</generator>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
			<item>
		<title>GroupWise Audit and Batch/Cmd Escaping</title>
		<link>http://www.dwarfsoft.com/blog/2009/07/27/groupwise-audit-and-batchcmd-escaping/</link>
		<comments>http://www.dwarfsoft.com/blog/2009/07/27/groupwise-audit-and-batchcmd-escaping/#comments</comments>
		<pubDate>Mon, 27 Jul 2009 04:52:37 +0000</pubDate>
		<dc:creator>dwarfsoft</dc:creator>
				<category><![CDATA[Novell]]></category>
		<category><![CDATA[Scripting]]></category>
		<category><![CDATA[Tweet]]></category>
		<category><![CDATA[Uncategorized]]></category>
		<category><![CDATA[Work]]></category>
		<category><![CDATA[Batch]]></category>
		<category><![CDATA[CMD]]></category>
		<category><![CDATA[Groupwise]]></category>
		<category><![CDATA[Script]]></category>
		<category><![CDATA[Strings]]></category>

		<guid isPermaLink="false">http://www.dwarfsoft.com/blog/?p=259</guid>
		<description><![CDATA[I was required recently to audit passwords on all the Novell GroupWise accounts in the cluster. This was not too much of a problem using existing solutions, except that all the existing solutions limited searches to either Users, or to objects with the &#8220;NGW: Object ID&#8221; attribute.
The Solutions I found include Check GroupWise Users for [...]]]></description>
			<content:encoded><![CDATA[<p>I was required recently to audit passwords on all the Novell GroupWise accounts in the cluster. This was not too much of a problem using existing solutions, except that all the existing solutions limited searches to either Users, or to objects with the &#8220;NGW: Object ID&#8221; attribute.</p>
<p>The Solutions I found include <a href="http://www.novell.com/coolsolutions/tools/17221.html">Check GroupWise Users for Password &#8211; Batch</a> and <a href="http://www.novell.com/communities/node/5753/check-groupwise-users-password">Check GroupWise Users for Password &#8211; Exe</a>, both of which made use of <a href="http://www.novell.com/coolsolutions/tools/13913.html">GWSend</a>. Being an avid scripting aficionado myself I opted for the first, so I could make changes.</p>
<p>First step was to export all User Objects with NGW: Object ID into an Excel sheet using <a href="http://www.novell.com/coolsolutions/tools/13908.html">DSReport</a>. Then export all GroupWise External Entities with NGW: Object ID to a different Excel Sheet. Finally I needed to export all GroupWise Resources, which do not have an NGW: Object ID, but their CN is effectively the NGW: Object ID for which we can log in and try to send emails. Upon completing this I compiled a single list of Allusers.csv which had the NGW: ObjectID/CN in the first column, and the DN for the Object Name and Location within the tree. This makes it far easier to track down the location of generic accounts (Something that none of the scripts account for).</p>
<p><span id="more-259"></span></p>
<p>So, my first run through the script ended in Fail. Apparently the above solutions are expecting Email Addresses without Spaces. So after fixing that up I run it again, and again hit a Fail. Apparently the above solutions fail to account for special characters (/ in this instance). So, I fix this up as well&#8230; My script runs, I get a report, then I realise that I have more results than original users. Apparently the above script also assumes that there is only 1 line in the result.txt file. So I modified the script to only print out a result once all lines of the result.txt file had been parsed, using a SET to determine if the &#8220;Error:&#8221; had been encountered or not (Error indicating that a Password is on the account, no Error indicating that there is No password).</p>
<p>So, now I run through and notice that the audit fails to log the full line of text, even though it is capable of actually running the audit against that account. Fortunately none of the accounts without passwords were affected by this, but I decided to fix this for future runs. To break down the issues I needed to fix into a list here is the summary:</p>
<ol>
<li>We have spaces in our Email Account Names</li>
<li>We have /&#8217;s in our Email Account Names</li>
<li>We have &amp;&#8217;s in our Email Account Names</li>
<li>We are trying to create a HTML file with the above special characters (specifically &amp;)</li>
<li>&amp; doesn&#8217;t tend to work too nicely in batch files when using an open Echo (tries to run everything after the &amp; as a program</li>
<li>I wanted to be able to track back to objects (DN&#8217;s) as opposed to just the GroupWise Account Name</li>
</ol>
<p>Spaces were easy to fix&#8230; When running the command against an account simply put the &#8220;account in quotes&#8221;. The next step is to handle the Email and DN objects as Strings within Quotes as well, but we don&#8217;t want to output the quotes, which means we need to handle them properly. Examine the following issue:</p>

<div class="wp_syntax"><div class="code"><pre class="winbatch" style="font-family:monospace;">SET <span style="color: #0080FF; font-weight: bold;">String</span>=A <span style="color: #0080FF; font-weight: bold;">String</span> With Special Chars<span style="color: #66cc66;">&amp;</span>Slashes<span style="color: #66cc66;">/</span> <span style="color: #800080;">In</span> \ it<span style="color: #66cc66;">^</span> OK
ECHO <span style="color: #66cc66;">%</span>String<span style="color: #66cc66;">%</span></pre></div></div>

<p>This will return &#8220;&#8216;Slashes&#8217; is not recognized as an internal or external command, operable program or batch file.&#8221; Also, if we try and escape each of the characters we run into issues too.</p>

<div class="wp_syntax"><div class="code"><pre class="winbatch" style="font-family:monospace;">SET <span style="color: #0080FF; font-weight: bold;">String</span>=A <span style="color: #0080FF; font-weight: bold;">String</span> With Special Chars<span style="color: #66cc66;">&amp;</span>Slashes<span style="color: #66cc66;">/</span> <span style="color: #800080;">In</span> \ it<span style="color: #66cc66;">^</span> OK
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:^=^^%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:\=^\%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:/=^/%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:&amp;=^&amp;%</span>
ECHO <span style="color: #66cc66;">%</span>String<span style="color: #66cc66;">%</span></pre></div></div>

<p>The Output ends up as &#8220;A String With Special Chars&#8221;. So, in order to fix this we need to handle the string within Quotes, and only output it at the end:</p>

<div class="wp_syntax"><div class="code"><pre class="winbatch" style="font-family:monospace;">SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #ff0000;">&quot;A String With Special Chars&amp;Slashes/ In \ it^ OK&quot;</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:^=^^%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:\=^\%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:/=^/%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:&amp;=^&amp;%</span>
ECHO <span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:~1,-1%</span></pre></div></div>

<p>Finally, our output is &#8220;A String With Special Chars&amp;Slashes/ In \ it^ OK&#8221;, without the quotes. So the first part works now for my output to the Logs (although my logs are CSV so I tend to leave the quotes on just in case there is a lurking comma, but this does provide a pretty Output to the Console). Now we need to convert these for HTML output.</p>

<div class="wp_syntax"><div class="code"><pre class="winbatch" style="font-family:monospace;">SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:^&amp;=^&amp;amp;%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:&lt;=^&amp;lt;%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:&gt;=^&amp;gt;%</span>
ECHO <span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:~1,-1%</span></pre></div></div>

<p>Just provides that little added extra safety net for an echo directly into a HTML log. I am annoyed that I didn&#8217;t manage to figure this out earlier as I had some scripts I went directly to VBScript for to avoid the &amp; errors I was getting. I had no idea that it was my subsequent SET x=%x:a=b%, SET x=%x:c=d% calls on an unquoted string.</p>
<p>The one caveat I have found to this is fixing quote marks within a string. I have yet to find a way to strip them effectively without breaking the string/batch file in the process. The closest I have come will work in most situations, but requires a placeholder you have to be sure will never exist within your string:</p>

<div class="wp_syntax"><div class="code"><pre class="winbatch" style="font-family:monospace;">SET <span style="color: #0080FF; font-weight: bold;">String</span>=Thi\s<span style="color: #ff0000;">&quot; is ^a &amp;t&quot;</span>es<span style="color: #66cc66;">/</span>t
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #ff0000;">&quot;%String:&quot;</span>=<span style="color: #66cc66;">&amp;</span>quot<span style="color: #008000; font-style: italic;">;%&quot;</span>
Set <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:^=^^%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:&amp;=^&amp;%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:/=^/%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:\=^\%</span>
SET <span style="color: #0080FF; font-weight: bold;">String</span>=<span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:&amp;quot;=^&quot;%</span>
ECHO <span style="color: #66cc66;">%</span><span style="color: #0080FF; font-weight: bold;">String</span><span style="color: #FF1010; font-weight: bold;">:~1,-1%</span></pre></div></div>

<p>One other note I&#8217;d like to add is that of Padding numbers. This is a fairly well known feature of WinBatch, but I found it useful for getting well aligned output for which record I was currently up to:</p>

<div class="wp_syntax"><div class="code"><pre class="winbatch" style="font-family:monospace;">Set _Num=<span style="color: #cc66cc;">0</span>
...
SET <span style="color: #66cc66;">/</span>A <span style="color: #66cc66;">%</span>_Num<span style="color: #66cc66;">%</span>=<span style="color: #66cc66;">%</span>_Num<span style="color: #66cc66;">%+</span><span style="color: #cc66cc;">1</span>
SET DispNum=     <span style="color: #66cc66;">%</span>_Num<span style="color: #66cc66;">%</span>
Echo <span style="color: #66cc66;">%</span>DispNum<span style="color: #FF1010; font-weight: bold;">:~-4%</span></pre></div></div>

<p>This pads the leading digits with spaces and specifies that I want a 4 character long string returned for the end of the string. (so &#8221;   1&#8243; in the first instance but in my case up to &#8221; 900&#8243; or  &#8220;9091&#8243;).</p>
<p>Hope this adds a little food for thought.</p>
<p>Cheers, Chris.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.dwarfsoft.com/blog/2009/07/27/groupwise-audit-and-batchcmd-escaping/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>Manipulating CSV Files &#8211; Part 1</title>
		<link>http://www.dwarfsoft.com/blog/2008/09/09/manipulating-csv-files-part-1/</link>
		<comments>http://www.dwarfsoft.com/blog/2008/09/09/manipulating-csv-files-part-1/#comments</comments>
		<pubDate>Tue, 09 Sep 2008 04:44:46 +0000</pubDate>
		<dc:creator>dwarfsoft</dc:creator>
				<category><![CDATA[Certification]]></category>
		<category><![CDATA[LDAP]]></category>
		<category><![CDATA[Novell]]></category>
		<category><![CDATA[Politics]]></category>
		<category><![CDATA[Scripting]]></category>
		<category><![CDATA[Study]]></category>
		<category><![CDATA[Training]]></category>
		<category><![CDATA[Work]]></category>
		<category><![CDATA[Batch]]></category>
		<category><![CDATA[MCSA]]></category>
		<category><![CDATA[MCSA Messaging]]></category>
		<category><![CDATA[MCSE]]></category>
		<category><![CDATA[VBScript]]></category>
		<category><![CDATA[Workstation]]></category>

		<guid isPermaLink="false">http://www.dwarfsoft.com/blog/?p=67</guid>
		<description><![CDATA[Before I jump in to the fun that is the Scripting that I have been wading through this past week, I thought I&#8217;d let all of those who may care that I finally have some kind of certification. I passed Microsoft exams 70-291 and 70-284. So I am now the proud holder of a Microsoft [...]]]></description>
			<content:encoded><![CDATA[<p>Before I jump in to the fun that is the Scripting that I have been wading through this past week, I thought I&#8217;d let all of those who may care that I finally have some kind of certification. I passed Microsoft exams 70-291 and 70-284. So I am now the proud holder of a Microsoft Certified Systems Administrator Messaging Specialist certification (MCSA Messaging). Now I am concentrating on the final three exams so that I can get my MCSE.</p>
<p>After a rather hectic week of scripting a solution and then distributing it under an excessively short deadline, I have been asked to provide stats on the result of forcing this solution out to clients. The solution that I had to develop keeps tabs on a System Volume Image of PCs, and ensures that this Image never gets out of date too far. Currently I am not forcing a Store every restart, as the planned solution was to do, but I do inform the client that their current Image is out of date and ask them if they want to do a Store now. If they click on Yes then their PC is rebooted and the Store is done (providing that one of many flaws in the current Store process do not interrupt the process).<span id="more-67"></span></p>
<p>Because of some of the issues that we have had, as well as Managements want to view the impact of what we have done in a statistical format, I was required to provide a secondary application, which is just an updated version of an AutoIT Script that logs the Image Date into a log file during login. Because this file was preexisting it has been used in this way to gather stats for some time.</p>
<p>The issue is that the log file includes EVERY instance of a login for a PC, and I want only the most recent with the timestamp of the image, not all the other instances, or the log values that are too old (I was unable to clear the logs due to current procedure). Therefore I was required to learn some new things in the VBScript world. This solution is surprisingly fast considering how much data it is required to crunch in order to produce the output CSV file.</p>
<pre>Const ForReading = 1
Const ForWriting = 2

Dim arrLines()
i = 0

Set objDictionary = CreateObject("Scripting.Dictionary")
Set objFSO = CreateObject("Scripting.FileSystemObject")

objStartFolder = WScript.ScriptFullNameobjStartFolder = Replace  (objStartFolder, WScript.ScriptName, "") 

Set objFolder = objFSO.GetFolder(objStartFolder &amp; "Logs")

Set colFiles = objFolder.Files
For Each objFile in colFiles
    Set textFile = objFSO.OpenTextFile(objFile.Path, ForReading)

    Do Until textFile.AtEndOfStream
        Redim Preserve arrLines(i)
        arrLines(i) = textFile.ReadLine
        i = i + 1
    Loop

    For i = Ubound(arrLines) to LBound(arrLines) Step -1
        splt = Split (arrLines(i),",")

        If (UBound(splt) &lt; 19) Then
            i = LBound(arrLines)
        Else
            If Splt(19) &lt;&gt; "" Then
               If Not objDictionary.Exists(splt(1)) Then
                  objDictionary.Add splt(1),splt(19)
               End If
            End If
        End If
    Next

    ReDim arrLines(0)
    i = 0
    textFile.Close
Next

Set writeFile = objFSO.CreateTextFile(objStartFolder &amp; "Stores.csv")
For Each strKey in objDictionary.Keys
   writeFile.WriteLine strKey &amp; "," &amp; objDictionary.Item(strKey)
Next
writeFile.Close</pre>
<p>In this particular script the splt Array has 20 columns (or 19 for those items that were prior to me updating the script). Column 2 (or splt(1)) is the PC Name, and Column 20 (or splt(19)) is the Timestamp of the Image. This script reads through every file in the Logs folder (in the same path as the script) and reads the file in reverse until it comes across the entries with only 19 columns. For every unique PC Name that it finds it adds this into a Dictionary object with the Timestamp as the value. The Dictionary is then written to the new CSV file one at a time.</p>
<p>In order to get the stats on how effective this has been, I paste a couple of Excel Macros:</p>
<pre>=COUNTIF(B:B,"&gt;22/08/2008")
=COUNT(B:B)</pre>
<p>The magic date here is the 22nd of August 2008. This was chosen purely because it was a recent date, and it was newer than most of the Images for the PCs within the IT Department, who already had the AutoStore scripts working on their machines. Which provided for a controlled Environment for testing some of the modifications I was forced to make last minute.</p>
<p>One thing that I could do to improve the accuracy of this script is to pull out all the Workstation Names from Novell using LDAP and get a total overall status of where the Store dates are across all PCs, but that would pollute the pool with a large number of machines that have been decommissioned (they still appear as a Workstation in Novell until they have not been logged on to for 3 months, then they fall off the system).</p>
<p>On a side note, I have to pull in the Logs into the Logs directory in some way, which is achieved for me by running a Batch file which I have written as follows:</p>
<pre>@ECHO OFF
ECHO Pulling Logs %DATE% - %TIME%&gt;&gt;Log.txt
ECHO.&gt;&gt;Log.txt
%~d0
cd %~dp0
FOR /F "tokens=*" %%A IN (Sources.txt) DO (
  FOR /F "tokens=1 delims=" %%Z IN ("%%A") DO (
     COPY "%%AlogsLogIn.csv" "%~dp0Logs%%Z.csv" /A /D /Y /Z &gt;&gt;Log.txt
  )
)
ECHO.&gt;&gt;Log.txt
ECHO Current Stores being Rebuilt at %DATE - %TIME%&gt;&gt;Log.txt
WScript Stores.vbs
Start Excel Stores.csv
ECHO Log Pull Finished at %DATE% - %TIME%&gt;&gt;Log.txt</pre>
<p>This script uses a file called &#8220;Sources.txt&#8221; which is a list of paths on newlines which point to where each of the Logs are stored. Each line consists of something like \ServerNameSYSPUBLIC. The line FOR /F &#8220;tokens=1 delims=&#8221; %%Z IN (&#8220;%%A&#8221;) allows me to pull out the servername from each line in the file, so that when I copy the file from the server, I can rename it so that its source can be identified.</p>
<p>I think this will conclude todays lesson in VBScript and Batch, and I&#8217;d best get back to studying.</p>
<p>Cheers, Chris.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.dwarfsoft.com/blog/2008/09/09/manipulating-csv-files-part-1/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
