{"info": {}, "config": {"looknfeel": "default", "personalizedMode": "false"}, "name": "cp/jupyter/house_building", "paragraphs": [{"settings": {"forms": {}, "params": {}}, "text": "%md\n\n# House Building with worker skills\n\nThis tutorial includes everything you need to set up decision optimization engines, build constraint programming models.\n\n\nWhen you finish this tutorial, you'll have a foundational knowledge of _Prescriptive Analytics_.\n\n>This notebook is part of the **[Prescriptive Analytics for Python](https://rawgit.com/IBMDecisionOptimization/docplex-doc/master/docs/index.html)**\n\n>It requires a valid subscription to **Decision Optimization on the Cloud** or a **local installation of CPLEX Optimizers**. \nDiscover us [here](https://developer.ibm.com/docloud)\n\n\nTable of contents:\n\n-  Describe the business problem\n*  How decision optimization (prescriptive analytics) can help\n*  Use decision optimization\n    *  Step 1: Download the library\n    *  Step 2: Set up the engines\n    -  Step 3: Model the Data\n    -  Step 4: Set up the prescriptive model\n        * Define the decision variables\n        * Express the business constraints\n        * Express the objective\n        * Solve with Decision Optimization solve service\n    *  Step 5: Investigate the solution and run an example analysis\n*  Summary\n****", "apps": [], "results": {"msg": [{"data": "<p></p>\n<h1>House Building with worker skills</h1>\n<p></p>\n<p>This tutorial includes everything you need to set up decision optimization engines, build constraint programming models.</p>\n<p></p>\n<p></p>\n<p>When you finish this tutorial, you'll have a foundational knowledge of <em>Prescriptive Analytics</em>.</p>\n<p></p>\n<blockquote>\n  <p>This notebook is part of the <strong><a href=\"https://rawgit.com/IBMDecisionOptimization/docplex-doc/master/docs/index.html\">Prescriptive Analytics for Python</a></strong></p>\n</blockquote>\n<p></p>\n<blockquote>\n  <p>It requires a valid subscription to <strong>Decision Optimization on the Cloud</strong> or a <strong>local installation of CPLEX Optimizers</strong>. </p>\n</blockquote>\n<p>Discover us <a href=\"https://developer.ibm.com/docloud\">here</a></p>\n<p></p>\n<p></p>\n<p>Table of contents:</p>\n<p></p>\n<ul>\n<li>Describe the business problem</li>\n</ul>\n<ul>\n<li>How decision optimization (prescriptive analytics) can help</li>\n</ul>\n<ul>\n<li>Use decision optimization</li>\n</ul>\n<pre><code>*  Step 1: Download the library\n</code></pre>\n<pre><code>*  Step 2: Set up the engines\n</code></pre>\n<pre><code>-  Step 3: Model the Data\n</code></pre>\n<pre><code>-  Step 4: Set up the prescriptive model\n</code></pre>\n<pre><code>    * Define the decision variables\n</code></pre>\n<pre><code>    * Express the business constraints\n</code></pre>\n<pre><code>    * Express the objective\n</code></pre>\n<pre><code>    * Solve with Decision Optimization solve service\n</code></pre>\n<pre><code>*  Step 5: Investigate the solution and run an example analysis\n</code></pre>\n<ul>\n<li>Summary</li>\n</ul>\n<hr />\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n### Describe the business problem\n\n* This is a problem of building five houses in different locations; the masonry, roofing, painting, etc. must be scheduled. Some tasks must necessarily take place before others and these requirements are expressed through precedence constraints.\n\n* There are three workers, and each worker has a given skill level for each task. Each task requires one worker; the worker assigned must have a non-null skill level for the task. A worker can be assigned to only one task at a time.\n\n* Each house has a deadline. \n\n* The objective is to maximize the skill levels of the workers assigned to the tasks.", "apps": [], "results": {"msg": [{"data": "<h3>Describe the business problem</h3>\n<p></p>\n<ul>\n<li>This is a problem of building five houses in different locations; the masonry, roofing, painting, etc. must be scheduled. Some tasks must necessarily take place before others and these requirements are expressed through precedence constraints.</li>\n</ul>\n<p></p>\n<ul>\n<li>There are three workers, and each worker has a given skill level for each task. Each task requires one worker; the worker assigned must have a non-null skill level for the task. A worker can be assigned to only one task at a time.</li>\n</ul>\n<p></p>\n<ul>\n<li>Each house has a deadline. </li>\n</ul>\n<p></p>\n<ul>\n<li>The objective is to maximize the skill levels of the workers assigned to the tasks.</li>\n</ul>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n*****\n## How  decision optimization can help\n* Prescriptive analytics technology recommends actions based on desired outcomes, taking into account specific scenarios, resources, and knowledge of past and current events. This insight can help your organization make better decisions and have greater control of business outcomes.  \n\n* Prescriptive analytics is the next step on the path to insight-based actions. It creates value through synergy with predictive analytics, which analyzes data to predict future outcomes.  \n\n* Prescriptive analytics takes that insight to the next level by suggesting the optimal way to handle that future situation. Organizations that can act fast in dynamic conditions and make superior decisions in uncertain environments gain a strong competitive advantage.  \n<br/>\n\n+ For example:\n    + Automate complex decisions and trade-offs to better manage limited resources.\n    + Take advantage of a future opportunity or mitigate a future risk.\n    + Proactively update recommendations based on changing events.\n    + Meet operational goals, increase customer loyalty, prevent threats and fraud, and optimize business processes.\n", "apps": [], "results": {"msg": [{"data": "<hr />\n<h2>How  decision optimization can help</h2>\n<ul>\n<li>Prescriptive analytics technology recommends actions based on desired outcomes, taking into account specific scenarios, resources, and knowledge of past and current events. This insight can help your organization make better decisions and have greater control of business outcomes.  </li>\n</ul>\n<p></p>\n<ul>\n<li>Prescriptive analytics is the next step on the path to insight-based actions. It creates value through synergy with predictive analytics, which analyzes data to predict future outcomes.  </li>\n</ul>\n<p></p>\n<ul>\n<li>Prescriptive analytics takes that insight to the next level by suggesting the optimal way to handle that future situation. Organizations that can act fast in dynamic conditions and make superior decisions in uncertain environments gain a strong competitive advantage.  </li>\n</ul>\n<p><br/></p>\n<p></p>\n<ul>\n<li>For example:</li>\n</ul>\n<pre><code>+ Automate complex decisions and trade-offs to better manage limited resources.\n</code></pre>\n<pre><code>+ Take advantage of a future opportunity or mitigate a future risk.\n</code></pre>\n<pre><code>+ Proactively update recommendations based on changing events.\n</code></pre>\n<pre><code>+ Meet operational goals, increase customer loyalty, prevent threats and fraud, and optimize business processes.\n</code></pre>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h2>About Detailed Scheduling concepts</h2>\n<p>\n<ul>\n<li> Scheduling consists of assigning starting and completion times to a set of activities while satisfying different types of constraints (resource availability, precedence relationships, \u2026 ) and optimizing some criteria (minimizing tardiness, \u2026)\n<!-- <img src = \"./house_building_utils/activity.png\" > -->\n<img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/activity.PNG?raw=true \" >\n<li> Time is considered as a continuous dimension: domain of possible start/completion times for an activity is potentially very large\n<li>Beside start and completion times of activities, other types of decision variables are often involved in real industrial scheduling problems (resource allocation, optional activities \u2026)\n</ul>", "apps": [], "results": {"msg": [{"data": "<h2>About Detailed Scheduling concepts</h2>\n<p><p></p>\n<p><ul></p>\n<p><li> Scheduling consists of assigning starting and completion times to a set of activities while satisfying different types of constraints (resource availability, precedence relationships, \u2026 ) and optimizing some criteria (minimizing tardiness, \u2026)</p>\n<!-- <img src = \"./house_building_utils/activity.png\" > -->\n<p><img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/activity.PNG?raw=true \" &gt;</p>\n<p><li> Time is considered as a continuous dimension: domain of possible start/completion times for an activity is potentially very large</p>\n<p><li>Beside start and completion times of activities, other types of decision variables are often involved in real industrial scheduling problems (resource allocation, optional activities \u2026)</p>\n<p></ul></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n## Use decision optimization", "apps": [], "results": {"msg": [{"data": "<h2>Use decision optimization</h2>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n### Step 1: Download the library\n\nRun the following code to install Decision Optimization CPLEX Modeling library.  The *DOcplex* library contains the two modeling packages, Mathematical Programming and Constraint Programming, referred to earlier.", "apps": [], "results": {"msg": [{"data": "<h3>Step 1: Download the library</h3>\n<p></p>\n<p>Run the following code to install Decision Optimization CPLEX Modeling library.  The <em>DOcplex</em> library contains the two modeling packages, Mathematical Programming and Constraint Programming, referred to earlier.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nimport sys\nimport pip\ntry:\n    import docplex.cp\nexcept:\n    if hasattr(sys, 'real_prefix'):\n        #we are in a virtual env.\n        pip.main(['install', docplex]) \n    else:\n        pip.main(['install --user', docplex])", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nNote that the more global package <i>docplex</i> contains another subpackage <i>docplex.mp</i> that is dedicated to Mathematical Programming, another branch of optimization.", "apps": [], "results": {"msg": [{"data": "<p>Note that the more global package <i>docplex</i> contains another subpackage <i>docplex.mp</i> that is dedicated to Mathematical Programming, another branch of optimization.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n### Step 2: Set up the prescriptive engine\n\n* Subscribe to the [Decision Optimization on Cloud solve service](https://developer.ibm.com/docloud).\n* Get the service URL and your personal API key.", "apps": [], "results": {"msg": [{"data": "<h3>Step 2: Set up the prescriptive engine</h3>\n<p></p>\n<ul>\n<li>Subscribe to the <a href=\"https://developer.ibm.com/docloud\">Decision Optimization on Cloud solve service</a>.</li>\n</ul>\n<ul>\n<li>Get the service URL and your personal API key.</li>\n</ul>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nSVC_URL = \"ENTER YOUR URL HERE\"\nSVC_KEY = \"ENTER YOUR KEY HERE\"", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nAnd for display of the solution, ensure last version of matplotlib is available:", "apps": [], "results": {"msg": [{"data": "<p>And for display of the solution, ensure last version of matplotlib is available:</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ntry:\n   import matplotlib\n   if matplotlib.__version__ < \"1.4.3\":\n        pip.main(['install --upgrade', matplotlib])\nexcept:\n   pip.main(['install --user', matplotlib])", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nNow, we need to import all required modeling functions that are provided by the <i>docplex.cp</i> package:", "apps": [], "results": {"msg": [{"data": "<p>Now, we need to import all required modeling functions that are provided by the <i>docplex.cp</i> package:</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nfrom docplex.cp.model import CpoModel\nfrom sys import stdout\nfrom collections import namedtuple", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n### Step 3: Model the data", "apps": [], "results": {"msg": [{"data": "<h3>Step 3: Model the data</h3>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nPlanning contains the number of houses and the max amount of periods (<i>days</i>) for our schedule", "apps": [], "results": {"msg": [{"data": "<p>Planning contains the number of houses and the max amount of periods (<i>days</i>) for our schedule</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nNB_HOUSES = 5\nMAX_AMOUNT_OF_PERIODS = 318\nHOUSES = range(1, NB_HOUSES + 1)", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nAll tasks must start and end between 0 and the max amount of periods", "apps": [], "results": {"msg": [{"data": "<p>All tasks must start and end between 0 and the max amount of periods</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nperiod_domain = (0, MAX_AMOUNT_OF_PERIODS)", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nFor each task type in the house building project, the following table shows the duration of the task in days along with the tasks that must be finished before the task can start. A worker can only work on one task at a time; each task, once started, may not be interrupted.\n<p>\n\n| *Task* | *Duration* | *Preceding tasks* |\n|---|---|---|\n| masonry \t| 35 |\t|\n| carpentry | 15 | masonry |\n| plumbing \t| 40 | masonry |\n| ceiling \t| 15 | masonry |\n| roofing \t| 5  | carpentry |\n| painting \t| 10 | ceiling |\n| windows \t| 5  | roofing |\n| facade \t| 10 | roofing, plumbing |\n| garden \t| 5  | roofing, plumbing |\n| moving \t| 5  | windows, facade, garden, painting | ", "apps": [], "results": {"msg": [{"data": "<p>For each task type in the house building project, the following table shows the duration of the task in days along with the tasks that must be finished before the task can start. A worker can only work on one task at a time; each task, once started, may not be interrupted.</p>\n<p><p></p>\n<p></p>\n<p>| <em>Task</em> | <em>Duration</em> | <em>Preceding tasks</em> |</p>\n<p>|---|---|---|</p>\n<p>| masonry   | 35 |  |</p>\n<p>| carpentry | 15 | masonry |</p>\n<p>| plumbing  | 40 | masonry |</p>\n<p>| ceiling   | 15 | masonry |</p>\n<p>| roofing   | 5  | carpentry |</p>\n<p>| painting  | 10 | ceiling |</p>\n<p>| windows   | 5  | roofing |</p>\n<p>| facade    | 10 | roofing, plumbing |</p>\n<p>| garden    | 5  | roofing, plumbing |</p>\n<p>| moving    | 5  | windows, facade, garden, painting | </p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n##### Tasks' durations", "apps": [], "results": {"msg": [{"data": "<h5>Tasks' durations</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nTask = (namedtuple(\"Task\", [\"name\", \"duration\"]))\nTASKS = {Task(\"masonry\",   35),\n         Task(\"carpentry\", 15),\n         Task(\"plumbing\",  40),\n         Task(\"ceiling\",   15),\n         Task(\"roofing\",    5),\n         Task(\"painting\",  10),\n         Task(\"windows\",    5),\n         Task(\"facade\",    10),\n         Task(\"garden\",     5),\n         Task(\"moving\",     5),\n        }", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n##### The tasks precedences", "apps": [], "results": {"msg": [{"data": "<h5>The tasks precedences</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nTaskPrecedence = (namedtuple(\"TaskPrecedence\", [\"beforeTask\", \"afterTask\"]))\nTASK_PRECEDENCES = {TaskPrecedence(\"masonry\",   \"carpentry\"),\n                    TaskPrecedence(\"masonry\",   \"plumbing\"),\n                    TaskPrecedence(\"masonry\",   \"ceiling\"),\n                    TaskPrecedence(\"carpentry\", \"roofing\"),\n                    TaskPrecedence(\"ceiling\",   \"painting\"),\n                    TaskPrecedence(\"roofing\",   \"windows\"),\n                    TaskPrecedence(\"roofing\",   \"facade\"),\n                    TaskPrecedence(\"plumbing\",  \"facade\"),\n                    TaskPrecedence(\"roofing\",   \"garden\"),\n                    TaskPrecedence(\"plumbing\",  \"garden\"),\n                    TaskPrecedence(\"windows\",   \"moving\"),\n                    TaskPrecedence(\"facade\",    \"moving\"),\n                    TaskPrecedence(\"garden\",    \"moving\"),\n                    TaskPrecedence(\"painting\",  \"moving\"),\n                   }", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nThere are three workers with varying skill levels in regard to the ten tasks. If a worker has a skill level of zero for a task, he may not be assigned to the task.\n<p>\n\n| *Task* | *Joe* | *Jack* | *Jim* |\n|---|---|---|---|\n|masonry   |9 |\t5 |\t0|\n|carpentry |7 |\t0 |\t5|\n|plumbing  |0 |\t7 |\t0|\n|ceiling   |5 |\t8 |\t0|\n|roofing   |6 |\t7 |\t0|\n|painting  |0 |\t9 |\t6|\n|windows   |8 |\t0 |\t5|\n|fa\u00e7ade    |5 |\t5 |\t0|\n|garden    |5 |\t5 |\t9|\n|moving    |6 |\t0 |\t8|", "apps": [], "results": {"msg": [{"data": "<p>There are three workers with varying skill levels in regard to the ten tasks. If a worker has a skill level of zero for a task, he may not be assigned to the task.</p>\n<p><p></p>\n<p></p>\n<p>| <em>Task</em> | <em>Joe</em> | <em>Jack</em> | <em>Jim</em> |</p>\n<p>|---|---|---|---|</p>\n<p>|masonry   |9 | 5 | 0|</p>\n<p>|carpentry |7 | 0 | 5|</p>\n<p>|plumbing  |0 | 7 | 0|</p>\n<p>|ceiling   |5 | 8 | 0|</p>\n<p>|roofing   |6 | 7 | 0|</p>\n<p>|painting  |0 | 9 | 6|</p>\n<p>|windows   |8 | 0 | 5|</p>\n<p>|fa\u00e7ade    |5 | 5 | 0|</p>\n<p>|garden    |5 | 5 | 9|</p>\n<p>|moving    |6 | 0 | 8|</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n##### Workers Names", "apps": [], "results": {"msg": [{"data": "<h5>Workers Names</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nWORKERS = {\"Joe\", \"Jack\", \"Jim\"}", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n##### Workers Name and level for each of there skill", "apps": [], "results": {"msg": [{"data": "<h5>Workers Name and level for each of there skill</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nSkill = (namedtuple(\"Skill\", [\"worker\", \"task\", \"level\"]))\nSKILLS = {Skill(\"Joe\",  \"masonry\",   9),\n          Skill(\"Joe\",  \"carpentry\", 7),\n          Skill(\"Joe\",  \"ceiling\",   5),\n          Skill(\"Joe\",  \"roofing\",   6),\n          Skill(\"Joe\",  \"windows\",   8),\n          Skill(\"Joe\",  \"facade\",    5),\n          Skill(\"Joe\",  \"garden\",    5),\n          Skill(\"Joe\",  \"moving\",    6),\n          Skill(\"Jack\", \"masonry\",   5),\n          Skill(\"Jack\", \"plumbing\",  7),\n          Skill(\"Jack\", \"ceiling\",   8),\n          Skill(\"Jack\", \"roofing\",   7),\n          Skill(\"Jack\", \"painting\",  9),\n          Skill(\"Jack\", \"facade\",    5),\n          Skill(\"Jack\", \"garden\",    5),\n          Skill(\"Jim\",  \"carpentry\", 5),\n          Skill(\"Jim\",  \"painting\",  6),\n          Skill(\"Jim\",  \"windows\",   5),\n          Skill(\"Jim\",  \"garden\",    9),\n          Skill(\"Jim\",  \"moving\",    8)\n          }", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n##### Utility functions", "apps": [], "results": {"msg": [{"data": "<h5>Utility functions</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nfind_tasks: returns the task it refers to in the TASKS vector", "apps": [], "results": {"msg": [{"data": "<p>find_tasks: returns the task it refers to in the TASKS vector</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ndef find_tasks(name):\n    return next(t for t in TASKS if t.name == name)", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nfind_skills: returns the skill it refers to in the SKILLS vector", "apps": [], "results": {"msg": [{"data": "<p>find_skills: returns the skill it refers to in the SKILLS vector</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ndef find_skills(worker, task):\n    return next(s for s in SKILLS if (s.worker == worker) and (s.task == task))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nfind_max_level_skill: returns the tuple \"skill\" where the level is themaximum for a given task", "apps": [], "results": {"msg": [{"data": "<p>find<em>max</em>level_skill: returns the tuple \"skill\" where the level is themaximum for a given task</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ndef find_max_level_skill(task):\n    st = [s for s in SKILLS if s.task == task]\n    return next(sk for sk in st if sk.level == max([s.level for s in st]))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n### Step 4: Set up the prescriptive model", "apps": [], "results": {"msg": [{"data": "<h3>Step 4: Set up the prescriptive model</h3>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h3>Create the model container</h3>\n<p>\nThe model is represented by a Python object that is filled with the different model elements (variables, constraints, objective function, etc). The first thing to do is then to create such an object:", "apps": [], "results": {"msg": [{"data": "<h3>Create the model container</h3>\n<p><p></p>\n<p>The model is represented by a Python object that is filled with the different model elements (variables, constraints, objective function, etc). The first thing to do is then to create such an object:</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nmdl = CpoModel(name=\"HouseBuilding\")", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Define the decision variables", "apps": [], "results": {"msg": [{"data": "<h4>Define the decision variables</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5><i><font color=blue>Concept: interval variable</font></i></h5>\n<p>\n<ul>\n<li> What for?<br>\n<blockquote> Modeling an interval of time during which a particular property holds <br>\n(an activity executes, a resource is idle, a tank must be non-empty, \u2026)</blockquote> \n<li> Example:<br>\n<blockquote><code><font color=green>interval_var(start=(0,1000), end=(0,1000), size=(10,20))</font></code>\n</blockquote> \n<!-- <img src = \"./house_building_utils/intervalVar.png\" > -->\n<img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/intervalVar.PNG?raw=true\" >\n<li>Properties:\n<ul>\n<li>The **value** of an interval variable is an integer interval [start,end) \n<li>**Domain**  of possible values: [0,10), [1,11), [2,12),...[990,1000), [0,11),[1,12),...\n<li>Domain of interval variables is represented **compactly** in CP Optimizer (a few bounds: smin, smax, emin, emax, szmin, szmax)\n</ul>\n</ul>", "apps": [], "results": {"msg": [{"data": "<h5><i><font color=blue>Concept: interval variable</font></i></h5>\n<p><p></p>\n<p><ul></p>\n<p><li> What for?<br></p>\n<p><blockquote> Modeling an interval of time during which a particular property holds <br></p>\n<p>(an activity executes, a resource is idle, a tank must be non-empty, \u2026)</blockquote> </p>\n<p><li> Example:<br></p>\n<p><blockquote><code><font color=green>interval_var(start=(0,1000), end=(0,1000), size=(10,20))</font></code></p>\n<p></blockquote> </p>\n<!-- <img src = \"./house_building_utils/intervalVar.png\" > -->\n<p><img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/intervalVar.PNG?raw=true\" &gt;</p>\n<p><li>Properties:</p>\n<p><ul></p>\n<p><li>The <strong>value</strong> of an interval variable is an integer interval [start,end) </p>\n<p><li><strong>Domain</strong>  of possible values: [0,10), [1,11), [2,12),...[990,1000), [0,11),[1,12),...</p>\n<p><li>Domain of interval variables is represented <strong>compactly</strong> in CP Optimizer (a few bounds: smin, smax, emin, emax, szmin, szmax)</p>\n<p></ul></p>\n<p></ul></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nFor each house, an interval variable is created for each task.<br>\nThis interval must start and end inside the period_domain and its duration is set as the value stated in TASKS definition.", "apps": [], "results": {"msg": [{"data": "<p>For each house, an interval variable is created for each task.<br></p>\n<p>This interval must start and end inside the period_domain and its duration is set as the value stated in TASKS definition.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ntasks = {}   # dict of interval variable for each house and task\nfor house in HOUSES:\n    for task in TASKS:\n        tasks[(house, task)] = mdl.interval_var(start=period_domain,\n                                                end=period_domain,\n                                                size=task.duration,\n                                                name=\"house {} task {}\".format(house, task))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5><i><font color=blue>Concept: optional interval variable</font></i></h5>\n<p>\n<ul>\n<li>Interval variables can be defined as being **optional** that is, it is part of the decisions of the problem to decide whether the interval will be **present** or **absent** in the solution<br>\n<li> What for?<br>\n<blockquote> Modeling optional activities, alternative execution modes for activities, and \u2026 most of the discrete decisions in a schedule</blockquote> \n<li> Example:<br>\n<blockquote><code><font color=green>interval_var(</font><font color=red>optional=True</font><font color=green>, start=(0,1000), end=(0,1000), size=(10,20))</font></code>\n</blockquote> \n<li>Properties:\n<ul>\n<li>An optional interval variable has an additional possible value in its domain (absence value)\n<li>**Optionality** is a powerful property that you must learn to leverage in your models\n</ul>\n</ul>", "apps": [], "results": {"msg": [{"data": "<h5><i><font color=blue>Concept: optional interval variable</font></i></h5>\n<p><p></p>\n<p><ul></p>\n<p><li>Interval variables can be defined as being <strong>optional</strong> that is, it is part of the decisions of the problem to decide whether the interval will be <strong>present</strong> or <strong>absent</strong> in the solution<br></p>\n<p><li> What for?<br></p>\n<blockquote> Modeling optional activities, alternative execution modes for activities, and \u2026 most of the discrete decisions in a schedule</blockquote> \n<p><li> Example:<br></p>\n<p><blockquote><code><font color=green>interval_var(</font><font color=red>optional=True</font><font color=green>, start=(0,1000), end=(0,1000), size=(10,20))</font></code></p>\n<p></blockquote> </p>\n<p><li>Properties:</p>\n<p><ul></p>\n<p><li>An optional interval variable has an additional possible value in its domain (absence value)</p>\n<p><li><strong>Optionality</strong> is a powerful property that you must learn to leverage in your models</p>\n<p></ul></p>\n<p></ul></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nFor each house, an __optional__ interval variable is created for each skill.<br>\nSkill being a tuple (worker, task, level), this means that for each house, an __optional__ interval variable is created for each couple worker-task such that the skill level of this worker for this task is > 0.<p>\nThe \"**set_optional()**\" specifier allows a choice between different variables, thus between different couples house-skill.\nThis means that the engine decides if the interval will be present or absent in the solution.", "apps": [], "results": {"msg": [{"data": "<p>For each house, an <strong>optional</strong> interval variable is created for each skill.<br></p>\n<p>Skill being a tuple (worker, task, level), this means that for each house, an <strong>optional</strong> interval variable is created for each couple worker-task such that the skill level of this worker for this task is &gt; 0.<p></p>\n<p>The \"<strong>set_optional()</strong>\" specifier allows a choice between different variables, thus between different couples house-skill.</p>\n<p>This means that the engine decides if the interval will be present or absent in the solution.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nwtasks = {}  # dict of interval variable for each house and skill\nfor house in HOUSES:\n    for skill in SKILLS:\n        iv = mdl.interval_var(name='H' + str(house) + '-' + skill.task + '(' + skill.worker + ')')\n        iv.set_optional()\n        wtasks[(house, skill)] = iv", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Express the business constraints", "apps": [], "results": {"msg": [{"data": "<h4>Express the business constraints</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5>Temporal constraints</h5>", "apps": [], "results": {"msg": [{"data": "<h5>Temporal constraints</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5><i><font color=blue>Concept: precedence constraint</font></i></h5>\n<p>\n<ul>\n<li> What for?<br>\n<ul>\n<li>Modeling temporal constraints between interval variables\n<li>Modeling constant or variable minimal delays\n</ul>\n<li>Properties\n<blockquote>Semantic of the constraints handles optionality (as for all constraints in CP Optimizer).<br>\nExample of endBeforeStart:<br>\n<code><font color=green>end_before_start(a,b,z)</font></code><br>\npresent(a) <font color=red>AND</font> present(b)  &Implies;  end(a)+z &LessSlantEqual; start(b) \n</blockquote>\n<ul>", "apps": [], "results": {"msg": [{"data": "<h5><i><font color=blue>Concept: precedence constraint</font></i></h5>\n<p><p></p>\n<p><ul></p>\n<p><li> What for?<br></p>\n<p><ul></p>\n<p><li>Modeling temporal constraints between interval variables</p>\n<p><li>Modeling constant or variable minimal delays</p>\n<p></ul></p>\n<p><li>Properties</p>\n<p><blockquote>Semantic of the constraints handles optionality (as for all constraints in CP Optimizer).<br></p>\n<p>Example of endBeforeStart:<br></p>\n<p><code><font color=green>end<em>before</em>start(a,b,z)</font></code><br></p>\n<p>present(a) <font color=red>AND</font> present(b)  &Implies;  end(a)+z &LessSlantEqual; start(b) </p>\n<p></blockquote></p>\n<p><ul></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nThe tasks in the model have precedence constraints that are added to the model.", "apps": [], "results": {"msg": [{"data": "<p>The tasks in the model have precedence constraints that are added to the model.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nfor h in HOUSES:\n    for p in TASK_PRECEDENCES:\n        mdl.add(mdl.end_before_start(tasks[(h, find_tasks(p.beforeTask))], tasks[(h, find_tasks(p.afterTask))]))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5>Alternative workers</h5>", "apps": [], "results": {"msg": [{"data": "<h5>Alternative workers</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5><i><font color=blue>Concept: alternative constraint</font></i></h5>\n<p>\n<ul>\n<li> What for?<br>\n<ul>\n<li>Modeling alternative resource/modes/recipes \n<li>In general modeling a discrete selection in the schedule \n</ul>\n<li> Example:<br>\n<blockquote><code><font color=green>alternative(a,[b1,...,bn])</font></code>\n</blockquote> \n<!-- <img src = \"./house_building_utils/alternative.png\" > -->\n<img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/alternative.PNG?raw=true\" >\n<li>Remark: Master interval variable **a** can of course be optional\n</ul>", "apps": [], "results": {"msg": [{"data": "<h5><i><font color=blue>Concept: alternative constraint</font></i></h5>\n<p><p></p>\n<p><ul></p>\n<p><li> What for?<br></p>\n<p><ul></p>\n<p><li>Modeling alternative resource/modes/recipes </p>\n<p><li>In general modeling a discrete selection in the schedule </p>\n<p></ul></p>\n<p><li> Example:<br></p>\n<p><blockquote><code><font color=green>alternative(a,[b1,...,bn])</font></code></p>\n<p></blockquote> </p>\n<!-- <img src = \"./house_building_utils/alternative.png\" > -->\n<p><img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/alternative.PNG?raw=true\" &gt;</p>\n<p><li>Remark: Master interval variable <strong>a</strong> can of course be optional</p>\n<p></ul></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nTo constrain the solution so that exactly one of the interval variables wtasks associated with a given task of a given house is to be present in the solution, an \"**alternative**\" constraint is used.", "apps": [], "results": {"msg": [{"data": "<p>To constrain the solution so that exactly one of the interval variables wtasks associated with a given task of a given house is to be present in the solution, an \"<strong>alternative</strong>\" constraint is used.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nfor h in HOUSES:\n    for t in TASKS:\n        mdl.add(mdl.alternative(tasks[(h, t)], [wtasks[(h, s)] for s in SKILLS if (s.task == t.name)], 1))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5>No overlap constraint</h5>", "apps": [], "results": {"msg": [{"data": "<h5>No overlap constraint</h5>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h5><i><font color=blue>Concept: No-overlap constraint</font></i></h5>\n<p>\n<ul>\n<li> Constraint noOverlap schedules a group of interval variables in such a way that they do not overlap in time.\n<li> Absent interval variables are ignored.\n<li>It is possible to constrain minimum delays between intervals using transition matrix.\n<li>It is possible to constraint the first, last in the sequence or next or preceding interval\n</ul>\n<!-- <img src = \"./house_building_utils/noOverlap.png\" > -->\n<img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/noOverlap.PNG?raw=true\" >", "apps": [], "results": {"msg": [{"data": "<h5><i><font color=blue>Concept: No-overlap constraint</font></i></h5>\n<p><p></p>\n<p><ul></p>\n<p><li> Constraint noOverlap schedules a group of interval variables in such a way that they do not overlap in time.</p>\n<p><li> Absent interval variables are ignored.</p>\n<p><li>It is possible to constrain minimum delays between intervals using transition matrix.</p>\n<p><li>It is possible to constraint the first, last in the sequence or next or preceding interval</p>\n<p></ul></p>\n<!-- <img src = \"./house_building_utils/noOverlap.png\" > -->\n<p><img src = \"https://github.com/IBMDecisionOptimization/docplex-examples/blob/master/examples/cp/jupyter/house_building_utils/noOverlap.PNG?raw=true\" &gt;</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nTo add the constraints that a given worker can be assigned only one task at a given moment in time, a  **noOverlap** constraint is used.", "apps": [], "results": {"msg": [{"data": "<p>To add the constraints that a given worker can be assigned only one task at a given moment in time, a  <strong>noOverlap</strong> constraint is used.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nfor w in WORKERS:\n    mdl.add(mdl.no_overlap([wtasks[(h, s)] for h in HOUSES for s in SKILLS if s.worker == w]))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Express the objective", "apps": [], "results": {"msg": [{"data": "<h4>Express the objective</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nThe presence of an interval variable in wtasks in the solution must be accounted for in the objective. Thus for each of these possible tasks, the cost is incremented by the product of the skill level and the expression representing the presence of the interval variable in the solution.<p>\nThe objective of this problem is to maximize the skill level used for all the tasks.", "apps": [], "results": {"msg": [{"data": "<p>The presence of an interval variable in wtasks in the solution must be accounted for in the objective. Thus for each of these possible tasks, the cost is incremented by the product of the skill level and the expression representing the presence of the interval variable in the solution.<p></p>\n<p>The objective of this problem is to maximize the skill level used for all the tasks.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nobj = mdl.sum([s.level * mdl.presence_of(wtasks[(h, s)]) for s in SKILLS for h in HOUSES])\nmdl.add(mdl.maximize(obj))", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nThe model is now completely defined. It is time to solve it !\n<p>\nTo use the CP Optimizer solver available on the IBM Decision Optimization on Cloud service:\n<ul>\n<li> Register for the DOcplexcloud free trial and use it free for 30 days by using https://developer.ibm.com/docloud/try-docloud-free\n<li> Get your access credentials (base URL and access key) by going this page: http://developer.ibm.com/docloud/docs/api-key/\n</ul>\n<p>", "apps": [], "results": {"msg": [{"data": "<p>The model is now completely defined. It is time to solve it !</p>\n<p><p></p>\n<p>To use the CP Optimizer solver available on the IBM Decision Optimization on Cloud service:</p>\n<p><ul></p>\n<p><li> Register for the DOcplexcloud free trial and use it free for 30 days by using https://developer.ibm.com/docloud/try-docloud-free</p>\n<p><li> Get your access credentials (base URL and access key) by going this page: http://developer.ibm.com/docloud/docs/api-key/</p>\n<p></ul></p>\n<p><p></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Solve with Decision Optimization solve service", "apps": [], "results": {"msg": [{"data": "<h4>Solve with Decision Optimization solve service</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\n# Solve the model\nprint(\"\\nSolving model....\")\nmsol = mdl.solve(url=SVC_URL, key=SVC_KEY, TimeLimit=10)", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n### Step 5: Investigate the solution and then run an example analysis", "apps": [], "results": {"msg": [{"data": "<h3>Step 5: Investigate the solution and then run an example analysis</h3>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nprint(\"Solve status: \" + msol.get_solve_status())\nif msol.is_solution():\n    stdout.write(\"Solve time: \" + str(msol.get_solve_time()) + \"\\n\")\n    # Sort tasks in increasing begin order\n    ltasks = []\n    for hs in HOUSES:\n        for tsk in TASKS:\n            (beg, end, dur) = msol[tasks[(hs, tsk)]]\n            ltasks.append((hs, tsk, beg, end, dur))\n    ltasks = sorted(ltasks, key = lambda x : x[2])\n    # Print solution\n    print(\"\\nList of tasks in increasing start order:\")\n    for tsk in ltasks:\n        print(\"From \" + str(tsk[2]) + \" to \" + str(tsk[3]) + \", \" + tsk[1].name + \" in house \" + str(tsk[0]))\nelse:\n    stdout.write(\"No solution found\\n\")", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Import graphical tools", "apps": [], "results": {"msg": [{"data": "<h4>Import graphical tools</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n*You can set __POP\\_UP\\_GRAPHIC=True__ if you prefer a pop up graphic window instead of an inline one.*", "apps": [], "results": {"msg": [{"data": "<p><em>You can set <strong>POP_UP_GRAPHIC=True</strong> if you prefer a pop up graphic window instead of an inline one.</em></p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nPOP_UP_GRAPHIC=False", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nimport docplex.cp.utils_visu as visu\nimport matplotlib.pyplot as plt\nif not POP_UP_GRAPHIC:\n    %matplotlib inline\n#Change the plot size\nfrom pylab import rcParams\nrcParams['figure.figsize'] = 15, 3", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Draw solution", "apps": [], "results": {"msg": [{"data": "<h4>Draw solution</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### Useful functions ", "apps": [], "results": {"msg": [{"data": "<h4>Useful functions</h4>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nWith the aim to facilitate the display of tasks names, we keep only the n first characters.", "apps": [], "results": {"msg": [{"data": "<p>With the aim to facilitate the display of tasks names, we keep only the n first characters.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ndef compact_name(name,n): return name[:n]", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nif msol and visu.is_visu_enabled():\n    workers_colors = {}\n    workers_colors[\"Joe\"] = 'lightblue'\n    workers_colors[\"Jack\"] = 'violet'\n    workers_colors[\"Jim\"] = 'lightgreen'\n    visu.timeline('Solution per houses', 0, MAX_AMOUNT_OF_PERIODS)\n    for h in HOUSES:\n        visu.sequence(name=\"house \" + str(h))\n        for s in SKILLS:\n            wt = msol.get_var_solution(wtasks[(h,s)])\n            if wt.is_present():\n                color = workers_colors[s.worker]\n                wtname = compact_name(s.task,2)\n                visu.interval(wt, color, wtname)\n    visu.show()", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nThe purpose of this function is to compact the names of the different tasks with the aim of making the graphical display readable. </p>\nFor example \"H3-garden\" becomes \"G3\"", "apps": [], "results": {"msg": [{"data": "<p>The purpose of this function is to compact the names of the different tasks with the aim of making the graphical display readable. </p></p>\n<p>For example \"H3-garden\" becomes \"G3\"</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\ndef compact_house_task(name):\n    loc, task = name[1:].split('-', 1)\n    return task[0].upper() + loc", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nGreen-like color when task is using the most skilled worker\nRed-like color when task does not use the most skilled worker", "apps": [], "results": {"msg": [{"data": "<p>Green-like color when task is using the most skilled worker</p>\n<p>Red-like color when task does not use the most skilled worker</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%python\nif msol and visu.is_visu_enabled():\n    visu.timeline('Solution per workers', 0, MAX_AMOUNT_OF_PERIODS)\n    for w in WORKERS:\n        visu.sequence(name=w)\n        for h in HOUSES:\n            for s in SKILLS:\n                if s.worker == w:\n                    wt = msol.get_var_solution(wtasks[(h,s)])\n                    if wt.is_present():\n                        ml = find_max_level_skill(s.task).level\n                        if s.level == ml:\n                            color = 'lightgreen'\n                        else:\n                            color = 'salmon'\n                        wtname = compact_house_task(wt.get_name())\n                        visu.interval(wt, color, wtname)\n    visu.show()", "apps": [], "results": {"msg": [{"data": "", "type": "ANGULAR"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": false, "language": "python"}, "editorMode": "ace/mode/python", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n<h4>Going further with Constraint Programming</h4>\n\nThe last available installable package is available on Pypi here: https://pypi.python.org/pypi/docplex\n\nA complete set of modeling examples can be downloaded here: https://github.com/IBMDecisionOptimization/docplex-examples  ", "apps": [], "results": {"msg": [{"data": "<h4>Going further with Constraint Programming</h4>\n<p></p>\n<p>The last available installable package is available on Pypi here: https://pypi.python.org/pypi/docplex</p>\n<p></p>\n<p>A complete set of modeling examples can be downloaded here: https://github.com/IBMDecisionOptimization/docplex-examples  </p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n## Summary\n\nYou learned how to set up and use the IBM Decision Optimization CPLEX Modeling for Python to formulate a Constraint Programming model and solve it with IBM Decision Optimization on the cloud.", "apps": [], "results": {"msg": [{"data": "<h2>Summary</h2>\n<p></p>\n<p>You learned how to set up and use the IBM Decision Optimization CPLEX Modeling for Python to formulate a Constraint Programming model and solve it with IBM Decision Optimization on the cloud.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\n#### References\n* [CPLEX Modeling for Python documentation](https://rawgit.com/IBMDecisionOptimization/docplex-doc/master/docs/index.html)\n* [Decision Optimization on Cloud](https://developer.ibm.com/docloud/)\n* Need help with DOcplex or to report a bug? Please go [here](https://developer.ibm.com/answers/smartspace/docloud)\n* Contact us at dofeedback@wwpdl.vnet.ibm.com", "apps": [], "results": {"msg": [{"data": "<h4>References</h4>\n<ul>\n<li><a href=\"https://rawgit.com/IBMDecisionOptimization/docplex-doc/master/docs/index.html\">CPLEX Modeling for Python documentation</a></li>\n</ul>\n<ul>\n<li><a href=\"https://developer.ibm.com/docloud/\">Decision Optimization on Cloud</a></li>\n</ul>\n<ul>\n<li>Need help with DOcplex or to report a bug? Please go <a href=\"https://developer.ibm.com/answers/smartspace/docloud\">here</a></li>\n</ul>\n<ul>\n<li>Contact us at dofeedback@wwpdl.vnet.ibm.com</li>\n</ul>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}, {"settings": {"forms": {}, "params": {}}, "text": "%md\nCopyright \u00a9 2017 IBM. IPLA licensed Sample Materials.", "apps": [], "results": {"msg": [{"data": "<p>Copyright \u00a9 2017 IBM. IPLA licensed Sample Materials.</p>\n", "type": "HTML"}], "code": "SUCCESS"}, "user": "anonymous", "config": {"editorSetting": {"editOnDblClick": true, "language": "markdown"}, "editorMode": "ace/mode/markdown", "colWidth": 12, "enabled": true, "results": {}}}]}