app.py 17.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
#!/user/bin/env python
# Copyright (c) 2017 VisualDL Authors. All Rights Reserve.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# =======================================================================
16
import json
C
chenjian 已提交
17
import multiprocessing
18
import os
C
chenjian 已提交
19
import re
20 21
import sys
import threading
C
chenjian 已提交
22
import time
23
import urllib
24
import webbrowser
25

C
chenjian 已提交
26 27 28 29 30 31 32
import requests
from flask import Flask
from flask import make_response
from flask import redirect
from flask import request
from flask import Response
from flask import send_file
P
Peter Pan 已提交
33
from flask_babel import Babel
34 35

import visualdl.server
C
chenjian 已提交
36
from visualdl import __version__
37 38
from visualdl.component.inference.fastdeploy_lib import get_start_arguments
from visualdl.component.inference.fastdeploy_server import create_fastdeploy_api_call
39
from visualdl.component.inference.model_convert_server import create_model_convert_api_call
C
chenjian 已提交
40
from visualdl.component.profiler.profiler_server import create_profiler_api_call
P
Peter Pan 已提交
41
from visualdl.server.api import create_api_call
42
from visualdl.server.api import get_component_tabs
C
chenjian 已提交
43 44
from visualdl.server.args import parse_args
from visualdl.server.args import ParseArgs
P
Peter Pan 已提交
45
from visualdl.server.log import info
C
chenjian 已提交
46
from visualdl.server.serve import upload_to_dev
47
from visualdl.server.template import Template
C
chenjian 已提交
48
from visualdl.utils import update_util
49 50 51 52 53 54 55

SERVER_DIR = os.path.join(visualdl.ROOT, 'server')

support_language = ["en", "zh"]
default_language = support_language[0]

server_path = os.path.abspath(os.path.dirname(sys.argv[0]))
P
Peter Pan 已提交
56
template_file_path = os.path.join(SERVER_DIR, "./dist")
57 58
mock_data_path = os.path.join(SERVER_DIR, "./mock_data/")

P
Peter Pan 已提交
59 60
check_live_path = '/alive'

61

C
chenjian 已提交
62
def create_app(args):  # noqa: C901
P
Peter Pan 已提交
63 64 65 66
    # disable warning from flask
    cli = sys.modules['flask.cli']
    cli.show_server_banner = lambda *x: None

67
    app = Flask('visualdl', static_folder=None)
P
Peter Pan 已提交
68 69
    app.logger.disabled = True

70 71 72
    # set static expires in a short time to reduce browser's memory usage.
    app.config['SEND_FILE_MAX_AGE_DEFAULT'] = 30

P
Peter Pan 已提交
73
    app.config['BABEL_DEFAULT_LOCALE'] = default_language
74 75 76 77 78 79 80 81 82

    def get_locale():
        lang = args.language
        if not lang or lang not in support_language:
            lang = request.accept_languages.best_match(support_language)
        return lang

    babel = Babel(app, locale_selector=get_locale)  # noqa:F841
    # Babel api from flask_babel v3.0.0
83
    api_call = create_api_call(args.logdir, args.model, args.cache_timeout)
C
chenjian 已提交
84
    profiler_api_call = create_profiler_api_call(args.logdir)
85
    inference_api_call = create_model_convert_api_call()
86
    fastdeploy_api_call = create_fastdeploy_api_call()
87 88
    if args.telemetry:
        update_util.PbUpdater(args.product).start()
89

90
    public_path = args.public_path
P
Peter Pan 已提交
91 92
    api_path = public_path + '/api'

P
Peter Pan 已提交
93 94 95 96 97 98
    def append_query_string(url):
        query_string = ''
        if request.query_string:
            query_string = '?' + request.query_string.decode()
        return url + query_string

99 100
    if not args.api_only:

P
Peter Pan 已提交
101 102
        template = Template(
            os.path.join(server_path, template_file_path),
103 104 105
            PUBLIC_PATH=public_path,
            BASE_URI=public_path,
            API_URL=api_path,
C
chenjian 已提交
106 107 108
            TELEMETRY_ID='63a600296f8a71f576c4806376a9245b'
            if args.telemetry else '',
            THEME='' if args.theme is None else args.theme)
P
Peter Pan 已提交
109

P
Peter Pan 已提交
110
        @app.route('/')
111
        def base():
P
Peter Pan 已提交
112
            return redirect(append_query_string(public_path), code=302)
113

P
Peter Pan 已提交
114
        @app.route('/favicon.ico')
115 116 117 118
        def favicon():
            icon = os.path.join(template_file_path, 'favicon.ico')
            if os.path.exists(icon):
                return send_file(icon)
P
Peter Pan 已提交
119
            return 'file not found', 404
120

121
        @app.route(public_path + '/')
122
        def index():
C
chenjian 已提交
123 124
            return redirect(
                append_query_string(public_path + '/index'), code=302)
125 126 127

        @app.route(public_path + '/<path:filename>')
        def serve_static(filename):
128
            is_not_page_request = re.search(r'\..+$', filename)
C
chenjian 已提交
129 130
            response = template.render(
                filename if is_not_page_request else 'index.html')
131
            if not is_not_page_request:
C
chenjian 已提交
132 133 134 135 136 137 138
                response.set_cookie(
                    'vdl_lng',
                    get_locale(),
                    path='/',
                    samesite='Strict',
                    secure=False,
                    httponly=False)
139
            return response
140

C
chenjian 已提交
141
    @app.route(api_path + '/<path:method>', methods=["GET", "POST"])
P
Peter Pan 已提交
142
    def serve_api(method):
143
        data, mimetype, headers = api_call(method, request.args)
C
chenjian 已提交
144 145
        return make_response(
            Response(data, mimetype=mimetype, headers=headers))
P
Peter Pan 已提交
146

C
chenjian 已提交
147 148 149 150 151 152
    @app.route(api_path + '/profiler/<path:method>', methods=["GET", "POST"])
    def serve_profiler_api(method):
        data, mimetype, headers = profiler_api_call(method, request.args)
        return make_response(
            Response(data, mimetype=mimetype, headers=headers))

153 154 155 156 157 158 159 160 161
    @app.route(api_path + '/inference/<path:method>', methods=["GET", "POST"])
    def serve_inference_api(method):
        if request.method == 'POST':
            data, mimetype, headers = inference_api_call(method, request.form)
        else:
            data, mimetype, headers = inference_api_call(method, request.args)
        return make_response(
            Response(data, mimetype=mimetype, headers=headers))

162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184
    @app.route(api_path + '/fastdeploy/<path:method>', methods=["GET", "POST"])
    def serve_fastdeploy_api(method):
        if request.method == 'POST':
            data, mimetype, headers = fastdeploy_api_call(method, request.form)
        else:
            data, mimetype, headers = fastdeploy_api_call(method, request.args)
        return make_response(
            Response(data, mimetype=mimetype, headers=headers))

    @app.route(
        api_path + '/fastdeploy/fastdeploy_client', methods=["GET", "POST"])
    def serve_fastdeploy_create_fastdeploy_client():
        try:
            if request.method == 'POST':
                fastdeploy_api_call('create_fastdeploy_client', request.form)
                request_args = request.form
            else:
                fastdeploy_api_call('create_fastdeploy_client', request.args)
                request_args = request.args
        except Exception as e:
            error_msg = '{}'.format(e)
            return make_response(error_msg)
        args = urllib.parse.urlencode(request_args)
185

186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205
        if args:
            return redirect(
                api_path + "/fastdeploy/fastdeploy_client/app?{}".format(args),
                code=302)
        return redirect(
            api_path + "/fastdeploy/fastdeploy_client/app", code=302)

    @app.route(
        api_path + "/fastdeploy/fastdeploy_client/<path:path>",
        methods=["GET", "POST"])
    def request_fastdeploy_create_fastdeploy_client_app(path: str):
        '''
        Gradio app server url interface. We route urls for gradio app to gradio server.

        Args:
            path(str): All resource path from gradio server.

        Returns:
            Any thing from gradio server.
        '''
206
        lang = 'zh'
207
        if request.method == 'POST':
208 209 210 211 212 213 214 215 216 217
            if request.mimetype == 'application/json':
                request_args = request.json
            else:
                request_args = request.form.to_dict()
            if 'data' in request_args:
                lang = request_args['data'][-1]
                request_args['lang'] = lang
            elif 'lang' in request_args:
                lang = request_args['lang']

218
            port = fastdeploy_api_call('create_fastdeploy_client',
219
                                       request_args)
220
        else:
221 222 223 224 225 226
            request_args = request.args.to_dict()
            if 'data' in request_args:
                lang = request_args['data'][-1]
                request_args['lang'] = lang
            elif 'lang' in request_args:
                lang = request_args['lang']
227
            port = fastdeploy_api_call('create_fastdeploy_client',
228 229
                                       request_args)

230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259
        if path == 'app':
            proxy_url = request.url.replace(
                request.host_url.rstrip('/') + api_path +
                '/fastdeploy/fastdeploy_client/app',
                'http://localhost:{}/'.format(port))
        else:
            proxy_url = request.url.replace(
                request.host_url.rstrip('/') + api_path +
                '/fastdeploy/fastdeploy_client/',
                'http://localhost:{}/'.format(port))
        resp = requests.request(
            method=request.method,
            url=proxy_url,
            headers={
                key: value
                for (key, value) in request.headers if key != 'Host'
            },
            data=request.get_data(),
            cookies=request.cookies,
            allow_redirects=False)
        if path == 'app':
            content = resp.content
            if request_args and 'server_id' in request_args:
                server_id = request_args.get('server_id')
                start_args = get_start_arguments(server_id)
                http_port = start_args.get('http-port', '')
                metrics_port = start_args.get('metrics-port', '')
                model_name = start_args.get('default_model_name', '')
                content = content.decode()
                try:
260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335
                    if request_args.get('lang', 'zh') == 'en':
                        default_server_addr = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps(
                                    "server ip", ensure_ascii=True).replace(
                                        '\\', '\\\\')), content).group(0)
                        cur_server_addr = default_server_addr.replace(
                            '"value": ""', '"value": "localhost"')
                        default_http_port = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps(
                                    "server port", ensure_ascii=True).replace(
                                        '\\', '\\\\')), content).group(0)
                        cur_http_port = default_http_port.replace(
                            '"value": ""', '"value": "{}"'.format(http_port))
                        default_metrics_port = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps(
                                    "metrics port", ensure_ascii=True).replace(
                                        '\\', '\\\\')), content).group(0)
                        cur_metrics_port = default_metrics_port.replace(
                            '"value": ""',
                            '"value": "{}"'.format(metrics_port))
                        default_model_name = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps(
                                    "model name", ensure_ascii=True).replace(
                                        '\\', '\\\\')), content).group(0)
                        cur_model_name = default_model_name.replace(
                            '"value": ""', '"value": "{}"'.format(model_name))
                        default_model_version = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps("model version",
                                           ensure_ascii=True).replace(
                                               '\\', '\\\\')),
                            content).group(0)
                        cur_model_version = default_model_version.replace(
                            '"value": ""', '"value": "{}"'.format('1'))
                        content = content.replace(default_server_addr,
                                                  cur_server_addr)
                    else:
                        default_server_addr = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps("服务ip", ensure_ascii=True).replace(
                                    '\\', '\\\\')), content).group(0)
                        cur_server_addr = default_server_addr.replace(
                            '"value": ""', '"value": "localhost"')
                        default_http_port = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps(
                                    "推理服务端口", ensure_ascii=True).replace(
                                        '\\', '\\\\')), content).group(0)
                        cur_http_port = default_http_port.replace(
                            '"value": ""', '"value": "{}"'.format(http_port))
                        default_metrics_port = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps(
                                    "性能服务端口", ensure_ascii=True).replace(
                                        '\\', '\\\\')), content).group(0)
                        cur_metrics_port = default_metrics_port.replace(
                            '"value": ""',
                            '"value": "{}"'.format(metrics_port))
                        default_model_name = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps("模型名称", ensure_ascii=True).replace(
                                    '\\', '\\\\')), content).group(0)
                        cur_model_name = default_model_name.replace(
                            '"value": ""', '"value": "{}"'.format(model_name))
                        default_model_version = re.search(
                            '"label": {}.*?"value": "".*?}}'.format(
                                json.dumps("模型版本", ensure_ascii=True).replace(
                                    '\\', '\\\\')), content).group(0)
                        cur_model_version = default_model_version.replace(
                            '"value": ""', '"value": "{}"'.format('1'))
                        content = content.replace(default_server_addr,
                                                  cur_server_addr)
336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357
                    if http_port:
                        content = content.replace(default_http_port,
                                                  cur_http_port)
                    if metrics_port:
                        content = content.replace(default_metrics_port,
                                                  cur_metrics_port)
                    if model_name:
                        content = content.replace(default_model_name,
                                                  cur_model_name)

                    content = content.replace(default_model_version,
                                              cur_model_version)
                except Exception:
                    pass
                finally:
                    content = content.encode()
        else:
            content = resp.content
        headers = [(name, value) for (name, value) in resp.raw.headers.items()]
        response = Response(content, resp.status_code, headers)
        return response

358 359 360 361 362 363 364 365 366 367
    @app.route(api_path + '/component_tabs')
    def component_tabs():
        data, mimetype, headers = get_component_tabs(
            api_call,
            profiler_api_call,
            vdl_args=args,
            request_args=request.args)
        return make_response(
            Response(data, mimetype=mimetype, headers=headers))

P
Peter Pan 已提交
368 369 370 371
    @app.route(check_live_path)
    def check_live():
        return '', 204

372 373 374
    return app


P
Peter Pan 已提交
375 376 377 378 379
def wait_until_live(args: ParseArgs):
    url = 'http://{host}:{port}'.format(host=args.host, port=args.port)
    while True:
        try:
            requests.get(url + check_live_path)
C
chenjian 已提交
380 381
            info('Running VisualDL at http://%s:%s/ (Press CTRL+C to quit)',
                 args.host, args.port)
P
Peter Pan 已提交
382 383

            if args.host == 'localhost':
C
chenjian 已提交
384 385 386
                info(
                    'Serving VisualDL on localhost; to expose to the network, use a proxy or pass --host 0.0.0.0'
                )
P
Peter Pan 已提交
387 388

            if args.api_only:
C
chenjian 已提交
389 390
                info('Running in API mode, only %s/* will be served.',
                     args.public_path + '/api')
P
Peter Pan 已提交
391 392 393 394 395 396 397 398 399 400

            break
        except Exception:
            time.sleep(0.5)
    if not args.api_only and args.open_browser:
        webbrowser.open(url + args.public_path)


def _run(args):
    args = ParseArgs(**args)
401
    os.system('')
P
Peter Pan 已提交
402
    info('\033[1;33mVisualDL %s\033[0m', __version__)
走神的阿圆's avatar
走神的阿圆 已提交
403
    app = create_app(args)
C
chenjian 已提交
404
    threading.Thread(target=wait_until_live, args=(args, )).start()
405
    app.run(debug=False, host=args.host, port=args.port, threaded=False)
406 407


408
def run(logdir=None, **options):
C
chenjian 已提交
409
    args = {'logdir': logdir}
P
Peter Pan 已提交
410
    args.update(options)
C
chenjian 已提交
411
    p = multiprocessing.Process(target=_run, args=(args, ))
412 413 414 415 416 417
    p.start()
    return p.pid


def main():
    args = parse_args()
走神的阿圆's avatar
走神的阿圆 已提交
418 419 420 421 422
    if args.get('dest') == 'service':
        if args.get('behavior') == 'upload':
            upload_to_dev(args.get('logdir'), args.get('model'))
    else:
        _run(args)
423 424


P
Peter Pan 已提交
425
if __name__ == '__main__':
P
Peter Pan 已提交
426
    main()