Basic Cycling§

In this section we will look at how to write cycling (repeating) workflows.

Repeating Workflows§

To make a workflow repeat we must tell Cylc three things:

Repeating Workflows§

 [scheduling]
+    cycling mode = integer
+    initial cycle point = 1
     [[dependencies]]
+        [[[P1]]]
             graph = """
                 purchase_ingredients => make_dough
                 pre_heat_oven & make_dough => bake_bread => sell_bread & clean_oven
             """

Repeating Workflows§

digraph example { size = "7,15" subgraph cluster_1 { label = 1 style = dashed "pur.1" [label="purchase_ingredients\n1"] "mak.1" [label="make_dough\n1"] "bak.1" [label="bake_bread\n1"] "sel.1" [label="sell_bread\n1"] "cle.1" [label="clean_oven\n1"] "pre.1" [label="pre_heat_oven\n1"] } subgraph cluster_2 { label = 2 style = dashed "pur.2" [label="purchase_ingredients\n2"] "mak.2" [label="make_dough\n2"] "bak.2" [label="bake_bread\n2"] "sel.2" [label="sell_bread\n2"] "cle.2" [label="clean_oven\n2"] "pre.2" [label="pre_heat_oven\n2"] } subgraph cluster_3 { label = 3 style = dashed "pur.3" [label="purchase_ingredients\n3"] "mak.3" [label="make_dough\n3"] "bak.3" [label="bake_bread\n3"] "sel.3" [label="sell_bread\n3"] "cle.3" [label="clean_oven\n3"] "pre.3" [label="pre_heat_oven\n3"] } "pur.1" -> "mak.1" -> "bak.1" -> "sel.1" "pre.1" -> "bak.1" -> "cle.1" "pur.2" -> "mak.2" -> "bak.2" -> "sel.2" "pre.2" -> "bak.2" -> "cle.2" "pur.3" -> "mak.3" -> "bak.3" -> "sel.3" "pre.3" -> "bak.3" -> "cle.3" }

Inter-Cycle Dependencies§

 [scheduling]
     cycling mode = integer
     initial cycle point = 1
     [[dependencies]]
         [[[P1]]]
             graph = """
                 purchase_ingredients => make_dough
                 pre_heat_oven & make_dough => bake_bread => sell_bread & clean_oven
+                clean_oven[-P1] => pre_heat_oven
             """

Inter-Cycle Dependencies§

digraph example { size = "7,15" subgraph cluster_1 { label = 1 style = dashed "pur.1" [label="purchase_ingredients\n1"] "mak.1" [label="make_dough\n1"] "bak.1" [label="bake_bread\n1"] "sel.1" [label="sell_bread\n1"] "cle.1" [label="clean_oven\n1"] "pre.1" [label="pre_heat_oven\n1"] } subgraph cluster_2 { label = 2 style = dashed "pur.2" [label="purchase_ingredients\n2"] "mak.2" [label="make_dough\n2"] "bak.2" [label="bake_bread\n2"] "sel.2" [label="sell_bread\n2"] "cle.2" [label="clean_oven\n2"] "pre.2" [label="pre_heat_oven\n2"] } subgraph cluster_3 { label = 3 style = dashed "pur.3" [label="purchase_ingredients\n3"] "mak.3" [label="make_dough\n3"] "bak.3" [label="bake_bread\n3"] "sel.3" [label="sell_bread\n3"] "cle.3" [label="clean_oven\n3"] "pre.3" [label="pre_heat_oven\n3"] } "pur.1" -> "mak.1" -> "bak.1" -> "sel.1" "pre.1" -> "bak.1" -> "cle.1" "cle.1" -> "pre.2" "pur.2" -> "mak.2" -> "bak.2" -> "sel.2" "pre.2" -> "bak.2" -> "cle.2" "cle.2" -> "pre.3" "pur.3" -> "mak.3" -> "bak.3" -> "sel.3" "pre.3" -> "bak.3" -> "cle.3" }

Inter-Cycle Dependencies§

 [scheduling]
     cycling mode = integer
     initial cycle point = 1
     [[dependencies]]
         [[[P1]]]
             graph = """
                 purchase_ingredients => make_dough
                 pre_heat_oven & make_dough => bake_bread => sell_bread & clean_oven
                 clean_oven[-P1] => pre_heat_oven
+                sell_bread[-P2] => purchase_ingredients
             """

Inter-Cycle Dependencies§

Note

The [-P2] suffix is used to reference a task two cycles before. For the first two cycles this doesn’t make sense as there was no cycle two cycles before, so this dependency will be ignored.

Any inter-cycle dependencies stretching back to before the initial cycle point will be ignored.

digraph example { size = "4.5,15" subgraph cluster_1 { label = 1 style = dashed "pur.1" [label="purchase_ingredients\n1"] "mak.1" [label="make_dough\n1"] "bak.1" [label="bake_bread\n1"] "sel.1" [label="sell_bread\n1"] "cle.1" [label="clean_oven\n1"] "pre.1" [label="pre_heat_oven\n1"] } subgraph cluster_2 { label = 2 style = dashed "pur.2" [label="purchase_ingredients\n2"] "mak.2" [label="make_dough\n2"] "bak.2" [label="bake_bread\n2"] "sel.2" [label="sell_bread\n2"] "cle.2" [label="clean_oven\n2"] "pre.2" [label="pre_heat_oven\n2"] } subgraph cluster_3 { label = 3 style = dashed "pur.3" [label="purchase_ingredients\n3"] "mak.3" [label="make_dough\n3"] "bak.3" [label="bake_bread\n3"] "sel.3" [label="sell_bread\n3"] "cle.3" [label="clean_oven\n3"] "pre.3" [label="pre_heat_oven\n3"] } subgraph cluster_4 { label = 4 style = dashed "pur.4" [label="purchase_ingredients\n4"] "mak.4" [label="make_dough\n4"] "bak.4" [label="bake_bread\n4"] "sel.4" [label="sell_bread\n4"] "cle.4" [label="clean_oven\n4"] "pre.4" [label="pre_heat_oven\n4"] } "pur.1" -> "mak.1" -> "bak.1" -> "sel.1" "pre.1" -> "bak.1" -> "cle.1" "cle.1" -> "pre.2" "sel.1" -> "pur.3" "pur.2" -> "mak.2" -> "bak.2" -> "sel.2" "pre.2" -> "bak.2" -> "cle.2" "cle.2" -> "pre.3" "sel.2" -> "pur.4" "pur.3" -> "mak.3" -> "bak.3" -> "sel.3" "pre.3" -> "bak.3" -> "cle.3" "cle.3" -> "pre.4" "pur.4" -> "mak.4" -> "bak.4" -> "sel.4" "pre.4" -> "bak.4" -> "cle.4" }

Recurrence Sections§

[scheduling]
    cycling mode = integer
    initial cycle point = 1
    [[dependencies]]
        [[[P1]]]  # Repeat every cycle.
            graph = foo
        [[[P2]]]  # Repeat every second cycle.
            graph = bar
        [[[P3]]]  # Repeat every third cycle.
            graph = baz

digraph example { subgraph cluster_1 { label = 1 style = dashed "foo.1" [label="foo\n1"] "bar.1" [label="bar\n1"] "baz.1" [label="baz\n1"] } subgraph cluster_2 { label = 2 style = dashed "foo.2" [label="foo\n2"] } subgraph cluster_3 { label = 3 style = dashed "foo.3" [label="foo\n3"] "bar.3" [label="bar\n3"] } }

Recurrence Sections§

5/P3
Repeat every third cycle starting from cycle number 5.
+P5/P3
Repeat every third cycle starting 5 cycles after the initial cycle point.

Recurrence Sections§

In this practical we will take the suite we wrote in the previous section and turn it into a cycling suite.

Next section: Date-Time Cycling

Practical

In this practical we will take the suite we wrote in the previous section and turn it into a cycling suite.

If you have not completed the previous practical use the following code for your suite.rc file.

[scheduling]
    [[dependencies]]
        graph = """
            foo & pub => bar => baz & wop
            baz => qux
        """
  1. Create a new suite.

    Within your ~/cylc-run/ directory create a new (sub-)directory called integer-cycling and move into it:

    mkdir ~/cylc-run/integer-cycling
    cd ~/cylc-run/integer-cycling
    

    Copy the above code into a suite.rc file in that directory.

  2. Make the suite cycle.

    Add in the following lines.

     [scheduling]
    +    cycling mode = integer
    +    initial cycle point = 1
         [[dependencies]]
    +        [[[P1]]]
                 graph = """
                     foo & pub => bar => baz & wop
                     baz => qux
                 """
    
  3. Visualise the suite.

    Try visualising the suite using cylc graph.

    cylc graph .
    

    Tip

    You can get Cylc graph to draw dotted boxes around the cycles by clicking the “Organise by cycle point” button on the toolbar:

    ../../../_images/cylc-graph-cluster.png

    Tip

    By default cylc graph displays the first three cycles of a suite. You can tell cylc graph to visualise the cycles between two points by providing them as arguments, for instance the following example would show all cycles between 1 and 5 (inclusive):

    cylc graph . 1 5 &
    
  4. Add another recurrence.

    Suppose we wanted the qux task to run every other cycle as opposed to every cycle. We can do this by adding another recurrence.

    Make the following changes to your suite.rc file.

     [scheduling]
         cycling mode = integer
         initial cycle point = 1
         [[dependencies]]
             [[[P1]]]
                 graph = """
                     foo & pub => bar => baz & wop
    -                baz => qux
                 """
    +        [[[P2]]]
    +            graph = """
    +                baz => qux
    +            """
    

    Use cylc graph to see the effect this has on the workflow.

  5. Inter-cycle dependencies.

    Next we need to add some inter-cycle dependencies. We are going to add three inter-cycle dependencies:

    1. Between wop from the previous cycle and pub.
    2. Between baz from the previous cycle and foo every odd cycle.
    3. Between qux from the previous cycle and foo every even cycle.

    Have a go at adding inter-cycle dependencies to your suite.rc file to make your workflow match the diagram below.

    Hint

    • P2 means every odd cycle.
    • 2/P2 means every even cycle.

    digraph example { size = "4.5,7" subgraph cluster_1 { label = 1 style = dashed "foo.1" [label="foo\n1"] "bar.1" [label="bar\n1"] "baz.1" [label="baz\n1"] "wop.1" [label="wop\n1"] "pub.1" [label="pub\n1"] "qux.1" [label="qux\n1"] } subgraph cluster_2 { label = 2 style = dashed "foo.2" [label="foo\n2"] "bar.2" [label="bar\n2"] "baz.2" [label="baz\n2"] "wop.2" [label="wop\n2"] "pub.2" [label="pub\n2"] } subgraph cluster_3 { label = 3 style = dashed "foo.3" [label="foo\n3"] "bar.3" [label="bar\n3"] "baz.3" [label="baz\n3"] "wop.3" [label="wop\n3"] "pub.3" [label="pub\n3"] "qux.3" [label="qux\n3"] } "foo.1" -> "bar.1" -> "wop.1" "bar.1" -> "baz.1" "pub.1" -> "bar.1" "foo.2" -> "bar.2" -> "wop.2" "bar.2" -> "baz.2" "pub.2" -> "bar.2" "foo.3" -> "bar.3" -> "wop.3" "bar.3" -> "baz.3" "pub.3" -> "bar.3" "baz.1" -> "qux.1" -> "foo.2" "baz.3" -> "qux.3" "baz.2" -> "foo.3" "wop.1" -> "pub.2" "wop.2" -> "pub.3" }

    Solution

    [scheduling]
        cycling mode = integer
        initial cycle point = 1
        [[dependencies]]
            [[[P1]]]
                graph = """
                    foo & pub => bar => baz & wop
                    wop[-P1] => pub  # (1)
                """
            [[[P2]]]
                graph = """
                    baz => qux
                    baz[-P1] => foo  # (2)
                """
            [[[2/P2]]]
                graph = """
                    qux[-P1] => foo  # (3)
                """