معرفی شرکت ها


cpypp-1.1.0


Card image cap
تبلیغات ما

مشتریان به طور فزاینده ای آنلاین هستند. تبلیغات می تواند به آنها کمک کند تا کسب و کار شما را پیدا کنند.

مشاهده بیشتر
Card image cap
تبلیغات ما

مشتریان به طور فزاینده ای آنلاین هستند. تبلیغات می تواند به آنها کمک کند تا کسب و کار شما را پیدا کنند.

مشاهده بیشتر
Card image cap
تبلیغات ما

مشتریان به طور فزاینده ای آنلاین هستند. تبلیغات می تواند به آنها کمک کند تا کسب و کار شما را پیدا کنند.

مشاهده بیشتر
Card image cap
تبلیغات ما

مشتریان به طور فزاینده ای آنلاین هستند. تبلیغات می تواند به آنها کمک کند تا کسب و کار شما را پیدا کنند.

مشاهده بیشتر
Card image cap
تبلیغات ما

مشتریان به طور فزاینده ای آنلاین هستند. تبلیغات می تواند به آنها کمک کند تا کسب و کار شما را پیدا کنند.

مشاهده بیشتر

توضیحات

c-style preprocessor for Python
ویژگی مقدار
سیستم عامل -
نام فایل cpypp-1.1.0
نام cpypp
نسخه کتابخانه 1.1.0
نگهدارنده []
ایمیل نگهدارنده []
نویسنده Wellington Rats
ایمیل نویسنده wellrats@gmail.com
آدرس صفحه اصلی https://github.com/wellrats/cpypp
آدرس اینترنتی https://pypi.org/project/cpypp/
مجوز -
Welcome to repository of cpypp ============================== **cpypp** is the implementation of a c-stype preprocessor for Python Programming Languages 2.7+ A preprocessor is a program that takes a input Source Code written using some programming language syntax and outputs an output Source Code that translate, expand or modify original programming language. For example, from this: .. code:: python from cpypp import py_preprocessor PYPP = py_preprocessor() PYPP.parse(__file__, __name__) """This source will work in both Python versions""" def main(): #ifdef __PYTHON2__ #expand print "This will work in", "Python 2", "like a charm" #else print("This will work in Python 3 like a charm") #endif if __name__ == "__main__": main() to this if we are using Python 3+ .. code:: python """This source will work in both Python versions""" def main(): print("This will work in Python 3 like a charm") if __name__ == "__main__": main() Table of Contents ================= .. contents:: Introduction ============ I love to write code, and I love to keep my code clean and organized, specially when I publish it on github for example. I love to use the latest features of a language and I specially love debug with ``print`` and a lot of dashes ``'-'`` and so on. But, when you write open source code, a lot of ``prints``, ``dashes`` and latest features are not welcome. Your code has to install and run in as many Python versions and platforms it it cans, and be kept up to date, stable and without bugs. Python 2.7 support is dropped since January 2020, but far away from be replaced. There is a lot of stuff that keeps working on it and still will be for a long time. Why? Because the cost of conversion is expensive, needs time and effort. And all this code has to be mantained, keep evolving and will need packages updates that be compatible with then and, when they began their conversion process, it will not be at once. So this was the motivation to get the best of both worlds (thank you for Hanna Montanna). Create a preprocessor for Python language that could keep final code clean, stable, runnable in any version of Python, working at run time, or used as a command line tool for batch conversion. Installing ---------- **cpypp** can be easily installed via common Python package managers such as pip. :: $ pip install cpypp You may also get the latest **cpypp** version by grabbing the source code from Github: :: $ git clone https://github.com/wellrats/cpypp $ cd cpypp $ python setup.py install To test your installation, cpypp come with a set of simple tutorials that can be executed once you have deployed the packages :: python -m cpypp.tests.test_helloworld or :: python3 -m cpypp.tests.test_helloworld Examples -------- cpypp comes with some examples to check it out go to cpypp install directory :: $ cd $(dirname $(python -c "import cpypp; print(cpypp.__file__)")) $ python tests/py2andpy3.py $ python -m cpypp py2andpy3.py -o - $ python -m cpypp py2andpy3.py -r -o - Runtime or command line processor? ================================== Usually a preprocessor do its job at compile time, but Python don't have this feature so, pypy has two options do its job: at run time, when you are executing a source file as a script or as a import module, or using a command line tool. Which to choose is up to you and your motivations to use a preprocessor. Let's see the diferences between then. Runtime preprocessor -------------------- RunTime preprocessor is when you have the source code and wants to execute this code respecting the preprocessor directives. Probably you are a developer and want to insert some tests or let the code be prepared to future releases of Python, but don't want this ``test code`` in your final code on GitHub or in you client, or wants be able to generate many diferent codes from this one. This source code can be executed as a script calling ``python yourscript.py`` or loaded as a module using import ``import your_module``. Like was written before, Python does not implements a preprocessor feature at compile time, so cpypp will simulate it at run time. But how can we do it at runtime? How can we modify a code that is compiled and running? The answer is obvious: We can't. We can rewrite the code and executs it again, inside itself. In other words, we will execute the source code twice. The first version is the original version where directives are seen as comments by compiler. This code will be compiled and executed normally. During this execution when the ``PYPP.parse( ...`` code is executed, the trick happens. All code is read again, preprocessed and directives do their job and a brand new code is created. This new code is executed and takes place of original code. There's a little difference if original code is executed as a script or is being imported. When the variable ``__name__`` has the value ``"__main__"``, cpypp assumes this is a script and not an import module. The differences are bellow: How Python and cpypp works when source code is a script ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ First let's see a graphical flow how Python and cpypp will do their job to guarantee that the preprocessor will work properly when running a script. :: Flow | 1️⃣ Python reads original .py script file 2️⃣ compiles it in memory (.pyc) 3️⃣ runs the compiled code | 1️⃣ All code before "PYPP.parse( ..." is executed ✅✅ 2️⃣ When "PYPP.parse( ..." is called | | | 1️⃣ PYPP reads original .py file | 2️⃣ parse directives, clean the code and saves new code in memory | 3️⃣ call exec( ... ) to execute all new code | 4️⃣ call sys.exit(0) to stop old code execution | 3️⃣ All code after "PYPP.parse( ..." from original compiled code is DISCARDED ❌❌ Be atention to step 3.2.4. When the code is executed as a script we can stop the execution of the old code, because we have sure that the new code was all executed stand alone. But we still have a problem. All the code before ``PYPP.parse( ...`` was executed twice. Right ? Yes and No. We have tools to prevent its to happen. Let's see an pratical example to understand better. So the first step is instantiate a preprocessor in our source code: .. code:: python from cpypp import py_preprocessor PYPP = py_preprocessor() ❗️ **YOU CAN NOT** change this piece of code neither change ``PYPP.`` variable. The reason is: When the preprocessor is doing its job, it will look for this piece of code and some references to ``PYPP`` variable so, it can be removed from final code. The second step is write the directives in source code to give work to the preprocessor. Directives are like coments and can be written anywhere in the code. All the directives and its rules will be explained bellow. .. code:: python #ifdef debug print("I'm a debug line running in Python __VERSION__ and won't be present in final code') #endif The third step is call the preprocessor so it can do its job: .. code:: python PYPP.parse(__file__, __name__) ❗️ **DON'T CHANGE** this line either. and finally we have our ``test_script01.py`` with other little stuff as : .. code:: python import sys from cpypp import py_preprocessor PYPP = py_preprocessor() #exclude if len(sys.argv) > 1 and sys.argv[1] == '-d': PYPP.define("debug") #endexclude print("PRINT me always but just once. I'm using Python __VERSION__ !!!") PYPP.parse(__file__, __name__) #ifdef debug print("PRINT me only if '-d' used. I'm using Python __VERSION__ !!!") #endif So, let's run it using python 3.7 :: $ python3 -m cpypp.tests.test_script01 PRINT me always but just once. I'm using Python __VERSION__ !!! PRINT me always but just once. I'm using Python 3.7.6 !!! :: $ python3 -m cpypp.tests.test_script01 -d PRINT me always but just once. I'm using Python __VERSION__ !!! PRINT me always but just once. I'm using Python 3.7.6 !!! PRINT me only if '-d' used. I'm using Python 3.7.6 !!! Well, it didn't work as expected 😩. The first print was executed twice. The reason was explained earlier. All code before ``PYPP.parse( ...`` is executed at original code and at preprocessed code. To solve this we have many options. First let's understand that this happens only to code that starts at column 1 (usually), with exception to classes and functions declarations. All code inside classes and funcions are executed only when called, but all the rest is executed instantly. The options to solve this issue are: #. Move ``PYPP.parse( ...`` next to top of code so, there is no relevant code before it, but only the necessary to its own execution. #. Use the special logical condition ``if PYPP.parsed():`` to all relevant code before ``PYPP.parse( ...``. This will prevent this code to be executed because this condition returns always ``False``, so nothing inside will be executed with original code. The preprocessor recognizes this special logical condition, removes it completely from final code and reident the code to it's original position. With option 1 we have ``test_script02.py``: .. code:: python import sys from cpypp import py_preprocessor #exclude if len(sys.argv) > 1 and sys.argv[1] == '-d': PYPP.define("debug") #endexclude PYPP.parse(__file__, __name__) print("PRINT me always but just once. I'm using Python __VERSION__ !!!") #ifdef debug print("PRINT me only if '-d' used. I'm using Python __VERSION__ !!!") #endif With option 2 we have ``test_script03.py``: .. code:: python import sys from cpypp import py_preprocessor PYPP = py_preprocessor() # This is just a comment #exclude if len(sys.argv) > 1 and sys.argv[1] == '-d': PYPP.define("debug") #endexclude if PYPP.parsed(): print("PRINT me always but just once. I'm using Python __VERSION__ !!!") #ifdef debug print("PRINT me only if '-d' used. I'm using Python __VERSION__ !!!") #endif PYPP.parse(__file__, __name__) So, let's try again ... :: $ python3 -m cpypp.tests.test_script02 PRINT me always but just once. I'm using Python 3.7.6 !!! :: $ python3 -m cpypp.tests.test_script03 -d PRINT me always but just once. I'm using Python 3.7.6 !!! PRINT me only if '-d' used. I'm using Python 3.7.6 !!! Yeah 😁. So the first rule of cpypp is that for scripts, we have to keep our ``PYPP.parse( ...`` call as next from top of code we can, or use ``if PYPP.parsed():`` logical condition, or both, so our code can be processed the way we wants. Another very important rule is: ❗️ Your original source code **HAS TO BE** compilable in all Python versions, because this original source code has to run so the processor can do its job. This means that a code like: .. code:: python #ifdef __PYTHON2__ print "This is", "a debug code ", "and will NOT run in Python 3" #else print("This is a debug code and is executes if we are using Python 3") #endif will not compile in Python 3 because second line will generate a syntax error. In these case if you can't change the original code you can use the directive ``#expand`` as you see bellow. .. code:: python #ifdef __PYTHON2__ #expand print "This is", "a debug code ", "and will NOT run in Python 3" #else print("This is a debug code and is executes if we are using Python 3") #endif How Python and cpypp works when source code is a module to be imported ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Now, let's see a graphical flow how Python and cpypp will do their job to guarantee that the preprocessor will work properly when importing a module :: Flow | 1️⃣ Another Python scripts or module imports our original .py file 2️⃣ Python reads original .py module file 3️⃣ compiles it in memory (.pyc) 4️⃣ loads, runs the compiled code and inserts an entry for it in ``sys.modules`` | 1️⃣ All code before "PYPP.parse( ..." is executed ✅✅ 2️⃣ When "PYPP.parse( ..." is called | | | 1️⃣ PYPP reads original .py file | 2️⃣ parse directives, clean the code and save new code in a file | 3️⃣ call __import__( ... ) to load the new code in the same ``sys.modules`` entry | 4️⃣ new code is executed | 3️⃣ All code after "PYPP.parse( ..." from original compiled code IS EXECUTED TOO ✅✅ There are diferences from a script code. We have ``sys.modules`` that has to be modified at runtime, we can't call ``sys.exit`` because Python will halt and the most important, when the source file is a module to be imported **all the original code will be executed** no matter which place you write ``PYPP.parse( ...``. So we **HAVE** to use ``if PYPP.parsed():`` logical condition to prevent our code to be executed twice and avoid unpredictable runtime errors write ``PYPP.parse(...`` after all references to ``PYPP.`` in source code. Let's see some pratical examples to understand better. You can protect only what is relevant. See ``import4.py`` .. code:: python import os from cpypp import py_preprocessor PYPP = py_preprocessor() class test_class(): # classes and functions are not a problem because are only compiled def __init__(self): print("This print has no problem") GLOBAL_VARIABLE = "PYTHON vr __VERSION__" # global variable definitions are most of time safe to become unprotected if PYPP.parsed(): # but code like this has to be protected at all if os.path.isfile("/tmp/test.txt"): os.remove("/tmp/test.txt") print("module import4 imported and I was printed just once") PYPP.parse(__file__, __name__) Or you can protect the all code. See ``import5.py`` .. code:: python import os from cpypp import py_preprocessor PYPP = py_preprocessor() if PYPP.parsed(): # You can protect the all code class test_class(): # classes and functions are not a problem because are only compiled def __init__(self): print("This print has no problem") GLOBAL_VARIABLE = "PYTHON vr __VERSION__" # global variable definitions are most of time safe to become unprotected if os.path.isfile("/tmp/test.txt"): os.remove("/tmp/test.txt") print("module import5 imported and I was printed just once") PYPP.parse(__file__, __name__) Or you can protect nothing if you code has only global, classes and defs definitions. See ``test06_import_module.py`` .. code:: python from cpypp import py_preprocessor PYPP = py_preprocessor() GLOBAL_VARIABLE = "PYTHON vr __VERSION__" # global variable definitions are most of time safe to become unprotected class test_class(): # classes and functions are not a problem because are only compiled def __init__(self): print("This print has no problem") print("I don't care if this line is printed twice") PYPP.parse(__file__, __name__) Let's import all of then >>> cpypp.tests.imports import impor4 module import4 was imported and I was printed just once >>> cpypp.tests.imports import impor5 module import5 was imported and I was printed just once >>> cpypp.tests.imports import impor6 I don't care if this line is printed twice I don't care if this line is printed twice Yeah again😁. Here the rule is that for import modules, we have to use ``if PYPP.parsed():`` logical condition, in the whole code if necessary, and write ``PYPP.parse (...`` at the end of our source code, so our code can be processed the way we wants. Another thing that is very important: | ❗️ ``if PYPP.parsed():`` logical condition, **CAN ALSO** be replaced by ``if False and False:``. | ❗️ cpypp will understand this too. Command line preprocessor ------------------------- Command line preprocessor is when you have source code files and wants to generate new preprocessed files in batch mode. Probably you are a developer and want to insert copyright marks, or remove something from final code, like ``prints``, debugs and so on, or you know exactly what version of Python will execute your code and wants to generate a final code totally compatible with it. Almost like C does with ``make``. Here we have no flow. It's just a command line tool that has some parameters and you can run it from shell. :: $ python2.7+ -m cpypp -r -d debug cpypp./tests/test_commandline01.pyp ❗️ The code generated is **DIRECTLY DEPENDENT** from the Python version you run the command line Options ^^^^^^^ Usage: ``cpyppc [options] filename-or-directory [...]`` -h, --help show this help message and exit -v, --version print cpypp version -d name same as #define. Ex. ``-d`` debug or ``-d "var=2+2"`` (eval 4) -e EXT include files with only these extensions. default is ``'.py'`` and extensions must be separated with ``'|'`` char. Ex. ``'.py|.pypp'`` -l MAXLEVELS levels to recurse into subdirectories. Use ``'0'`` to don't recurse. Default is no limit -p PATH directory to prepend to file names and paths before save processed files. The full path will be created if it does not exists -f force overwrite of files when output file name has the same name of input file name -r remove meta tags and commented lines from final code -o FILE output file name when you are preprocessing just one file at once. Use ``'-o -'`` to stdout -i FILE add all the files and directories listed in FILE to the list considered for compilation; Use ``'-'`` to read names from stdin -q output only error messages; ``-qq`` will suppress the error messages as well -c, --compileall compile each file after preprocessing. When this option is used, no preprocessed source file will be saved to disk and options ``'-o'``, ``'-r'`` and ``'-f'`` are discarded -b use legacy (pre-PEP3147) compiled file locations. Valid only when ``'-c'`` is used Examples ^^^^^^^^ Let's use one of our script files ``test_script03.py`` .. code:: python import sys from cpypp import py_preprocessor PYPP = py_preprocessor() # This is just a comment #exclude if len(sys.argv) > 1 and sys.argv[1] == '-d': PYPP.define("debug") #endexclude if PYPP.parsed(): print("PRINT me always but just once. I'm using Python __VERSION__ !!!") #ifdef debug print("PRINT me only if '-d' used. I'm using Python __VERSION__ !!!") #endif PYPP.parse(__file__, __name__) Now let's run ``cpypp`` and don't remove metada to see all preprocessor work. :: $ python3 -m cpypp cpypp/tests/test_script03.py -o - .. code:: python import sys # from cpypp import py_preprocessor # PYPP = py_preprocessor() # This is just a comment # #exclude # if len(sys.argv) > 1 and sys.argv[1] == '-d': PYPP.define("debug") # #endexclude # if PYPP.parsed(): print("PRINT me always but just once. I'm using Python 3.7.6 !!!") # #ifdef debug # print("PRINT me only if '-d' used. I'm using Python __VERSION__ !!!") # #endif Realize that any references to ``py_preprocessor`` were commented and the block ``if PYPP.parsed():`` has been commented too and all code bellow was reidented. Let's remove metada data now. :: $ python3 -m cpypp -r cpypp/tests/test_script03.py -o - .. code:: python import sys # This is just a comment print("PRINT me always but just once. I'm using Python 3.7.6 !!!") ❗️ As you can see there is **NO DEPENDENCY** in final code from cpypp. Preprocessing and compiling code at once ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ cpypp can compile bytecode files (.pyc) directly from the preprocessed file without need to save the new file to disk an execute ``python -m compileall``, and is as simple as this. Just use ``-c`` or ``--compileall`` option. :: $ python -m cpypp --compileall diretory_or_file # generate __pycache__/.pyc files $ python -m cpypp -b --compileall diretory_or_file # generate .pyc files at same dir Your bytecodes (.pyc) where generated using the preprocessed file and are ready to deploy. For more information and usage execute ``python cpypp --help``. Expander ======== cpypp implements some expanders that look for names and special characters in the source code and replace then for defined values or expressions. Name definitions expander ------------------------- Any references to name definitions done with ``#define`` in source file will be replaced by its ``repr()`` value. Note than any references means any place where definition names appears, even inside strings. To avoid this replacement precede the name with ``'!'`` For example, the code: .. code:: Python #define SIZE 100 * 2 #define TEXT "cpypp" + " is " + "the best" for i in range(0, SIZE): print(TEXT) print("The value of !TEXT is TEXT") after preprocessing will become: .. code:: Python for i in range(0, 200): print('cpypp is the best') print("The value of TEXT is 'cpypp is the best'") Expressions expander -------------------- For more complexes replacements there is expression expander. All text between expander begin mark ``'#{'`` and expander end mark ``}#'`` will be evaluated and replaced exactly at same place by its ``str()`` value. If you wants that replacement value be by its ``repr()`` value, insert a modifier at end of expression, without spaces from ``'}'``. The modifiers are ``',s`` for ``str()`` value and ``',r'`` for ``repr()`` value. For example, the code: .. code:: Python #! env python#{__VERSION__[0]}# #define VALUE 5 if __PYTHON2__ else 6 #define TEXT "cpypp" + " is " + "the best" print("!VALUE + 1 = #{VALUE + 1}# and #{TEXT}# and #{TEXT,r}#") after preprocessing with ``python2`` will become: .. code:: Python #! env python2 print("VALUE + 1 = 6 and cpypp is the best and 'cpypp is the best'") and after preprocessing with ``python3`` will become: .. code:: Python #! env python3 print("VALUE + 1 = 7 and cpypp is the best and 'cpypp is the best'") Directives ========== Name definitions (#define, #undef) ---------------------------------- Define names, or names with values that will be stored in definition dictionary. ``#define identifier [expression]`` When the preprocessor encounters this directive, it creates an entry in its definition dictionary with name ``identifier`` and the value with evaluation of ``expression``. If ``expression`` is ommited, ``True`` is used. .. code:: Python #define DEBUG #define TABLE_SIZE (50 * 50 if __PYTHON2__ else 100 * 100) block = list(" " * TABLE_SIZE) #ifdef DEBUG print("Debug is ON !!!") #endif To remove an entry from definition dictionary use ``#undef identifier`` Conditional inclusions (#if, #else, #endif and its variations) ----------------------------------------------------------------- These directives allow to include or discard part of the original code, also called code blocks if a certain condition is met or not. It works the same manner that ``if/elif/else`` in python. The diference is that we have ``#endif`` and its variations to close opened ``#if blocks`` cause we don't have identation and there are more directives that gives more flexibility. Here are all the conditional inclusions directives and how they are evalueted. +---------------------------------+-----------------------------------------------------------------------------------+ | **#ifdef** ``identifier`` | | Opens a block of code and includes its content if ``identifier`` is a name | | | | in definition dictionary, no matter its value. | +---------------------------------+-----------------------------------------------------------------------------------+ | **#ifndef** ``identifier`` | | Open a block of code and includes its content if ``identifier`` is not a name | | | | in definition dictionary. | +---------------------------------+-----------------------------------------------------------------------------------+ | **#if** ``expression`` | | Open a block of code and includes it if ``expression`` is evaluated to ``True``.| +---------------------------------+-----------------------------------------------------------------------------------+ | **#else** | | Closes the last opened block of code, opens a new block of code | | | | and includes it if any block above and at same level was not ``True`` | +---------------------------------+-----------------------------------------------------------------------------------+ | **#elif** ``expression`` | | Closes the last opened block of code, opens a new block of code | | | | and includes it if ``expression`` is evaluated to ``True`` and any block | | | | above and at same level was not ``True`` | +---------------------------------+-----------------------------------------------------------------------------------+ |**#elifdef** ``identifier`` | | Closes the last opened block of code, opens a new block of code | | | | and includes it if ``identifier`` is a name in definition dictionary, | | | | no matter its value and any block above and at same level was not ``True``. | +---------------------------------+-----------------------------------------------------------------------------------+ | **#endif** | | Closes the last opened block of code at same level | +---------------------------------+-----------------------------------------------------------------------------------+ | **#endififdef** ``identifier`` | | Same as **#endif** + **#ifdef** but at the same line | +---------------------------------+-----------------------------------------------------------------------------------+ | **#endifif** ``expression`` | | Same as **#endif** + **#if** but at the same line | +---------------------------------+-----------------------------------------------------------------------------------+ | **#endifall** | | Close all opened blocks no matter if they are inner or outter. Use with care. | +---------------------------------+-----------------------------------------------------------------------------------+ Examples of Conditional inclusions ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ .. code:: Python # compile block if DEBUG was defined before #ifdef DEBUG print("Debug is ON !!!") #endif # compile block if DEBUG was not defined or was excluded from definition dictionary #undef DEBUG #ifndef DEBUG print("Debug is ON !!!") #endif # compile block if expression is True # define INT_VER int(VERSION[0]) #if DEBUG is True and INT_VER > 2 print("Debug is ON and Python is 3+!!!") #elif INT_VER == 2 print("Python certainly is 2.x") #else print("I'm sure Debug is ON or OFF :)") #endif Exclusion blocks (#exclude and #endexclude) ----------------------------------------------------------------- Depending of your code, sometimes you want to exclude an entire block of code if some conditions are met or not or not, or maybe you wants that only the preprocessor executes this portion of code, but wants remove this portion from final code. For this we have the directives **#exclude** and **#endexclude**. All inside this two directives will be completely exclude from final code, but will be seen by preprocessor. .. code:: Python #define NUMPY_E __import__("numpy").e #define NUMPY_EULER_GAMA __import__("numpy").euler_gama numpy_e = NUMPY_E numpy_euler_gama = NUMPY_EULER_GAMA We imported numpy package only at preprocessor time to get the values of these two constants and set two local variables. ``numpy`` package don't need to be installed where the final code will execute. Neither cpypp as we already know. Ignore blocks (#ignore and #endignore) ----------------------------------------------------------------- These directives do exactly what their name says. They ignore from the preprocessor an entire block of code. .. code:: Python #ignore #define NONE "This defines will never occurs because this block is ignored" #ifndef NONE print("This !TEXT will be printed exactly how it is #{PRINT_ME#}") # Will print always #endif #endignore Source file inclusion (#include and #includeident) ----------------------------------------------------------------- When the preprocessor finds an ``#include`` or ``#includeident`` directive it replaces it by the entire content of the specified file. The diference between ``#include`` and ``#includeident`` is that ``#include`` will always include each line of included file starting of column 1 and ``#includeident`` will always include each line of incuded file starting at same column where ``#includeident`` was written. There are two ways to use #include: +-----------------------------+------------------------------------------------------------------------------------+ | **#include** ``<filename>`` | | When filename is specified between angle-backets, cpypp looks for the | | | | filename in all directories listed in ``sys.path``. The first existing file | | | | will be included. | +-----------------------------+------------------------------------------------------------------------------------+ | **#include** ``expression`` | | When filename is not specified between angle-brackes, cpypp assumes that this | | | | is an expression, evaluates it and the result as used as absolute path of file. | | | | If the file exists it will be included. | +-----------------------------+------------------------------------------------------------------------------------+ Supose that ``include.py`` has the following content. :: print("I'm an included file") Now let's see this code. .. code:: Python #include "include.py" #include "include.py" if some_condition: #includeident "include.py" If we check the preprocessed file we will find: .. code:: Python print("I'm an included file") print("I'm an included file") if some_condition: print("I'm an included file") Code protection (#expand) ------------------------- As you remember, the original code is compiled twice, one before the preprocessor and other after. So at both compiling steps the code must be correct and without syntax or grammar errors. Sometimes we want to implement some features in our code that are not available in all Python versions, but we need that our code compile correctly so the preprocessor can do its work. For this we have ``#expand`` directive. If you had a piece of code that is not compatible with all versions of Python you put it as parameter of ``#expand``. The first step of compiling will consider this line as a comment and will ignore it. And the preprocessor will do its job. For example: .. code:: Python #if __PYTHON2__ print "This","is", "a","python","program" #else print ("This is a python program") #endif The code above will generate a compile error if we try to run it on Python 3+, because the first ``print`` statement does not meet the Python 3 syntax. So the solution is to hide this code from first compiling. ``#expand`` will do this for us. .. code:: Python #if __PYTHON2__ #expand print "This","is", "a","python","program" #else print ("This is a python program") #endif Now the first compiling will occur with no problem no matter what Python version is used and the preprocessor will go on. Contributing ============ Please send an email to `wellrats@gmail.com <mailto:wellrats@gmail.com>`_


نحوه نصب


نصب پکیج whl cpypp-1.1.0:

    pip install cpypp-1.1.0.whl


نصب پکیج tar.gz cpypp-1.1.0:

    pip install cpypp-1.1.0.tar.gz