<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1" />
<meta name="generator" content="pdoc 0.10.0" />
<title>utils.wandb_wrapper API documentation</title>
<meta name="description" content="" />
<link rel="preload stylesheet" as="style" href="https://cdnjs.cloudflare.com/ajax/libs/10up-sanitize.css/11.0.1/sanitize.min.css" integrity="sha256-PK9q560IAAa6WVRRh76LtCaI8pjTJ2z11v0miyNNjrs=" crossorigin>
<link rel="preload stylesheet" as="style" href="https://cdnjs.cloudflare.com/ajax/libs/10up-sanitize.css/11.0.1/typography.min.css" integrity="sha256-7l/o7C8jubJiy74VsKTidCy1yBkRtiUGbVkYBylBqUg=" crossorigin>
<link rel="stylesheet preload" as="style" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/10.1.1/styles/github.min.css" crossorigin>
<style>:root{--highlight-color:#fe9}.flex{display:flex !important}body{line-height:1.5em}#content{padding:20px}#sidebar{padding:30px;overflow:hidden}#sidebar > *:last-child{margin-bottom:2cm}.http-server-breadcrumbs{font-size:130%;margin:0 0 15px 0}#footer{font-size:.75em;padding:5px 30px;border-top:1px solid #ddd;text-align:right}#footer p{margin:0 0 0 1em;display:inline-block}#footer p:last-child{margin-right:30px}h1,h2,h3,h4,h5{font-weight:300}h1{font-size:2.5em;line-height:1.1em}h2{font-size:1.75em;margin:1em 0 .50em 0}h3{font-size:1.4em;margin:25px 0 10px 0}h4{margin:0;font-size:105%}h1:target,h2:target,h3:target,h4:target,h5:target,h6:target{background:var(--highlight-color);padding:.2em 0}a{color:#058;text-decoration:none;transition:color .3s ease-in-out}a:hover{color:#e82}.title code{font-weight:bold}h2[id^="header-"]{margin-top:2em}.ident{color:#900}pre code{background:#f8f8f8;font-size:.8em;line-height:1.4em}code{background:#f2f2f1;padding:1px 4px;overflow-wrap:break-word}h1 code{background:transparent}pre{background:#f8f8f8;border:0;border-top:1px solid #ccc;border-bottom:1px solid #ccc;margin:1em 0;padding:1ex}#http-server-module-list{display:flex;flex-flow:column}#http-server-module-list div{display:flex}#http-server-module-list dt{min-width:10%}#http-server-module-list p{margin-top:0}.toc ul,#index{list-style-type:none;margin:0;padding:0}#index code{background:transparent}#index h3{border-bottom:1px solid #ddd}#index ul{padding:0}#index h4{margin-top:.6em;font-weight:bold}@media (min-width:200ex){#index .two-column{column-count:2}}@media (min-width:300ex){#index .two-column{column-count:3}}dl{margin-bottom:2em}dl dl:last-child{margin-bottom:4em}dd{margin:0 0 1em 3em}#header-classes + dl > dd{margin-bottom:3em}dd dd{margin-left:2em}dd p{margin:10px 0}.name{background:#eee;font-weight:bold;font-size:.85em;padding:5px 10px;display:inline-block;min-width:40%}.name:hover{background:#e0e0e0}dt:target .name{background:var(--highlight-color)}.name > span:first-child{white-space:nowrap}.name.class > span:nth-child(2){margin-left:.4em}.inherited{color:#999;border-left:5px solid #eee;padding-left:1em}.inheritance em{font-style:normal;font-weight:bold}.desc h2{font-weight:400;font-size:1.25em}.desc h3{font-size:1em}.desc dt code{background:inherit}.source summary,.git-link-div{color:#666;text-align:right;font-weight:400;font-size:.8em;text-transform:uppercase}.source summary > *{white-space:nowrap;cursor:pointer}.git-link{color:inherit;margin-left:1em}.source pre{max-height:500px;overflow:auto;margin:0}.source pre code{font-size:12px;overflow:visible}.hlist{list-style:none}.hlist li{display:inline}.hlist li:after{content:',\2002'}.hlist li:last-child:after{content:none}.hlist .hlist{display:inline;padding-left:1em}img{max-width:100%}td{padding:0 .5em}.admonition{padding:.1em .5em;margin-bottom:1em}.admonition-title{font-weight:bold}.admonition.note,.admonition.info,.admonition.important{background:#aef}.admonition.todo,.admonition.versionadded,.admonition.tip,.admonition.hint{background:#dfd}.admonition.warning,.admonition.versionchanged,.admonition.deprecated{background:#fd4}.admonition.error,.admonition.danger,.admonition.caution{background:lightpink}</style>
<style media="screen and (min-width: 700px)">@media screen and (min-width:700px){#sidebar{width:30%;height:100vh;overflow:auto;position:sticky;top:0}#content{width:70%;max-width:100ch;padding:3em 4em;border-left:1px solid #ddd}pre code{font-size:1em}.item .name{font-size:1em}main{display:flex;flex-direction:row-reverse;justify-content:flex-end}.toc ul ul,#index ul{padding-left:1.5em}.toc > ul > li{margin-top:.5em}}</style>
<style media="print">@media print{#sidebar h1{page-break-before:always}.source{display:none}}@media print{*{background:transparent !important;color:#000 !important;box-shadow:none !important;text-shadow:none !important}a[href]:after{content:" (" attr(href) ")";font-size:90%}a[href][title]:after{content:none}abbr[title]:after{content:" (" attr(title) ")"}.ir a:after,a[href^="javascript:"]:after,a[href^="#"]:after{content:""}pre,blockquote{border:1px solid #999;page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}img{max-width:100% !important}@page{margin:0.5cm}p,h2,h3{orphans:3;widows:3}h1,h2,h3,h4,h5,h6{page-break-after:avoid}}</style>
<script async src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.7/latest.js?config=TeX-AMS_CHTML" integrity="sha256-kZafAc6mZvK3W3v1pHOcUix30OHQN6pU/NO2oFkqZVw=" crossorigin></script>
<script defer src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/10.1.1/highlight.min.js" integrity="sha256-Uv3H6lx7dJmRfRvH8TH6kJD1TSK1aFcwgx+mdg3epi8=" crossorigin></script>
<script>window.addEventListener('DOMContentLoaded', () => hljs.initHighlighting())</script>
</head>
<body>
<main>
<article id="content">
<header>
<h1 class="title">Module <code>utils.wandb_wrapper</code></h1>
</header>
<section id="section-intro">
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">#!/usr/bin/env python3
import threading
import wandb
from .logger import Logger

_is_init = False
_locker = threading.Lock()
_init_counter = 0


def initWandbProject(api_key, project, name, args):
    &#34;&#34;&#34;
    Thread safe way to initialize wand library, that is originally is not thread safe.
    If library has already been initialized this method only increase counter
    and does not perform any actual initialization.

    Args:
        api_key(str): personal api key for the use
        project(str): project name
        name(str): experiment name
        args: command line arguments

    Returns:
        Instance of the project from wandb library if library is initialized and None otherwise
    &#34;&#34;&#34;
    logger = Logger.get(args.run_id)

    global _is_init
    global _init_counter

    _locker.acquire()
    if _is_init:
        _init_counter += 1
        _locker.release()
        return None
    else:
        try:
            wandb.login(key=api_key)
            logger.info(f&#34;Wandb login completed successfully&#34;)
            run = wandb.init(
                # project=&#34;federated_nas&#34;,
                project=project,
                name=name,
                config=args,
                reinit=True
            )
            _is_init = True
            _init_counter += 1
            _locker.release()
            return run

        except ValueError as err:
            logger.error(f&#34;ERROR: Ignore Wandb login due to problems with login with API KEY: {str(err)}&#34;)
            _is_init = False
            _locker.release()
            return None


def finishProject(projectRun):
    &#34;&#34;&#34;
    Thread safe way to deinitialize wand library, that is originally is not thread safe.
    It will decrease counter, and once counter of library users will be zero it will perform final deinitialized.

    Args:
        projectRun: Instance of the project from wandb library

    Returns:
        None
    &#34;&#34;&#34;

    global _is_init
    global _init_counter

    _locker.acquire()
    _init_counter -= 1
    if _init_counter == 0:
        if projectRun:
            projectRun.finish()
        _is_init = False
    _locker.release()


def logStatistics(H, round):
    &#34;&#34;&#34;
    Log statistics from experiments and publish them via using wandb.
    To distinguish between experiments thr run_id is embeded into plot names.

    Args:
        H(dict): current server state
        round(int): current round

    Returns:
        None
    &#34;&#34;&#34;

    global _is_init
    if not _is_init:
        return

    item = H[&#39;history&#39;][round]

    full_gradient_oracles = \
        sum([v[&#39;client_state&#39;][&#39;stats&#39;][&#39;full_gradient_oracles&#39;] for k, v in item[&#34;client_states&#34;].items()])

    samples_gradient_oracles = \
        sum([v[&#39;client_state&#39;][&#39;stats&#39;][&#39;samples_gradient_oracles&#39;] for k, v in item[&#34;client_states&#34;].items()])

    send_scalars_to_master = \
        sum([v[&#39;client_state&#39;][&#39;stats&#39;][&#39;send_scalars_to_master&#39;] for k, v in item[&#34;client_states&#34;].items()])

    run_id = H[&#39;args&#39;].run_id

    # All quantities are per current round
    msg = {f&#34;full gradient oracles({run_id})&#34;: full_gradient_oracles,
           f&#34;samples gradient oracles({run_id})&#34;: samples_gradient_oracles,
           f&#34;send scalars to master({run_id})&#34;: send_scalars_to_master,
           f&#34;round({run_id})&#34;: round,
           f&#34;progress in percentage ({run_id})&#34;: float(round + 1.0) / H[&#34;args&#34;].rounds * 100.0
           }

    items_elements = [&#34;full_gradient_norm_train&#34;, &#34;x_before_round&#34;, &#34;approximate_f_avg_value&#34;, &#34;grad_sgd_server_l2&#34;,
                      &#34;full_objective_value_train&#34;, &#34;full_gradient_norm_val&#34;, &#34;full_objective_value_val&#34;, &#34;train_time&#34;,
                      &#34;number_of_client_in_round&#34;]

    for elem in items_elements:
        if elem in item.keys():
            element_name = elem.replace(&#39;_&#39;, &#39; &#39;)
            msg.update({f&#34;{element_name}({run_id})&#34;: item[elem]})

    if H[&#39;eval_metrics&#39;]:
        last_eval_round = max(H[&#39;eval_metrics&#39;].keys())
        last_eval_metrics = H[&#39;eval_metrics&#39;][last_eval_round]

        msg.update({f&#34;Loss validation({run_id})&#34;: last_eval_metrics[&#39;loss&#39;]})
        msg.update({f&#34;Top1 acc validation({run_id})&#34;: last_eval_metrics[&#39;top_1_acc&#39;]})
        msg.update({f&#34;Top5 acc validation({run_id})&#34;: last_eval_metrics[&#39;top_5_acc&#39;]})

    _locker.acquire()
    wandb.log(msg)
    _locker.release()</code></pre>
</details>
</section>
<section>
</section>
<section>
</section>
<section>
<h2 class="section-title" id="header-functions">Functions</h2>
<dl>
<dt id="utils.wandb_wrapper.finishProject"><code class="name flex">
<span>def <span class="ident">finishProject</span></span>(<span>projectRun)</span>
</code></dt>
<dd>
<div class="desc"><p>Thread safe way to deinitialize wand library, that is originally is not thread safe.
It will decrease counter, and once counter of library users will be zero it will perform final deinitialized.</p>
<h2 id="args">Args</h2>
<dl>
<dt><strong><code>projectRun</code></strong></dt>
<dd>Instance of the project from wandb library</dd>
</dl>
<h2 id="returns">Returns</h2>
<p>None</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def finishProject(projectRun):
    &#34;&#34;&#34;
    Thread safe way to deinitialize wand library, that is originally is not thread safe.
    It will decrease counter, and once counter of library users will be zero it will perform final deinitialized.

    Args:
        projectRun: Instance of the project from wandb library

    Returns:
        None
    &#34;&#34;&#34;

    global _is_init
    global _init_counter

    _locker.acquire()
    _init_counter -= 1
    if _init_counter == 0:
        if projectRun:
            projectRun.finish()
        _is_init = False
    _locker.release()</code></pre>
</details>
</dd>
<dt id="utils.wandb_wrapper.initWandbProject"><code class="name flex">
<span>def <span class="ident">initWandbProject</span></span>(<span>api_key, project, name, args)</span>
</code></dt>
<dd>
<div class="desc"><p>Thread safe way to initialize wand library, that is originally is not thread safe.
If library has already been initialized this method only increase counter
and does not perform any actual initialization.</p>
<h2 id="args">Args</h2>
<dl>
<dt>api_key(str): personal api key for the use</dt>
<dt>project(str): project name</dt>
<dt>name(str): experiment name</dt>
<dt><strong><code>args</code></strong></dt>
<dd>command line arguments</dd>
</dl>
<h2 id="returns">Returns</h2>
<p>Instance of the project from wandb library if library is initialized and None otherwise</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def initWandbProject(api_key, project, name, args):
    &#34;&#34;&#34;
    Thread safe way to initialize wand library, that is originally is not thread safe.
    If library has already been initialized this method only increase counter
    and does not perform any actual initialization.

    Args:
        api_key(str): personal api key for the use
        project(str): project name
        name(str): experiment name
        args: command line arguments

    Returns:
        Instance of the project from wandb library if library is initialized and None otherwise
    &#34;&#34;&#34;
    logger = Logger.get(args.run_id)

    global _is_init
    global _init_counter

    _locker.acquire()
    if _is_init:
        _init_counter += 1
        _locker.release()
        return None
    else:
        try:
            wandb.login(key=api_key)
            logger.info(f&#34;Wandb login completed successfully&#34;)
            run = wandb.init(
                # project=&#34;federated_nas&#34;,
                project=project,
                name=name,
                config=args,
                reinit=True
            )
            _is_init = True
            _init_counter += 1
            _locker.release()
            return run

        except ValueError as err:
            logger.error(f&#34;ERROR: Ignore Wandb login due to problems with login with API KEY: {str(err)}&#34;)
            _is_init = False
            _locker.release()
            return None</code></pre>
</details>
</dd>
<dt id="utils.wandb_wrapper.logStatistics"><code class="name flex">
<span>def <span class="ident">logStatistics</span></span>(<span>H, round)</span>
</code></dt>
<dd>
<div class="desc"><p>Log statistics from experiments and publish them via using wandb.
To distinguish between experiments thr run_id is embeded into plot names.</p>
<h2 id="args">Args</h2>
<p>H(dict): current server state
round(int): current round</p>
<h2 id="returns">Returns</h2>
<p>None</p></div>
<details class="source">
<summary>
<span>Expand source code</span>
</summary>
<pre><code class="python">def logStatistics(H, round):
    &#34;&#34;&#34;
    Log statistics from experiments and publish them via using wandb.
    To distinguish between experiments thr run_id is embeded into plot names.

    Args:
        H(dict): current server state
        round(int): current round

    Returns:
        None
    &#34;&#34;&#34;

    global _is_init
    if not _is_init:
        return

    item = H[&#39;history&#39;][round]

    full_gradient_oracles = \
        sum([v[&#39;client_state&#39;][&#39;stats&#39;][&#39;full_gradient_oracles&#39;] for k, v in item[&#34;client_states&#34;].items()])

    samples_gradient_oracles = \
        sum([v[&#39;client_state&#39;][&#39;stats&#39;][&#39;samples_gradient_oracles&#39;] for k, v in item[&#34;client_states&#34;].items()])

    send_scalars_to_master = \
        sum([v[&#39;client_state&#39;][&#39;stats&#39;][&#39;send_scalars_to_master&#39;] for k, v in item[&#34;client_states&#34;].items()])

    run_id = H[&#39;args&#39;].run_id

    # All quantities are per current round
    msg = {f&#34;full gradient oracles({run_id})&#34;: full_gradient_oracles,
           f&#34;samples gradient oracles({run_id})&#34;: samples_gradient_oracles,
           f&#34;send scalars to master({run_id})&#34;: send_scalars_to_master,
           f&#34;round({run_id})&#34;: round,
           f&#34;progress in percentage ({run_id})&#34;: float(round + 1.0) / H[&#34;args&#34;].rounds * 100.0
           }

    items_elements = [&#34;full_gradient_norm_train&#34;, &#34;x_before_round&#34;, &#34;approximate_f_avg_value&#34;, &#34;grad_sgd_server_l2&#34;,
                      &#34;full_objective_value_train&#34;, &#34;full_gradient_norm_val&#34;, &#34;full_objective_value_val&#34;, &#34;train_time&#34;,
                      &#34;number_of_client_in_round&#34;]

    for elem in items_elements:
        if elem in item.keys():
            element_name = elem.replace(&#39;_&#39;, &#39; &#39;)
            msg.update({f&#34;{element_name}({run_id})&#34;: item[elem]})

    if H[&#39;eval_metrics&#39;]:
        last_eval_round = max(H[&#39;eval_metrics&#39;].keys())
        last_eval_metrics = H[&#39;eval_metrics&#39;][last_eval_round]

        msg.update({f&#34;Loss validation({run_id})&#34;: last_eval_metrics[&#39;loss&#39;]})
        msg.update({f&#34;Top1 acc validation({run_id})&#34;: last_eval_metrics[&#39;top_1_acc&#39;]})
        msg.update({f&#34;Top5 acc validation({run_id})&#34;: last_eval_metrics[&#39;top_5_acc&#39;]})

    _locker.acquire()
    wandb.log(msg)
    _locker.release()</code></pre>
</details>
</dd>
</dl>
</section>
<section>
</section>
</article>
<nav id="sidebar">
<h1>Index</h1>
<div class="toc">
<ul></ul>
</div>
<ul id="index">
<li><h3><a href="#header-functions">Functions</a></h3>
<ul class="">
<li><code><a title="utils.wandb_wrapper.finishProject" href="#utils.wandb_wrapper.finishProject">finishProject</a></code></li>
<li><code><a title="utils.wandb_wrapper.initWandbProject" href="#utils.wandb_wrapper.initWandbProject">initWandbProject</a></code></li>
<li><code><a title="utils.wandb_wrapper.logStatistics" href="#utils.wandb_wrapper.logStatistics">logStatistics</a></code></li>
</ul>
</li>
</ul>
</nav>
</main>
<footer id="footer">
FL_PyTorch. The document generated 06-March-2023 18:58:57.
<p>Generated by <a href="https://pdoc3.github.io/pdoc" title="pdoc: Python API documentation generator"><cite>pdoc</cite> 0.10.0</a>.</p>
</footer>
</body>
</html>