![SOURCEdefender - advanced encryption protecting your Python codebase](https://images.sourcedefender.co.uk/logo.png "SOURCEdefender - advanced encryption protecting your python3 codebase")
- - -
[![python](https://shields.io/pypi/pyversions/sourcedefender)][python-url]
[![downloads](https://pepy.tech/badge/sourcedefender)][pepy-url]
[![downloads](https://pepy.tech/badge/sourcedefender/week)][pepy-url]
SOURCEdefender is the easiest way to obfuscate Python code using AES-256 encryption. AES is a symmetric algorithm which uses the same key for both encryption and decryption (the security of an AES system increases exponentially with key length). There is no impact on the performance of your running application as the decryption process takes place during the import of your module, so encrypted code won't run any slower once loaded from a _.pye_ file compared to loading from a _.py_ or _.pyc_ file.
# Features
- No end-user device license required
- Symmetric AES 256-bit encryption
- Set your own password & salt for encryption
- Enforced an expiration time on encrypted code
- Bundle encrypted files or folders into a single executable binary using PyInstaller
## Supported Environments
We support the following Operating System and architecture combinations and hook directly into the import process, so there are no cross-platform compatibility issues. Encrypted code will run on ___ANY___ other target using the same version of Python. For example, files encrypted in Windows using Python 3.10 will run with Python 3.10 on Linux.
| CPU Architecture | Operating System | Python Architecture | Python Versions |
| ---------------- | ---------------- | ------------------- | --------------- |
| AMD64 | Windows | 64-bit | 3.9 - 3.13 |
| x86_64 | Linux | 64-bit | 3.9 - 3.13 |
| x86_64 | macOS | 64-bit | 3.9 - 3.13 |
| ARM64 | macOS | 64-bit | 3.9 - 3.13 |
| AARCH64 | Linux | 64-bit | 3.9 - 3.13 |
###### If you do not see your required combination here, please [contact][sourcedefender-hello-email] us so we can help find you a solution
# Trial License
The installation of SOURCEdefender will grant you a trial license to encrypt files. This trial license will only allow your script to work for a maximum of 24 hours; after that, it won't be usable. This is so you can test whether our solution is suitable for your needs. If you get stuck, then please [contact][sourcedefender-hello-email] us so we can help.
# Subscribe
To distribute encrypt code without limitation, you will need to create an [account][sourcedefender-dashboard] and set up your payment method. Once you have set up the account, you will be able to retrieve your activation token and use it to authorise your installation:
$ sourcedefender activate --token 470a7f2e76ac11eb94390242ac130002
SOURCEdefender
Registration:
- Account Status : Active
- Email Address : hello@sourcedefender.co.uk
- Account ID : bfa41ccd-9738-33c0-83e9-cfa649c05288
- System ID : 42343554645384
- Valid Until : Sun, May 9, 2024 10:59 PM
Without activating your SDK, any encrypted code you create will only be usable for a maximum of __24hrs__. Access to our dashboard (via HTTPS) from your system is required so we can validate your account status.
If you want to view your activated license status, you can use the __validate__ option:
$ sourcedefender validate
SOURCEdefender
Registration:
- Account Status : Active
- Email Address : hello@sourcedefender.co.uk
- Account ID : bfa41ccd-9738-33c0-83e9-cfa649c05288
- System ID : 42343554645384
- Valid Until : Sun, May 9, 2024 10:59 PM
$
If your license is valid, this command will give the Exit Code (EC) of #0 (zero); otherwise, an invalid licence will be indicated by the EC of #1 (one). You should run this command after any automated build tasks to ensure you haven't created code with an unexpected 24-hour limitation.
## Price Plans
Our price plans are detailed on our [Dashboard][sourcedefender-dashboard]. If you do not see a price you like, please [email][sourcedefender-hello-email] us so we can discuss your situation and requirements.
# Usage
We have worked hard to ensure that the encryption/decryption process is as simple as possible. Here are a few examples of how it works and how to use the features provided. If you need advice on how to encrypt or import your code, please [contact][sourcedefender-hello-email] us for assistance.
### How do I protect my Python source code?
First, let's have a look at an example of the encryption process:
$ cat /home/ubuntu/helloworld.py
print("Hello World!")
$
This is a very basic example, but we do not want anyone to get at our source code. We also don't want anyone to run this code after 1 hour so when we encrypt the file we can enforce an expiration time of 1 hour from now with the __--ttl__ option, and we can delete the plaintext .py file after encryption by adding the __--remove__ option.
The command would look like this:
$ sourcedefender encrypt --remove --ttl=1h /home/ubuntu/helloworld.py
SOURCEdefender
Processing:
/home/ubuntu/helloworld.py
$
The TTL argument offers the following options: weeks(w), days(d), hours(h), minutes(m), and seconds(s).
Usage is for example: --ttl=10s, or --ttl=24m, or --ttl=1m, or just --ttl=3600. This can't be changed after encryption.
The '--remove' option deletes the original .py file. Make sure you use this so you don't accidentally distribute the plain-text code. Now the file is encrypted, its contents are as follows:
$ cat /home/ubuntu/helloworld.pye
-----BEGIN SOURCEDEFENDER FILE-----
GhP6+FOEA;qsm6NrRnXHnlU5E!(pT(E<#t=
GhN0L!7UrbN"Am#(8iPPAG;nm-_4d!F9"*7
T1q4VZdj>uLBghNY)[;Ber^L=*a-I[MA.-4
------END SOURCEDEFENDER FILE------
$
Once a file has been encrypted, its new extension is __.pye__ so our loader can identify encrypted files. All you need to remember is to include __sourcedefender__ as a Python dependency while packaging your project and import the sourcedefender module before you attempt to import and use your encrypted code.
### Importing packages & modules
The usual import system can still be used, and you can import encrypted code from within encrypted code, so you don't need to do anything special with your import statements.
$ cd /home/ubuntu
$ ls
helloworld.pye
$ python3
>>>
>>> import sourcedefender
>>> import helloworld
Hello World!
>>> exit()
$
### Using your own password or salt for encryption
It's easy to use your own encryption password and salt. If you do not set these, we generate unique ones for each file you encrypt. Should you wish to set your own, these can be set from either
a command option:
sourcedefender encrypt --password 1234abcd --salt dcba4321 mycode.py
or as an Environment variable:
export SOURCEDEFENDER_PASSWORD="1234abcd"
export SOURCEDEFENDER_SALT="dcba4321"
sourcedefender encrypt mycode.py
To import the code, you can either set an environment variable (as with the encryption process). You can also set these in your code before the import:
$ python3
>>> import sourcedefender
>>> from os import environ
>>> environ["SOURCEDEFENDER_PASSWORD"] = "1234abcd"
>>> environ["SOURCEDEFENDER_SALT"] = "dcba4321"
>>> import mycode
The password and salt set are specific to the next import, so if you want different ones for different files, feel free to encrypt with different values. Remember to set 'sourcedefender.password/salt=something' before your import.
### Can I still run Python from the command line?
Yes, you can still run scripts from the command line, but there are some differences due to the way Python loads command-line scripts. For example, you need to ask Python to load the sourcedefender package and then tell it what to import:
$ python3 -m sourcedefender /home/ubuntu/helloworld.pye
Hello World!
$
However, due to the way Python works - and the fact that we need to run the 'sourcedefender' module first - you won't be able to compare \_\_name\_\_ == "\_\_main\_\_" in the code to see if it is being imported or executed as a script. This means that the usual starting code block will not get executed.
### Dynamic Downloads
You can download individual .pye files from a URI at runtime. As an example, we have encrypted the following script and made it publicly available:
$ cat hello.py
def message():
print("hi!")
$
To download the file from the Internet, you can use the following code example:
$ cat download.py
from sourcedefender.tools import getUrl
getUrl("https://downloads.sourcedefender.co.uk/hello.pye")
from hello import message
message()
$
> We can only download a single file at a time. Python packages or zip files are not supported.
When executed, this will do the following:
$ python3 download.py
hi!
$
We know this is a simple example, and security can be increased by using your own salt/password.
### Integrating encrypted code with PyInstaller
PyInstaller scans your plain-text code for import statements so it knows what packages to freeze. This scanning is not possible inside encrypted code, so we have created a 'pack' command to help. However, you will need to ask PyInstaller to include any hidden libs by using the '--hidden-import' or '--add-binary' options.
We are unable to guess what parts of your code you want to encrypt. If you encrypt all your code, sometimes that stops Python from working. So, with that in mind, please ensure you encrypt your code before using the pack command.
For this example, we have the following project structure:
pyexe.py
lib
└── helloworld.pye
In our pyexe script, we have the following code:
$ cat pyexe.py
import helloworld
To ensure that PyInstaller includes our encrypted files, we need to tell it where they are with the --add-binary option. So, for the above project, we could use this command:
sourcedefender encrypt pyexe.py --remove
sourcedefender pack pyexe.pye -- --add-binary $(pwd)/lib:.
There is a strange quirk with PyInstaller that we haven't yet found a workaround for. When you include extra args after '--', you need to provide full paths of the source folders otherwise, you will get a tmp folder not found error such as this:
Unable to find "/tmp/tmpp9pt6l97/lib" when adding binary and data files.
### Integrating encrypted code with Django
You can encrypt your Django project just the same as you can any other Python code. Don't forget to include "import sourcedefender" in the __init__.py file that is in the same directory as your settings.py file. Only obfuscate your own code and not code generated by the Django commands. There is no point in protecting files such as urls.py as these should not contain much/any of your own code other than things that have been imported.
### requirements.txt
Because we only keep the last available version of a branch online, you can lock your version to a branch by including this in your requirements.txt file:
sourcedefneder~=14.0
This will install the latest release >= 14.0.0 but less than 15.0.0, so major branch updates will need to be completed manually.
We always endeavour to keep the latest release of a branch on PyPi, so
# Legal
THE SOFTWARE IS PROVIDED "AS IS," AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. REVERSE ENGINEERING IS STRICTLY PROHIBITED.
##### __Copyright © 2018-2024 SOURCEdefender. All rights reserved.__
<!-- URLs -->
[python-url]: https://www.python.org
[pepy-url]: https://pepy.tech/project/sourcedefender
[pypi-url]: https://pypi.org/project/sourcedefender
[snyk-advisor-url]: https://snyk.io/advisor/python/sourcedefender#security
[sourcedefender-hello-email]: mailto:hello@sourcedefender.co.uk
[sourcedefender-dashboard]: https://dashboard.sourcedefender.co.uk/signup?src=pypi-readme
[flux-audio]: https://www.flux.audio/
[corsight]: https://www.corsight.ai/
Raw data
{
"_id": null,
"home_page": "https://sourcedefender.co.uk/?src=pypi-url",
"name": "sourcedefender",
"maintainer": null,
"docs_url": null,
"requires_python": "!=2.*,>=3.9",
"maintainer_email": null,
"keywords": "encryption source aes",
"author": "SOURCEdefender",
"author_email": "hello@sourcedefender.co.uk",
"download_url": "https://files.pythonhosted.org/packages/0f/a9/637b9bd6da4cded35b2f3dceb5982ed37b40f19715b29a6156be57abac39/sourcedefender-15.0.1.tar.gz",
"platform": null,
"description": "![SOURCEdefender - advanced encryption protecting your Python codebase](https://images.sourcedefender.co.uk/logo.png \"SOURCEdefender - advanced encryption protecting your python3 codebase\")\n- - -\n[![python](https://shields.io/pypi/pyversions/sourcedefender)][python-url]\n[![downloads](https://pepy.tech/badge/sourcedefender)][pepy-url]\n[![downloads](https://pepy.tech/badge/sourcedefender/week)][pepy-url]\n\nSOURCEdefender is the easiest way to obfuscate Python code using AES-256 encryption. AES is a symmetric algorithm which uses the same key for both encryption and decryption (the security of an AES system increases exponentially with key length). There is no impact on the performance of your running application as the decryption process takes place during the import of your module, so encrypted code won't run any slower once loaded from a _.pye_ file compared to loading from a _.py_ or _.pyc_ file.\n\n# Features\n\n- No end-user device license required\n- Symmetric AES 256-bit encryption\n- Set your own password & salt for encryption\n- Enforced an expiration time on encrypted code\n- Bundle encrypted files or folders into a single executable binary using PyInstaller\n\n## Supported Environments\n\nWe support the following Operating System and architecture combinations and hook directly into the import process, so there are no cross-platform compatibility issues. Encrypted code will run on ___ANY___ other target using the same version of Python. For example, files encrypted in Windows using Python 3.10 will run with Python 3.10 on Linux.\n\n| CPU Architecture | Operating System | Python Architecture | Python Versions |\n| ---------------- | ---------------- | ------------------- | --------------- |\n| AMD64 | Windows | 64-bit | 3.9 - 3.13 |\n| x86_64 | Linux | 64-bit | 3.9 - 3.13 |\n| x86_64 | macOS | 64-bit | 3.9 - 3.13 |\n| ARM64 | macOS | 64-bit | 3.9 - 3.13 |\n| AARCH64 | Linux | 64-bit | 3.9 - 3.13 |\n\n###### If you do not see your required combination here, please [contact][sourcedefender-hello-email] us so we can help find you a solution\n\n# Trial License\n\nThe installation of SOURCEdefender will grant you a trial license to encrypt files. This trial license will only allow your script to work for a maximum of 24 hours; after that, it won't be usable. This is so you can test whether our solution is suitable for your needs. If you get stuck, then please [contact][sourcedefender-hello-email] us so we can help.\n\n# Subscribe\n\nTo distribute encrypt code without limitation, you will need to create an [account][sourcedefender-dashboard] and set up your payment method. Once you have set up the account, you will be able to retrieve your activation token and use it to authorise your installation:\n\n $ sourcedefender activate --token 470a7f2e76ac11eb94390242ac130002\n SOURCEdefender\n\n Registration:\n\n - Account Status : Active\n - Email Address : hello@sourcedefender.co.uk\n - Account ID : bfa41ccd-9738-33c0-83e9-cfa649c05288\n - System ID : 42343554645384\n - Valid Until : Sun, May 9, 2024 10:59 PM\n\nWithout activating your SDK, any encrypted code you create will only be usable for a maximum of __24hrs__. Access to our dashboard (via HTTPS) from your system is required so we can validate your account status.\n\nIf you want to view your activated license status, you can use the __validate__ option:\n\n $ sourcedefender validate\n SOURCEdefender\n\n Registration:\n\n - Account Status : Active\n - Email Address : hello@sourcedefender.co.uk\n - Account ID : bfa41ccd-9738-33c0-83e9-cfa649c05288\n - System ID : 42343554645384\n - Valid Until : Sun, May 9, 2024 10:59 PM\n $\n\nIf your license is valid, this command will give the Exit Code (EC) of #0 (zero); otherwise, an invalid licence will be indicated by the EC of #1 (one). You should run this command after any automated build tasks to ensure you haven't created code with an unexpected 24-hour limitation.\n\n## Price Plans\n\nOur price plans are detailed on our [Dashboard][sourcedefender-dashboard]. If you do not see a price you like, please [email][sourcedefender-hello-email] us so we can discuss your situation and requirements.\n\n# Usage\n\nWe have worked hard to ensure that the encryption/decryption process is as simple as possible. Here are a few examples of how it works and how to use the features provided. If you need advice on how to encrypt or import your code, please [contact][sourcedefender-hello-email] us for assistance.\n\n### How do I protect my Python source code?\n\nFirst, let's have a look at an example of the encryption process:\n\n $ cat /home/ubuntu/helloworld.py\n print(\"Hello World!\")\n $\n\nThis is a very basic example, but we do not want anyone to get at our source code. We also don't want anyone to run this code after 1 hour so when we encrypt the file we can enforce an expiration time of 1 hour from now with the __--ttl__ option, and we can delete the plaintext .py file after encryption by adding the __--remove__ option.\n\nThe command would look like this:\n\n $ sourcedefender encrypt --remove --ttl=1h /home/ubuntu/helloworld.py\n SOURCEdefender\n\n Processing:\n\n /home/ubuntu/helloworld.py\n\n $\n\nThe TTL argument offers the following options: weeks(w), days(d), hours(h), minutes(m), and seconds(s).\nUsage is for example: --ttl=10s, or --ttl=24m, or --ttl=1m, or just --ttl=3600. This can't be changed after encryption.\n\nThe '--remove' option deletes the original .py file. Make sure you use this so you don't accidentally distribute the plain-text code. Now the file is encrypted, its contents are as follows:\n\n $ cat /home/ubuntu/helloworld.pye\n -----BEGIN SOURCEDEFENDER FILE-----\n GhP6+FOEA;qsm6NrRnXHnlU5E!(pT(E<#t=\n GhN0L!7UrbN\"Am#(8iPPAG;nm-_4d!F9\"*7\n T1q4VZdj>uLBghNY)[;Ber^L=*a-I[MA.-4\n ------END SOURCEDEFENDER FILE------\n $\n\nOnce a file has been encrypted, its new extension is __.pye__ so our loader can identify encrypted files. All you need to remember is to include __sourcedefender__ as a Python dependency while packaging your project and import the sourcedefender module before you attempt to import and use your encrypted code.\n\n### Importing packages & modules\n\nThe usual import system can still be used, and you can import encrypted code from within encrypted code, so you don't need to do anything special with your import statements.\n\n $ cd /home/ubuntu\n $ ls\n helloworld.pye\n $ python3\n >>>\n >>> import sourcedefender\n >>> import helloworld\n Hello World!\n >>> exit()\n $\n\n### Using your own password or salt for encryption\n\nIt's easy to use your own encryption password and salt. If you do not set these, we generate unique ones for each file you encrypt. Should you wish to set your own, these can be set from either\na command option:\n\n sourcedefender encrypt --password 1234abcd --salt dcba4321 mycode.py\n\nor as an Environment variable:\n\n export SOURCEDEFENDER_PASSWORD=\"1234abcd\"\n export SOURCEDEFENDER_SALT=\"dcba4321\"\n sourcedefender encrypt mycode.py\n\nTo import the code, you can either set an environment variable (as with the encryption process). You can also set these in your code before the import:\n\n $ python3\n >>> import sourcedefender\n >>> from os import environ\n >>> environ[\"SOURCEDEFENDER_PASSWORD\"] = \"1234abcd\"\n >>> environ[\"SOURCEDEFENDER_SALT\"] = \"dcba4321\"\n >>> import mycode\n\nThe password and salt set are specific to the next import, so if you want different ones for different files, feel free to encrypt with different values. Remember to set 'sourcedefender.password/salt=something' before your import.\n\n### Can I still run Python from the command line?\n\nYes, you can still run scripts from the command line, but there are some differences due to the way Python loads command-line scripts. For example, you need to ask Python to load the sourcedefender package and then tell it what to import:\n\n $ python3 -m sourcedefender /home/ubuntu/helloworld.pye\n Hello World!\n $\n\nHowever, due to the way Python works - and the fact that we need to run the 'sourcedefender' module first - you won't be able to compare \\_\\_name\\_\\_ == \"\\_\\_main\\_\\_\" in the code to see if it is being imported or executed as a script. This means that the usual starting code block will not get executed.\n\n### Dynamic Downloads\n\nYou can download individual .pye files from a URI at runtime. As an example, we have encrypted the following script and made it publicly available:\n\n $ cat hello.py\n def message():\n print(\"hi!\")\n $\n\nTo download the file from the Internet, you can use the following code example:\n\n $ cat download.py\n from sourcedefender.tools import getUrl\n getUrl(\"https://downloads.sourcedefender.co.uk/hello.pye\")\n from hello import message\n message()\n $\n\n> We can only download a single file at a time. Python packages or zip files are not supported.\n\nWhen executed, this will do the following:\n\n $ python3 download.py\n hi!\n $\n\nWe know this is a simple example, and security can be increased by using your own salt/password.\n\n### Integrating encrypted code with PyInstaller\n\nPyInstaller scans your plain-text code for import statements so it knows what packages to freeze. This scanning is not possible inside encrypted code, so we have created a 'pack' command to help. However, you will need to ask PyInstaller to include any hidden libs by using the '--hidden-import' or '--add-binary' options.\n\nWe are unable to guess what parts of your code you want to encrypt. If you encrypt all your code, sometimes that stops Python from working. So, with that in mind, please ensure you encrypt your code before using the pack command.\n\nFor this example, we have the following project structure:\n\n pyexe.py\n lib\n \u2514\u2500\u2500 helloworld.pye\n\nIn our pyexe script, we have the following code:\n\n $ cat pyexe.py\n import helloworld\n\nTo ensure that PyInstaller includes our encrypted files, we need to tell it where they are with the --add-binary option. So, for the above project, we could use this command:\n\n sourcedefender encrypt pyexe.py --remove\n sourcedefender pack pyexe.pye -- --add-binary $(pwd)/lib:.\n\nThere is a strange quirk with PyInstaller that we haven't yet found a workaround for. When you include extra args after '--', you need to provide full paths of the source folders otherwise, you will get a tmp folder not found error such as this:\n\n Unable to find \"/tmp/tmpp9pt6l97/lib\" when adding binary and data files.\n\n### Integrating encrypted code with Django\n\nYou can encrypt your Django project just the same as you can any other Python code. Don't forget to include \"import sourcedefender\" in the __init__.py file that is in the same directory as your settings.py file. Only obfuscate your own code and not code generated by the Django commands. There is no point in protecting files such as urls.py as these should not contain much/any of your own code other than things that have been imported.\n\n### requirements.txt\n\nBecause we only keep the last available version of a branch online, you can lock your version to a branch by including this in your requirements.txt file:\n\n sourcedefneder~=14.0\n\nThis will install the latest release >= 14.0.0 but less than 15.0.0, so major branch updates will need to be completed manually.\n\n\nWe always endeavour to keep the latest release of a branch on PyPi, so\n# Legal\n\nTHE SOFTWARE IS PROVIDED \"AS IS,\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. REVERSE ENGINEERING IS STRICTLY PROHIBITED.\n\n##### __Copyright \u00a9 2018-2024 SOURCEdefender. All rights reserved.__\n\n<!-- URLs -->\n[python-url]: https://www.python.org\n[pepy-url]: https://pepy.tech/project/sourcedefender\n[pypi-url]: https://pypi.org/project/sourcedefender\n[snyk-advisor-url]: https://snyk.io/advisor/python/sourcedefender#security\n[sourcedefender-hello-email]: mailto:hello@sourcedefender.co.uk\n[sourcedefender-dashboard]: https://dashboard.sourcedefender.co.uk/signup?src=pypi-readme\n[flux-audio]: https://www.flux.audio/\n[corsight]: https://www.corsight.ai/\n\n",
"bugtrack_url": null,
"license": "Other/Proprietary License",
"summary": "Advanced encryption protecting your python codebase.",
"version": "15.0.1",
"project_urls": {
"Dashboard": "https://dashboard.sourcedefender.co.uk/login?src=pypi-navbar",
"Homepage": "https://sourcedefender.co.uk/?src=pypi-url"
},
"split_keywords": [
"encryption",
"source",
"aes"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "0fa9637b9bd6da4cded35b2f3dceb5982ed37b40f19715b29a6156be57abac39",
"md5": "768bb92f6e43eeed04350e1b1ca6b59c",
"sha256": "2b9e45167635a51a29ea9263863ec2560d2c7cae8059a3d94c97a4c996df1221"
},
"downloads": -1,
"filename": "sourcedefender-15.0.1.tar.gz",
"has_sig": false,
"md5_digest": "768bb92f6e43eeed04350e1b1ca6b59c",
"packagetype": "sdist",
"python_version": "source",
"requires_python": "!=2.*,>=3.9",
"size": 11352166,
"upload_time": "2024-12-10T23:21:03",
"upload_time_iso_8601": "2024-12-10T23:21:03.332186Z",
"url": "https://files.pythonhosted.org/packages/0f/a9/637b9bd6da4cded35b2f3dceb5982ed37b40f19715b29a6156be57abac39/sourcedefender-15.0.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-12-10 23:21:03",
"github": false,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"lcname": "sourcedefender"
}