py-eureka-client


Namepy-eureka-client JSON
Version 0.11.13 PyPI version JSON
download
home_pageNone
SummaryA eureka client written in python. Support registering your python component to Eureka Server, as well as calling remote services by pulling the the Eureka registry.
upload_time2024-11-11 03:16:22
maintainerNone
docs_urlNone
authorNone
requires_python>=3.7
licenseMIT License Copyright (c) 2018 Keijack Wu Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
keywords eureka netflix microservice
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # python-eureka-client

[![PyPI version](https://badge.fury.io/py/py-eureka-client.png)](https://badge.fury.io/py/py-eureka-client)

## Discription

This is an eureka client written in python, you can easily intergrate your python components with spring cloud.

## Support Python Version

Python 3.7+

*From `0.9.0`, python 2 is no longer supported, if you are using python 2, please use  version `0.8.12`.*

## Why choose

* Register your python components to eureka server.
* Support failover.
* Support DNS discovery. 
* Send heartbeat to eureka server.
* Pull registry from eureka server.
* Easy to use interface to use other REST service.
* HA when calling other REST service.
* Both trandictional and async def interfaces are provided.
* The http client lib is replacable.

## How to use

### Install

```Shell
pip install py_eureka_client
```

### Getting Start

This is the easiest way to use this component.

```python
import py_eureka_client.eureka_client as eureka_client

your_rest_server_port = 9090
# The flowing code will register your server to eureka server and also start to send heartbeat every 30 seconds
eureka_client.init(eureka_server="http://your-eureka-server-peer1,http://your-eureka-server-peer2",
                   app_name="your_app_name",
                   instance_port=your_rest_server_port)
```

Then, in your business code, use

```python
import py_eureka_client.eureka_client as eureka_client

res = eureka_client.do_service("OTHER-SERVICE-NAME", "/service/context/path")
print("result of other service" + res)

```

You can also use the `EurekaClient` class. 

```python
from py_eureka_client.eureka_client import EurekaClient
client = EurekaClient(eureka_server="http://my_eureka_server_peer_1/eureka/v2,http://my_eureka_server_peer_2/eureka/v2", app_name="python_module_1", instance_port=9090)
await client.start()
res = await client.do_service("OTHER-SERVICE-NAME", "/service/context/path")
print("result of other service" + res)
# when server is shutted down:
await client.stop()
```

In fact, the `init` function is a facade of the EurekaClient, it holds a client object behind, you can get that by catching its return value or use `eureka_client.get_client()` to get it. The `init` function will automatically start the client, while using raw `EurekaClient`, you must call the `start()` and `stop()` method explicitly. 


From `0.11.0`, all the methods in `EurekaClient` are defined `async`, and there are also async facade for `init`, `do_servise`, `stop` functions names `init_async`, `do_service_async`, `sto_async`.

### Registering to Eureka Server

The most common method to will be like:

```Python
import py_eureka_client.eureka_client as eureka_client

your_rest_server_port = 9090
eureka_client.init(eureka_server="http://your-eureka-server-peer1,http://your-eureka-server-peer2",
                                app_name="python_module_1",
                                instance_port=your_rest_server_port)
```

But if you have deploy your eureka server in several zones, you should specify the `eureka_availability_zones` parameter.

```python
import py_eureka_client.eureka_client as eureka_client
eureka_client.init(eureka_availability_zones={
                "us-east-1c": "http://ec2-552-627-568-165.compute-1.amazonaws.com:7001/eureka/v2/,http://ec2-368-101-182-134.compute-1.amazonaws.com:7001/eureka/v2/",
                "us-east-1d": "http://ec2-552-627-568-170.compute-1.amazonaws.com:7001/eureka/v2/",
                "us-east-1e": "http://ec2-500-179-285-592.compute-1.amazonaws.com:7001/eureka/v2/"}, 
                zone="us-east-1c",
                app_name="python_module_1", 
                instance_port=9090,
                data_center_name="Amazon")
```

If you are looking for flexibility, you should configure Eureka service URLs using DNS.

For instance, following is a DNS TXT record created in the DNS server that lists the set of available DNS names for a zone.

```
txt.us-east-1.mydomaintest.netflix.net="us-east-1c.mydomaintest.netflix.net" "us-east-1d.mydomaintest.netflix.net" "us-east-1e.mydomaintest.netflix.net"
```

Then, you can define TXT records recursively for each zone similar to the following (if more than one hostname per zone, space delimit)

```
txt.us-east-1c.mydomaintest.netflix.net="ec2-552-627-568-165.compute-1.amazonaws.com" "ec2-368-101-182-134.compute-1.amazonaws.com"
txt.us-east-1d.mydomaintest.netflix.net="ec2-552-627-568-170.compute-1.amazonaws.com"
txt.us-east-1e.mydomaintest.netflix.net="ec2-500-179-285-592.compute-1.amazonaws.com"
```

And then you can create the client like: 

```python
import py_eureka_client.eureka_client as eureka_client
eureka_client.init(eureka_domain="mydomaintest.netflix.net",
                region="us-east-1",
                zone="us-east-1c",
                app_name="python_module_1", 
                instance_port=9090,
                data_center_name="Amazon")
```

You can specify the protocol, basic authentication and context path of your eureka server separatly rather than setting it at the URL. 

```python
import py_eureka_client.eureka_client as eureka_client
eureka_client.init(eureka_domain="mydomaintest.netflix.net",
                region="us-east-1",
                zone="us-east-1c",
                eureka_protocol="https",
                eureka_basic_auth_user="keijack",
                eureka_basic_auth_password="kjauthpass",
                eureka_context="/eureka/v2",
                app_name="python_module_1", 
                instance_port=9090,
```

or

```python
import py_eureka_client.eureka_client as eureka_client
eureka_client.init(eureka_server="your-eureka-server-peer1,your-eureka-server-peer2",
                eureka_protocol="https",
                eureka_basic_auth_user="keijack",
                eureka_basic_auth_password="kjauthpass",
                eureka_context="/eureka/v2",
                app_name="python_module_1", 
                instance_port=9090)
```

*About the instance `IP` and `hostname`*:

If you are using a `Amazon` data center, `py-eureka-client` will try to use `local-ipv4` and `local-hostname` get from Amazon metadata service. In other cases, `py-eureka-client` will use the first non-loopback ip address and hostname from your net interface. 

You can also specify both these tow field or just one of them explicitly:

```python
eureka_client.init(eureka_server="your-eureka-server-peer1,your-eureka-server-peer2",
                eureka_protocol="https",
                eureka_basic_auth_user="keijack",
                eureka_basic_auth_password="kjauthpass",
                eureka_context="/eureka/v2",
                app_name="python_module_1", 
                instance_ip="192.168.10.168",
                instance_host="my-py-component.mydomian.com",
                instance_port=9090)
```

In some case you might have more than one interfaces attached, for example, you are running your application in a docker-container. In this case you can specify a network via `instance_ip_network` to be used to get the container's ip and host. You can use:

```python
eureka_client.init(eureka_server="your-eureka-server-peer1,your-eureka-server-peer2",
                eureka_protocol="https",
                eureka_basic_auth_user="keijack",
                eureka_basic_auth_password="kjauthpass",
                eureka_context="/eureka/v2",
                app_name="python_module_1", 
                instance_ip_network="192.168.10.0/24",
                instance_port=9090)
```

If you want to get the ip only and sepecify the host by yourself, try:

```python
import py_eureka_client.netint_utils as netint_utils

# you can get the ip only
ip = netint_utils.get_first_non_loopback_ip("192.168.10.0/24")
host = "my-py-component.mydomian.com"

eureka_client.init(eureka_server="your-eureka-server-peer1,your-eureka-server-peer2",
                eureka_protocol="https",
                eureka_basic_auth_user="keijack",
                eureka_basic_auth_password="kjauthpass",
                eureka_context="/eureka/v2",
                app_name="python_module_1", 
                instance_ip=ip,
                instance_host=host,
                instance_port=9090)
```

### Error Callback

You can specify a callback function when initializing the eureka client, when errors occur in `register`, `discover` or `status update` phase, the callback function will be called to inform you. The callback function will be called only when all the eureka server url are all tried and fails. 

The callback function should accept 2 arguments. which are the error type and the exception itself. please check:

```python
def on_err(err_type: str, err: Exception):
    if err_type in (eureka_client.ERROR_REGISTER, eureka_client.ERROR_DISCOVER):
        eureka_client.stop()
    else:
        print(f"{err_type}::{err}")

your_rest_server_port = 9090
eureka_client.init(eureka_server="http://your-eureka-server-peer1,http://your-eureka-server-peer2",
                                app_name="python_module_1",
                                instance_port=your_rest_server_port,
                                on_error=on_err)

```

### Call Remote Service

After `init` the eureka client, this is the most simplist way to do service:

```python
import py_eureka_client.eureka_client as eureka_client

try:
    res = eureka_client.do_service("OTHER-SERVICE-NAME", "/service/context/path")
    print("result of other service" + res)
except urllib.request.HTTPError as e:
    # If all nodes are down, a `HTTPError` will raise.
    print(e)
```

`do_service` function also recieve a `return_type` keyword parameter, which when `json` was passed, the result will be a `dict` type object whereas `response_object` is pass, the original HTTPResponse object will be return. Please read the relative document for more information.

You can also use its `async` version:

```python
import py_eureka_client.eureka_client as eureka_client

res = await eureka_client.do_service_async("OTHER-SERVICE-NAME", "/service/context/path")
```

*do_service method will automatically try other nodes when one node return a HTTP error, until one success or all nodes being tried.*

If you want to handle all the services' calling, you can use `walk_nodes` function:

```python
import py_eureka_client.eureka_client as eureka_client

# you can define this function with `async def`
def walk_using_your_own_urllib(url):
    print(url)
    """
    # Connect to url and read result, then return it.
    # The result you return here will be returned to the `eureka_client.walk_nodes` function
    # If you want find this node is down, you can raise a `urllib.request.HTTPError`(urllib2.HTTPError in python2)
    # Then the `eureka_client.walk_nodes` will try to find another node to do the service.
    """

# result is the result that you return in walk_using_your_own_urllib function
try:
    res = eureka_client.walk_nodes("OTHER-SERVICE-NAME", "/service/context/path", walker=walk_using_your_own_urllib)
    print(res)
except urllib.request.HTTPError as e:
    # If all nodes are down, a `HTTPError` will raise.
    print(e)
```

A `async` version is also provied:

```python
import py_eureka_client.eureka_client as eureka_client

def walk_using_your_own_urllib(url):
    print(url)
    """
    # Connect to url and read result, then return it.
    # The result you return here will be returned to the `eureka_client.walk_nodes` function
    # If provided node is down, you can raise a `urllib.request.HTTPError`(urllib2.HTTPError in python2)
    # Then the `eureka_client.walk_nodes` will try to find another node to do the service.
    """

res = await eureka_client.walk_nodes_async("OTHER-SERVICE-NAME", "/service/context/path",
                          walker=walk_using_your_own_urllib)
```

### High Available Strategies

There are several HA strategies when using discovery client. They are:

* HA_STRATEGY_RANDOM, default strategy, find an node randamly.
* HA_STRATEGY_STICK, use one node until it goes down.
* HA_STRATEGY_OTHER, always use a different node from the last time.

In your `init` function, you can specify one of the above strategies:

```python
import py_eureka_client.eureka_client as eureka_client
# General init method
eureka_client.init(eureka_server="http://your-eureka-server-peer1,http://your-eureka-server-peer2",
                   app_name="your_app_name",
                   instance_port=your_rest_server_port,
                   ha_strategy=eureka_client.HA_STRATEGY_STICK)
```

If the build-in stratergies do not satify you, you can load all the registry by following code:

```python
import py_eureka_client.eureka_client as eureka_client

client = eureka_client.get_client()
app = client.applications.get_application("OTHER-SERVICE-NAME")
up_instances = app.up_instances
up_instances_same_zone = app.up_instances_in_zone(client.zone)
up_instances_other_zone = app.up_instances_not_in_zone(client.zone)
inst = up_instances[0]

# ... construct your url and do the service call

```

### Use Other Http Client

You can use other http client to connect to eureka server and other service rather than the build-in urlopen method. It should be useful if you use https connections via self-signed cetificates. 

From `0.11.0`, the methods of the `http_client.HttpClient` are defined `async`, you can now use some async http libs like `aiohttp`

To do this, you should:

1. (Optional) At most scenario, you should also write a class that inherited from `py_eureka_client.http_client.HttpResponse`, for the reason of the `py_eureka_client.http_client.HttpResponse` class wraps the `http.client.HTTPResponse` which may not return by the third http libs. 
2. Write a class inherited the `HttpClient` class in `py_eureka_client.http_client`.
3. Rewrite the `urlopen` method in your class. this method must return an subclass of `py_eureka_client.http_client.HttpResponse`, which is a wrapper class that hold to properties called `raw_response` and `body_text`. 
4. Set you own HttpClient object into `py_eureka_client.http_client` by `py_eureka_client.set_http_client`

```python
import py_eureka_client.http_client as http_client

# 1. A class inherited `py_eureka_client.http_client.HttpResonse`

class MyHttpResponse(http_client.HttpResponse):

    def __init__(self, raw_response):
        """
        " This raw response will return when you pass `response_object` in the `do_service` function.
        """
        self.raw_response = raw_response
    
    @property
    def body_text(self):
        txt = ""
        """
        " Read the body text from `self.raw_response`
        """
        return txt

# 2. A class inherited `py_eureka_client.http_client.HttpClient`.
class MyHttpClient(http_client.HttpClient):

    # 3. Rewrite the `urlopen` method in your class.
    # If you want to raise an exception, please make sure that the exception is an `http_client.HTTPError` or `http_client.URLError`.     
    async def urlopen(self, request: Union[str, http_client.HttpRequest] = None,
                      data: bytes = None, timeout: float = None) -> http_client.HttpResponse:
        res = await your_own_http_client_lib.do_the_visit(request, data, timeout)
        return MyHttpResponse(res)
        # You can parse your response object here, and set the body_text to http_client.HttpResponse, then you may ignore the http_client.HttpResponse inheritance.
        # body_txt = parse_res_body(res)
        # http_res = http_client.HttpResponse()
        # http_res.raw_response = res
        # http_res.body_text = body_text
        # return http_res
            

# 4. Set your class to `py_eureka_client.http_client`. 
http_client.set_http_client(MyHttpClient())
```

### Logger

The default logger is try to write logs to the screen, you can specify the logger handler to write it to a file.

```python
import py_eureka_client.logger as logger
import logging

_formatter = logging.Formatter(fmt='[%(asctime)s]-[%(name)s]-%(levelname)-4s: %(message)s')
_handler = logging.TimedRotatingFileHandler("/var/log/py-eureka-client.log", when="midnight", backupCount=7)
_handler.setFormatter(_formatter)
_handler.setLevel("INFO")

logger.set_handler(_handler)
```

If you want to add a handler rather than replace the inner one, you can use:

```python
logger.add_handler(_handler)
```

If you want to change the logger level:

```python
logger.set_level("DEBUG")
```

This logger will first save all the log record to a global queue, and then output them in a background thread, so it is very suitable for getting several logger with a same handler, especialy the `TimedRotatingFileHandler` which may slice the log files not quite well in a mutiple thread environment. 

### Costom Logger

If you want to use your own logger, you can do this:

```python
import py_eureka_client.logger as logger

logger.set_custom_logger(your_logger)
```

## Amazon Data Center Support

This component should support deploying in Amazone EC2, it should automatically load metadata from Amazon metadata service. All the metadata keys come from `com.netflix.appinfo.AmazonInfo` in Netflix's java client. BUT for the reason that I have no amazon environment to test, it may not work. If errors occurs, please submit an issue and provide some detail logs, I will try to fix it as far as I can. If it works, a reply in this [issue](https://github.com/keijack/python-eureka-client/issues/33) is wellcomed.

## More Infomation

You can find more information in the project comments.


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "py-eureka-client",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": null,
    "keywords": "eureka, netflix, microservice",
    "author": null,
    "author_email": "keijack <keijack.wu@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/aa/62/034f1281fd348455200d5221da70a071a29f8ac97c2acc32eb21841fc6eb/py_eureka_client-0.11.13.tar.gz",
    "platform": null,
    "description": "# python-eureka-client\n\n[![PyPI version](https://badge.fury.io/py/py-eureka-client.png)](https://badge.fury.io/py/py-eureka-client)\n\n## Discription\n\nThis is an eureka client written in python, you can easily intergrate your python components with spring cloud.\n\n## Support Python Version\n\nPython 3.7+\n\n*From `0.9.0`, python 2 is no longer supported, if you are using python 2, please use  version `0.8.12`.*\n\n## Why choose\n\n* Register your python components to eureka server.\n* Support failover.\n* Support DNS discovery. \n* Send heartbeat to eureka server.\n* Pull registry from eureka server.\n* Easy to use interface to use other REST service.\n* HA when calling other REST service.\n* Both trandictional and async def interfaces are provided.\n* The http client lib is replacable.\n\n## How to use\n\n### Install\n\n```Shell\npip install py_eureka_client\n```\n\n### Getting Start\n\nThis is the easiest way to use this component.\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\nyour_rest_server_port = 9090\n# The flowing code will register your server to eureka server and also start to send heartbeat every 30 seconds\neureka_client.init(eureka_server=\"http://your-eureka-server-peer1,http://your-eureka-server-peer2\",\n                   app_name=\"your_app_name\",\n                   instance_port=your_rest_server_port)\n```\n\nThen, in your business code, use\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\nres = eureka_client.do_service(\"OTHER-SERVICE-NAME\", \"/service/context/path\")\nprint(\"result of other service\" + res)\n\n```\n\nYou can also use the `EurekaClient` class. \n\n```python\nfrom py_eureka_client.eureka_client import EurekaClient\nclient = EurekaClient(eureka_server=\"http://my_eureka_server_peer_1/eureka/v2,http://my_eureka_server_peer_2/eureka/v2\", app_name=\"python_module_1\", instance_port=9090)\nawait client.start()\nres = await client.do_service(\"OTHER-SERVICE-NAME\", \"/service/context/path\")\nprint(\"result of other service\" + res)\n# when server is shutted down:\nawait client.stop()\n```\n\nIn fact, the `init` function is a facade of the EurekaClient, it holds a client object behind, you can get that by catching its return value or use `eureka_client.get_client()` to get it. The `init` function will automatically start the client, while using raw `EurekaClient`, you must call the `start()` and `stop()` method explicitly. \n\n\nFrom `0.11.0`, all the methods in `EurekaClient` are defined `async`, and there are also async facade for `init`, `do_servise`, `stop` functions names `init_async`, `do_service_async`, `sto_async`.\n\n### Registering to Eureka Server\n\nThe most common method to will be like:\n\n```Python\nimport py_eureka_client.eureka_client as eureka_client\n\nyour_rest_server_port = 9090\neureka_client.init(eureka_server=\"http://your-eureka-server-peer1,http://your-eureka-server-peer2\",\n                                app_name=\"python_module_1\",\n                                instance_port=your_rest_server_port)\n```\n\nBut if you have deploy your eureka server in several zones, you should specify the `eureka_availability_zones` parameter.\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\neureka_client.init(eureka_availability_zones={\n                \"us-east-1c\": \"http://ec2-552-627-568-165.compute-1.amazonaws.com:7001/eureka/v2/,http://ec2-368-101-182-134.compute-1.amazonaws.com:7001/eureka/v2/\",\n                \"us-east-1d\": \"http://ec2-552-627-568-170.compute-1.amazonaws.com:7001/eureka/v2/\",\n                \"us-east-1e\": \"http://ec2-500-179-285-592.compute-1.amazonaws.com:7001/eureka/v2/\"}, \n                zone=\"us-east-1c\",\n                app_name=\"python_module_1\", \n                instance_port=9090,\n                data_center_name=\"Amazon\")\n```\n\nIf you are looking for flexibility, you should configure Eureka service URLs using DNS.\n\nFor instance, following is a DNS TXT record created in the DNS server that lists the set of available DNS names for a zone.\n\n```\ntxt.us-east-1.mydomaintest.netflix.net=\"us-east-1c.mydomaintest.netflix.net\" \"us-east-1d.mydomaintest.netflix.net\" \"us-east-1e.mydomaintest.netflix.net\"\n```\n\nThen, you can define TXT records recursively for each zone similar to the following (if more than one hostname per zone, space delimit)\n\n```\ntxt.us-east-1c.mydomaintest.netflix.net=\"ec2-552-627-568-165.compute-1.amazonaws.com\" \"ec2-368-101-182-134.compute-1.amazonaws.com\"\ntxt.us-east-1d.mydomaintest.netflix.net=\"ec2-552-627-568-170.compute-1.amazonaws.com\"\ntxt.us-east-1e.mydomaintest.netflix.net=\"ec2-500-179-285-592.compute-1.amazonaws.com\"\n```\n\nAnd then you can create the client like: \n\n```python\nimport py_eureka_client.eureka_client as eureka_client\neureka_client.init(eureka_domain=\"mydomaintest.netflix.net\",\n                region=\"us-east-1\",\n                zone=\"us-east-1c\",\n                app_name=\"python_module_1\", \n                instance_port=9090,\n                data_center_name=\"Amazon\")\n```\n\nYou can specify the protocol, basic authentication and context path of your eureka server separatly rather than setting it at the URL. \n\n```python\nimport py_eureka_client.eureka_client as eureka_client\neureka_client.init(eureka_domain=\"mydomaintest.netflix.net\",\n                region=\"us-east-1\",\n                zone=\"us-east-1c\",\n                eureka_protocol=\"https\",\n                eureka_basic_auth_user=\"keijack\",\n                eureka_basic_auth_password=\"kjauthpass\",\n                eureka_context=\"/eureka/v2\",\n                app_name=\"python_module_1\", \n                instance_port=9090,\n```\n\nor\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\neureka_client.init(eureka_server=\"your-eureka-server-peer1,your-eureka-server-peer2\",\n                eureka_protocol=\"https\",\n                eureka_basic_auth_user=\"keijack\",\n                eureka_basic_auth_password=\"kjauthpass\",\n                eureka_context=\"/eureka/v2\",\n                app_name=\"python_module_1\", \n                instance_port=9090)\n```\n\n*About the instance `IP` and `hostname`*:\n\nIf you are using a `Amazon` data center, `py-eureka-client` will try to use `local-ipv4` and `local-hostname` get from Amazon metadata service. In other cases, `py-eureka-client` will use the first non-loopback ip address and hostname from your net interface. \n\nYou can also specify both these tow field or just one of them explicitly:\n\n```python\neureka_client.init(eureka_server=\"your-eureka-server-peer1,your-eureka-server-peer2\",\n                eureka_protocol=\"https\",\n                eureka_basic_auth_user=\"keijack\",\n                eureka_basic_auth_password=\"kjauthpass\",\n                eureka_context=\"/eureka/v2\",\n                app_name=\"python_module_1\", \n                instance_ip=\"192.168.10.168\",\n                instance_host=\"my-py-component.mydomian.com\",\n                instance_port=9090)\n```\n\nIn some case you might have more than one interfaces attached, for example, you are running your application in a docker-container. In this case you can specify a network via `instance_ip_network` to be used to get the container's ip and host. You can use:\n\n```python\neureka_client.init(eureka_server=\"your-eureka-server-peer1,your-eureka-server-peer2\",\n                eureka_protocol=\"https\",\n                eureka_basic_auth_user=\"keijack\",\n                eureka_basic_auth_password=\"kjauthpass\",\n                eureka_context=\"/eureka/v2\",\n                app_name=\"python_module_1\", \n                instance_ip_network=\"192.168.10.0/24\",\n                instance_port=9090)\n```\n\nIf you want to get the ip only and sepecify the host by yourself, try:\n\n```python\nimport py_eureka_client.netint_utils as netint_utils\n\n# you can get the ip only\nip = netint_utils.get_first_non_loopback_ip(\"192.168.10.0/24\")\nhost = \"my-py-component.mydomian.com\"\n\neureka_client.init(eureka_server=\"your-eureka-server-peer1,your-eureka-server-peer2\",\n                eureka_protocol=\"https\",\n                eureka_basic_auth_user=\"keijack\",\n                eureka_basic_auth_password=\"kjauthpass\",\n                eureka_context=\"/eureka/v2\",\n                app_name=\"python_module_1\", \n                instance_ip=ip,\n                instance_host=host,\n                instance_port=9090)\n```\n\n### Error Callback\n\nYou can specify a callback function when initializing the eureka client, when errors occur in `register`, `discover` or `status update` phase, the callback function will be called to inform you. The callback function will be called only when all the eureka server url are all tried and fails. \n\nThe callback function should accept 2 arguments. which are the error type and the exception itself. please check:\n\n```python\ndef on_err(err_type: str, err: Exception):\n    if err_type in (eureka_client.ERROR_REGISTER, eureka_client.ERROR_DISCOVER):\n        eureka_client.stop()\n    else:\n        print(f\"{err_type}::{err}\")\n\nyour_rest_server_port = 9090\neureka_client.init(eureka_server=\"http://your-eureka-server-peer1,http://your-eureka-server-peer2\",\n                                app_name=\"python_module_1\",\n                                instance_port=your_rest_server_port,\n                                on_error=on_err)\n\n```\n\n### Call Remote Service\n\nAfter `init` the eureka client, this is the most simplist way to do service:\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\ntry:\n    res = eureka_client.do_service(\"OTHER-SERVICE-NAME\", \"/service/context/path\")\n    print(\"result of other service\" + res)\nexcept urllib.request.HTTPError as e:\n    # If all nodes are down, a `HTTPError` will raise.\n    print(e)\n```\n\n`do_service` function also recieve a `return_type` keyword parameter, which when `json` was passed, the result will be a `dict` type object whereas `response_object` is pass, the original HTTPResponse object will be return. Please read the relative document for more information.\n\nYou can also use its `async` version:\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\nres = await eureka_client.do_service_async(\"OTHER-SERVICE-NAME\", \"/service/context/path\")\n```\n\n*do_service method will automatically try other nodes when one node return a HTTP error, until one success or all nodes being tried.*\n\nIf you want to handle all the services' calling, you can use `walk_nodes` function:\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\n# you can define this function with `async def`\ndef walk_using_your_own_urllib(url):\n    print(url)\n    \"\"\"\n    # Connect to url and read result, then return it.\n    # The result you return here will be returned to the `eureka_client.walk_nodes` function\n    # If you want find this node is down, you can raise a `urllib.request.HTTPError`(urllib2.HTTPError in python2)\n    # Then the `eureka_client.walk_nodes` will try to find another node to do the service.\n    \"\"\"\n\n# result is the result that you return in walk_using_your_own_urllib function\ntry:\n    res = eureka_client.walk_nodes(\"OTHER-SERVICE-NAME\", \"/service/context/path\", walker=walk_using_your_own_urllib)\n    print(res)\nexcept urllib.request.HTTPError as e:\n    # If all nodes are down, a `HTTPError` will raise.\n    print(e)\n```\n\nA `async` version is also provied:\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\ndef walk_using_your_own_urllib(url):\n    print(url)\n    \"\"\"\n    # Connect to url and read result, then return it.\n    # The result you return here will be returned to the `eureka_client.walk_nodes` function\n    # If provided node is down, you can raise a `urllib.request.HTTPError`(urllib2.HTTPError in python2)\n    # Then the `eureka_client.walk_nodes` will try to find another node to do the service.\n    \"\"\"\n\nres = await eureka_client.walk_nodes_async(\"OTHER-SERVICE-NAME\", \"/service/context/path\",\n                          walker=walk_using_your_own_urllib)\n```\n\n### High Available Strategies\n\nThere are several HA strategies when using discovery client. They are:\n\n* HA_STRATEGY_RANDOM, default strategy, find an node randamly.\n* HA_STRATEGY_STICK, use one node until it goes down.\n* HA_STRATEGY_OTHER, always use a different node from the last time.\n\nIn your `init` function, you can specify one of the above strategies:\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n# General init method\neureka_client.init(eureka_server=\"http://your-eureka-server-peer1,http://your-eureka-server-peer2\",\n                   app_name=\"your_app_name\",\n                   instance_port=your_rest_server_port,\n                   ha_strategy=eureka_client.HA_STRATEGY_STICK)\n```\n\nIf the build-in stratergies do not satify you, you can load all the registry by following code:\n\n```python\nimport py_eureka_client.eureka_client as eureka_client\n\nclient = eureka_client.get_client()\napp = client.applications.get_application(\"OTHER-SERVICE-NAME\")\nup_instances = app.up_instances\nup_instances_same_zone = app.up_instances_in_zone(client.zone)\nup_instances_other_zone = app.up_instances_not_in_zone(client.zone)\ninst = up_instances[0]\n\n# ... construct your url and do the service call\n\n```\n\n### Use Other Http Client\n\nYou can use other http client to connect to eureka server and other service rather than the build-in urlopen method. It should be useful if you use https connections via self-signed cetificates. \n\nFrom `0.11.0`, the methods of the `http_client.HttpClient` are defined `async`, you can now use some async http libs like `aiohttp`\n\nTo do this, you should:\n\n1. (Optional) At most scenario, you should also write a class that inherited from `py_eureka_client.http_client.HttpResponse`, for the reason of the `py_eureka_client.http_client.HttpResponse` class wraps the `http.client.HTTPResponse` which may not return by the third http libs. \n2. Write a class inherited the `HttpClient` class in `py_eureka_client.http_client`.\n3. Rewrite the `urlopen` method in your class. this method must return an subclass of `py_eureka_client.http_client.HttpResponse`, which is a wrapper class that hold to properties called `raw_response` and `body_text`. \n4. Set you own HttpClient object into `py_eureka_client.http_client` by `py_eureka_client.set_http_client`\n\n```python\nimport py_eureka_client.http_client as http_client\n\n# 1. A class inherited `py_eureka_client.http_client.HttpResonse`\n\nclass MyHttpResponse(http_client.HttpResponse):\n\n    def __init__(self, raw_response):\n        \"\"\"\n        \" This raw response will return when you pass `response_object` in the `do_service` function.\n        \"\"\"\n        self.raw_response = raw_response\n    \n    @property\n    def body_text(self):\n        txt = \"\"\n        \"\"\"\n        \" Read the body text from `self.raw_response`\n        \"\"\"\n        return txt\n\n# 2. A class inherited `py_eureka_client.http_client.HttpClient`.\nclass MyHttpClient(http_client.HttpClient):\n\n    # 3. Rewrite the `urlopen` method in your class.\n    # If you want to raise an exception, please make sure that the exception is an `http_client.HTTPError` or `http_client.URLError`.     \n    async def urlopen(self, request: Union[str, http_client.HttpRequest] = None,\n                      data: bytes = None, timeout: float = None) -> http_client.HttpResponse:\n        res = await your_own_http_client_lib.do_the_visit(request, data, timeout)\n        return MyHttpResponse(res)\n        # You can parse your response object here, and set the body_text to http_client.HttpResponse, then you may ignore the http_client.HttpResponse inheritance.\n        # body_txt = parse_res_body(res)\n        # http_res = http_client.HttpResponse()\n        # http_res.raw_response = res\n        # http_res.body_text = body_text\n        # return http_res\n            \n\n# 4. Set your class to `py_eureka_client.http_client`. \nhttp_client.set_http_client(MyHttpClient())\n```\n\n### Logger\n\nThe default logger is try to write logs to the screen, you can specify the logger handler to write it to a file.\n\n```python\nimport py_eureka_client.logger as logger\nimport logging\n\n_formatter = logging.Formatter(fmt='[%(asctime)s]-[%(name)s]-%(levelname)-4s: %(message)s')\n_handler = logging.TimedRotatingFileHandler(\"/var/log/py-eureka-client.log\", when=\"midnight\", backupCount=7)\n_handler.setFormatter(_formatter)\n_handler.setLevel(\"INFO\")\n\nlogger.set_handler(_handler)\n```\n\nIf you want to add a handler rather than replace the inner one, you can use:\n\n```python\nlogger.add_handler(_handler)\n```\n\nIf you want to change the logger level:\n\n```python\nlogger.set_level(\"DEBUG\")\n```\n\nThis logger will first save all the log record to a global queue, and then output them in a background thread, so it is very suitable for getting several logger with a same handler, especialy the `TimedRotatingFileHandler` which may slice the log files not quite well in a mutiple thread environment. \n\n### Costom Logger\n\nIf you want to use your own logger, you can do this:\n\n```python\nimport py_eureka_client.logger as logger\n\nlogger.set_custom_logger(your_logger)\n```\n\n## Amazon Data Center Support\n\nThis component should support deploying in Amazone EC2, it should automatically load metadata from Amazon metadata service. All the metadata keys come from `com.netflix.appinfo.AmazonInfo` in Netflix's java client. BUT for the reason that I have no amazon environment to test, it may not work. If errors occurs, please submit an issue and provide some detail logs, I will try to fix it as far as I can. If it works, a reply in this [issue](https://github.com/keijack/python-eureka-client/issues/33) is wellcomed.\n\n## More Infomation\n\nYou can find more information in the project comments.\n\n",
    "bugtrack_url": null,
    "license": "MIT License  Copyright (c) 2018 Keijack Wu  Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \"Software\"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:  The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.  THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.",
    "summary": "A eureka client written in python. Support registering your python component to Eureka Server, as well as calling remote services by pulling the the Eureka registry.",
    "version": "0.11.13",
    "project_urls": {
        "homepage": "https://github.com/keijack/python-eureka-client",
        "repository": "https://github.com/keijack/python-eureka-client"
    },
    "split_keywords": [
        "eureka",
        " netflix",
        " microservice"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "6b0854851d6d3240b342a954f50dc9246e9cf72ccba5c8942b96b284c4c2842b",
                "md5": "dd32de6c11b3f80f870db118a9b2f98d",
                "sha256": "41c376d9d1ba55fb325f14cdcbaa007a2c8af8b06fcf03ad2f30155482ac39fd"
            },
            "downloads": -1,
            "filename": "py_eureka_client-0.11.13-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "dd32de6c11b3f80f870db118a9b2f98d",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 33951,
            "upload_time": "2024-11-11T03:16:20",
            "upload_time_iso_8601": "2024-11-11T03:16:20.619802Z",
            "url": "https://files.pythonhosted.org/packages/6b/08/54851d6d3240b342a954f50dc9246e9cf72ccba5c8942b96b284c4c2842b/py_eureka_client-0.11.13-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "aa62034f1281fd348455200d5221da70a071a29f8ac97c2acc32eb21841fc6eb",
                "md5": "74b7eafd7c99f344074fa232cc5f8626",
                "sha256": "277e033975d1d9a452949de1de0b81f4faf82a93b484a167e6cc66046eda9126"
            },
            "downloads": -1,
            "filename": "py_eureka_client-0.11.13.tar.gz",
            "has_sig": false,
            "md5_digest": "74b7eafd7c99f344074fa232cc5f8626",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 32152,
            "upload_time": "2024-11-11T03:16:22",
            "upload_time_iso_8601": "2024-11-11T03:16:22.281738Z",
            "url": "https://files.pythonhosted.org/packages/aa/62/034f1281fd348455200d5221da70a071a29f8ac97c2acc32eb21841fc6eb/py_eureka_client-0.11.13.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-11-11 03:16:22",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "keijack",
    "github_project": "python-eureka-client",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "requirements": [],
    "lcname": "py-eureka-client"
}
        
Elapsed time: 0.81371s