1#!/usr/bin/env python3
  2# ^shebang:  load the correct Python interpreter and environment.
  3#============================================================================
  4#
  5# NAME
  6#
  7#     PrimpolyRunningTimeAnalysis.py
  8#
  9# DESCRIPTION
 10#
 11#     Primitive polynomial running time plot in Python using Sympy and Numpy.
 12#
 13# USAGE
 14#
 15#     Collect data using the script 
 16#
 17#         runningTime.sh
 18#
 19#     Copy the times to this script.
 20#
 21# NOTES
 22#
 23#     numpy:                            http://www.numpy.org
 24#     matplotlib                        https://matplotlib.org
 25#     Python interpreter:               http://www.python.org
 26#     Python tutorial and reference:    htttp://docs.python.org/lib/lib.html
 27#
 28# LEGAL
 29#
 30#     Primpoly Version 16.3 - A Program for Computing Primitive Polynomials.
 31#     Copyright (C) 1999-2025 by Sean Erik O'Connor.  All Rights Reserved.
 32#
 33#     This program is free software: you can redistribute it and/or modify
 34#     it under the terms of the GNU General Public License as published by
 35#     the Free Software Foundation, either version 3 of the License, or
 36#     (at your option) any later version.
 37#
 38#     This program is distributed in the hope that it will be useful,
 39#     but WITHOUT ANY WARRANTY; without even the implied warranty of
 40#     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 41#     GNU General Public License for more details.
 42#
 43#     You should have received a copy of the GNU General Public License
 44#     along with this program.  If not, see <http://www.gnu.org/licenses/>.
 45#    
 46#     The author's address is seanerikoconnor!AT!gmail!DOT!com
 47#     with the !DOT! replaced by . and the !AT! replaced by @
 48#
 49#============================================================================
 50
 51import os   # Path 
 52import sys  # argc, argv, read/write I/O
 53import platform # platform, uname
 54import re  # Regular expression support
 55import math
 56import numpy as np
 57import pylab as py
 58import subprocess
 59import pickle                  # Python's own data file format.
 60from deepdiff import DeepDiff  # Deep difference for complicated data structures.
 61
 62def write_to_pickle_file( data, file_name ):
 63    """Write data to a pickle file."""
 64    try:
 65        with open(file_name, 'wb') as f:
 66            pickle.dump(data, f, pickle.HIGHEST_PROTOCOL)
 67
 68        if not os.path.exists( file_name ):
 69            print( f"ERROR:  Problem writing to {file_name} file doesn't exist." )
 70            return None
 71
 72    except Exception as detail:
 73        print( "ERROR:  Problem writing to file {file_name}  Caught RuntimeError exception; detail = {detail}")
 74        return None
 75
 76
 77def read_from_pickle_file( file_name ):
 78    """Read data from a pickle file."""
 79
 80    if not os.path.exists( file_name ):
 81        print( f"ERROR:  {file_name} does not exist" )
 82        return None
 83    try: 
 84        with open( file_name, 'rb') as f:
 85            data = pickle.load(f)
 86    except Exception as detail:
 87        print( "ERROR:  Problem reading file {file_name} Caught RuntimeError exception; detail = {detail}")
 88        return None
 89
 90    return data
 91    
 92def data_differences( data1, data2 ):
 93    """Return the differences between to dictionaries."""
 94
 95    changes = DeepDiff(data1, data2)
 96    return changes
 97
 98
 99def get_float_pattern():
100    """Generate a regular expression which matches floating point numbers.
101
102    Regular expressions format for Python is referenced here:  https://docs.python.org/3/howto/regex.html
103
104    From [how to extract floating number from string](https://stackoverflow.com/questions/4703390/how-to-extract-a-floating-number-from-a-string)
105    Some people, when confronted with a problem, think 'I know, I'll use regular expressions.'  Now they have two problems.
106         -Jamie Zawinski <jwz@netscape.com> wrote on Tue, 12 Aug 1997 13:16:22 -0700:
107     """
108
109    float_pattern = r"""
110            [-+]?                       # Optional sign:   + or - or nothing
111            (?:
112                (?: \d* \. \d+)         # .1 .12, 1.2 1.23
113                |
114                (?: \d+ \.?)            # 1. 12. 1 12
115            )
116            (?: [Ee] [+-]? \d+ )?       # Optional exponent:  e1 e-2
117            """
118
119    return float_pattern
120
121
122def parse_primpoly_timing_output(line_of_file, **kwargs):
123    """
124    Parse the output of the command
125        time Primipoly p n 
126    to get the running time.
127    """
128
129    real_time = None 
130
131    # kwargs is a dictionary.
132    for key, value in kwargs.items():
133        if key in kwargs:
134            print( f"key = {key} value = {value}" )
135
136    debug = False
137    debug_value = kwargs.get("debug")
138    if debug_value is not None and debug_value is True:
139        debug = True
140
141    if line_of_file is None or len(line_of_file) == 0:
142        return None
143
144    #first_char = line_of_file[0]
145    # print("first char = ", first_char)
146
147    # Don't forget to backslash the vertical bar:  \| since we want to match this char.
148    # Match all spaces with \s+ even those within a line.
149    # Tremendous speedup if we check the first char in the line before applying regular expressions.
150    #if first_char == 'X':
151    pattern1 = fr"""
152                  real
153                  \s+
154                  (?P<real_time>
155                      {get_float_pattern()}
156                  )
157               """
158
159    pattern2 = fr"""
160                real
161                \s+
162                (?P<real_time>
163                    {get_float_pattern()}
164                )
165               """
166    if debug:
167        print(f"pattern1 = {pattern1}")
168        print(f"pattern2 = {pattern2}")
169        print(f"line of file =\n|{line_of_file}|")
170
171    p1 = re.compile(pattern1, re.VERBOSE | re.IGNORECASE)
172    p2 = re.compile(pattern2, re.VERBOSE | re.IGNORECASE)
173
174    # Try both matches.
175    m1 = p1.search(line_of_file)
176    m2 = p2.search(line_of_file)
177
178    if m1:
179        real_time = float(m1.group('real_time'))
180        if debug:
181            print( "Match pattern 1. real_time = {real_time}")
182    elif m2:
183        real_time = float(m2.group('real_time'))
184        if debug:
185            print( "Match pattern 2. real_time = {real_time}")
186
187    return real_time
188
189
190def time_primpoly(**kwargs):
191
192    # kwargs is a dictionary.
193    for key, value in kwargs.items():
194        if key in kwargs:
195            print( f"key = {key} value = {value}" )
196
197    # Enable debugging?
198    debug = False
199    debug_value = kwargs.get("debug")
200    if debug_value is not None and debug_value is True:
201        debug = True
202
203    # All the degrees we wish to test.
204    degrees = [   2,  10,  30, 50, 70, 90, 100, 120, 140, 145, 150, 160, 180, 200, 202, 210, 300, 400, 500, 550 ]
205    times   = []
206
207    p = 2
208    for n in degrees:
209        # Run time Primpoly and collect the outputs.
210        print( f"Primpoly p = {p} n = {n}", file=sys.stderr )
211        
212        standard_output = None
213        standard_error = None
214
215        # Set up the executable and the arguments for Linux.  Try that first.
216        executableFileName="Bin/Primpoly.exe"
217        timeCommand="/usr/bin/time" 
218        timeOptions = "-f 'real %e'"  # %e is Wall clock time in seconds
219        completed_process=subprocess.run([timeCommand, timeOptions, executableFileName, str(p), str(n)], capture_output=True)
220        if completed_process.returncode == 0:
221            standard_output = completed_process.stdout.decode("utf-8")
222            standard_error = completed_process.stderr.decode("utf-8")
223        else:
224            if debug:
225                print("Could not run Linux version of time Primpoly.  Running macOS version of time Primpoly")
226            # Didn't work?  Set up the executable and the arguments for macOS.  Try that next.
227            executableFile="Bin/Primpoly.exe"
228            timeCommand="time"
229            completed_process=subprocess.run([timeCommand, executableFileName, str(p), str(n)], capture_output=True)
230            if completed_process.returncode == 0:
231                standard_output = completed_process.stdout.decode("utf-8")
232                standard_error = completed_process.stderr.decode("utf-8")
233            # Still didn't work?  Then abort.
234            else:
235                print( "ERROR:  Could not run time command on the Primpoly executable.  Did you build it with make?  Aborting..." )
236                return None
237
238        if debug:
239            for line in standard_output.split('\n'):
240                print( f">>> {line}" )
241            for line in standard_error.split('\n'):
242                print( f"!!! {line}" )
243
244        # Parse the results to get the running time.
245        running_time_sec = parse_primpoly_timing_output(standard_error)
246        if running_time_sec is None:
247            print( "ERROR:  Could not parse the running time.  Aborting..." )
248            return None
249        else:
250            print( f"\trunning time = {running_time_sec}", file=sys.stderr )
251            
252        times.append( running_time_sec )
253   
254    return degrees, times
255
256
257def plot_running_times( file_name, **kwargs ):
258    """Plot the running times"""
259
260    debug = False
261    debug_value = kwargs.get("debug")
262    if debug_value is not None and debug_value is True:
263        debug = True
264
265    if (timing_data := read_from_pickle_file( file_name )) == None:
266        print( f"ERROR:  No data in the running times file or no file {file_name}")
267        return
268    elif debug:
269        print( f"timing data (computer name, degrees and times = \n\t{timing_data}" )
270
271    try:
272        py.figure('Primpoly Running Time', figsize=(20,10))
273        py.title('Running Time vs Degree for Mod 2 Primitive Polynomials') 
274        py.xlabel('Polynomial Degree') 
275        py.ylabel('Running Time, Seconds') 
276        py.semilogy()
277
278        legends = []
279        for computer_name, degrees_and_times in timing_data.items():
280            degrees, times = degrees_and_times
281            if debug:
282                print(f"degrees = {degrees}")
283                print(f"times   = {times}")
284                print(f"name    = {computer_name}")
285            
286            py.plot( degrees, times )
287            legends.append( computer_name )
288
289        py.legend(legends)
290        py.show()
291        return
292    except Exception as err:
293        print( f"ERROR:  Plotting went haywire for file {file_name}.  Caught a general Exception {err}" )
294        return
295
296def compute_running_time_and_record(file_name): 
297    """Time my primitive polynomial program and record the times."""
298
299    # Do we have any previously recorded timing data?  If not, create an empty dictionary for it.
300    if (timing_data := read_from_pickle_file( file_name )) == None:
301        timing_data = {}
302
303    # Find out which type of computer system we are running on.
304    if platform.uname().system == 'Darwin' and platform.uname().machine == 'arm64' and platform.uname().processor == 'arm':
305        computer_name = "MacBook Pro 2021 Apple M1 Max macOS"
306    elif platform.uname().system == 'Linux' and platform.uname().machine == 'x86_64' and platform.uname().processor == 'x86_64':
307        computer_name = "CyberPowerPC Model C Series AMD Intel Ubuntu/Linux 22.04 LTS Jammy Jellyfish"
308    else:
309        computer_name = None
310        print( f"ERROR:  unknown computer type" )
311        return
312
313    # Run the timing analysis.
314    degrees, times = time_primpoly()
315
316    if computer_name not in timing_data:
317        # If we don't yet have data for this computer, create a new dictionary entry.
318        timing_data[ computer_name ] = [degrees, times]
319    else:
320        # Else overwrite the old timing data.
321        timing_data[ computer_name ] = [degrees, times]
322
323    # Record to file.
324    write_to_pickle_file( timing_data, file_name )
325
326#    timing_data_results = read_from_pickle_file( file_name )
327#    diff = data_differences( timing_data, timing_data_results )
328#    if diff is None or len(diff) == 0:
329#        print( "no differences" )
330
331
332def update_running_times_and_plot_all():
333    """Read the running time from file and plot it"""
334
335    print( "Running Python Version {0:d}.{1:d}.{2:d}".format( sys.version_info[ 0 ], sys.version_info[ 1 ], sys.version_info[ 2 ] ) )
336
337    file_name = "primpoly_running_time.pickle"
338
339    # Compute running time on this system and record to file.
340    compute_running_time_and_record(file_name)
341
342    # Plot all running times so far recorded.
343    plot_running_times(file_name)
344
345if __name__ == '__main__':
346    """ If you run this as a standalone program:
347             chmod +x PrimpolyRunningTimeAnalysis.py
348             ./PrimpolyRunningTimeAnalysis.py
349    """
350
351    update_running_times_and_plot_all()