Introduction

Providing output to the screen is an important part of any program. In this lesson we will review the most common techniques for printing to the console and knitted HTML documents. This lesson does not cover how to output results to a user interface or the web. For this you need to use packages such as tcltk or shiny.

Outputting values to the console is also important for ad hoc debugging.

There are several functions that are commonly used to output results to the console:

  • print(), often combined with paste()
  • head() and tail() for dataframes and vectors
  • sprintf()
  • cat()
  • message()
  • knitr::raw_html()

To add HTML to the knitted document, we often use knitr::raw_html().

Let’s take a look at each of the functions and their most common use cases. We will start with the most commonly used function for printing to the console: print().

print()

print() is the most commonly used function for printing messages and the value of R objects and variables to the console. In its simplest form, just using a variable as a stand-along statement prints its contents. Alternatively, calling print() with a single argument will print the variable. Calling print() is preferable as it makes the intent of printing the variables clear. As an aside, be careful printing large dataframes or vectors. For that use the head() function instead.

x <- 23.77

# display the contents of the variable
x
## [1] 23.77

# display contents using print()
print(x)
## [1] 23.77

For more complex output or to output several variables, a single argument has to be formed using the paste() function. The paste() function is the R way to concatenate strings.

x <- 23.77

# display the contents of the variable
x
## [1] 23.77
# display contents with surrounding text
print(paste("x =", x))
## [1] "x = 23.77"

paste() and paste0()

The paste() function is used to concatenate (combine) strings. By default, the arguments are separated by a space. To separate the strings with any other string, use the argument sep. To get rid of any separator, either use sep=““ or the paste0() function which sets sep=““ by default but otherwise works like paste(). The code below illustrates the various parameters.

x <- 100.99
s <- paste("The result is", x)
print(s)
FALSE [1] "The result is 100.99"
print(paste("Amount: ", "$", x, sep=""))
FALSE [1] "Amount: $100.99"
print(paste0("Amount: ", "R$", round((x/5.14),2)))
FALSE [1] "Amount: R$19.65"

head() and tail()

The head() function is the preferred way to display a subset of a dataframe or a vector. The function takes an optional argument that determines how many rows will be printed; the default is 6. The tail() function is the same as head() except that it prints at rows at the end rather than beginning (from the tail rather than the head of the structure).

df <- read.csv("customertxndata.csv")

head(df, 3)
##   NumVisits NumOrders      OS Gender    Total
## 1         7         0 Android   Male   0.0000
## 2        20         1     iOS   <NA> 576.8668
## 3        22         1     iOS Female 850.0000
tail(df,2)
##       NumVisits NumOrders      OS Gender Total
## 22799         1         0 Android   Male     0
## 22800        13         1 Android   Male   340

Comment Option for Code Blocks

Note that the output within an R Notebook code block (or code fence) display the ## prefix. Ti illustrate look at the output of the code block below.

print(x)
## [1] 100.99

To suppress the printing of the prefix, use the code block option:

```{r comment=""}
```

The result is shown below. Of course, you can specify any other string as a prefix.

print(x)
[1] 100.99

cat()

Notice that it still prints the result as a vector with the element number, e.g., above it printed [1] before. This can be eliminated by using cat() rather than print().

cat(x)
100.99

Unlike print(), cat() takes a variable number of argument, so the need for paste() is not as common.

cat('x =', x)
x = 100.99

Rather than printing the contents to the console (or as part of a knitted document), cat() can also direct the output to a file using the argument file. This is particularly useful for log messages in R Scripts. Use the argument append=T to append subsequent outputs to the same file. Also, by default, subsequent output is to the same line, unless the argument fill=T is specified or the separator is set to a newline character with sep=““. The code below illustrates these features.

cat('x =', x, file = "log.txt", append = T, sep="\n")

Notice that by default the arguments are separated by a space, although the argument sep can modify that to a newline or another separator character such as “,” for CSV files.

sprintf()

The sprintf() function provides output formatting similar to the function with the same name in C. The result is “output” to a string which can then be echoed to the console using either print() or cat().

The format string options are the same as in C.

x <- 87.34
c <- "R$"
s <- sprintf("The amount is %s%f", c, x)

print(s)
## [1] "The amount is R$87.340000"

The first argument is the format string which contains text plus formatting options. The remaining arguments are added to the first argument in place of the formatting options. So, in the example above, the value of c is added where %s appear while the value of x is added to where %f appears. The value is formatted according to the format, e.g., %s is for strings and %f is for floating point numbers. Other formats include %e for scientific notation, %d for integers, among others. The code below illustrates this with examples. Note the format specification for floating point numbers.

x <- 97.958         # floating point number
i <- 1024            # integer
s <- 'brown cow'     # string/text

print(
  sprintf("%d / %.2f / %s", i, x, s)
)
## [1] "1024 / 97.96 / brown cow"

message()

The output of the message() function within an R code block (code fence) can be suppressed with the argument message = F.

In the block below, the argument the message argument is TRUE, so messages are displayed.

```{r message=T}
...
```
x <- 12.77
message("debug info / x = ", x)
## debug info / x = 12.77

On the other hand, in the block below, the argument is now set to FALSE, so output from the message() is suppressed. This can be useful for debug message that can be turned off for a block or the entire document with the knitr::opt_chunks() function.

x <- 12.77
message("debug info / x = ", x)

Notice that there is no message being printed.

knitr::raw_html()

This function only applies when knitting a document to HTML, although there are similar functions for LaTeX and PDF output formats. Recall that knitting a document produces an HTML file. Using knit::raw_html() one can add any arbitrary HTML sequence to the result document at the location in the document where the function is called.

knitr::raw_html("<pre>Random <b>HTML</b>.</pre>")
Random HTML.

This can be particularly useful for special formatting, computed HTML, or special characters.

Summary

R provides several ways to display results on the console. The most common functions are print with paste0 and cat, while debug and other information messages are displayed using messag_e. For injecting HTML we generally use knitr::raw_html. The sprintf function can be useful for producing formatted strings.


Files & Resources

All Files for Lesson 6.190

Errata

None collected yet. Let us know.

LS0tDQp0aXRsZTogIkNvbnNvbGUgT3V0cHV0IGluIFIiDQpwYXJhbXM6DQogIGNhdGVnb3J5OiA2DQogIG51bWJlcjogMTkwDQogIHRpbWU6IDIwDQogIGxldmVsOiBiZWdpbm5lcg0KICB0YWdzOiAicixjb25zb2xlLG1lc3NhZ2Usc3ByaW50ZixwcmludCxwYXN0ZSxjYXQiDQogIGRlc2NyaXB0aW9uOiAiSWxsdXN0cmF0ZXMgZGlmZmVyZW50IGZ1bmN0aW9ucyBpbiBSIGZvciBwZXJmb3JtaW5nDQogICAgICAgICAgICAgICAgY29uc29sZSBvdXRwdXQsIGluY2x1ZGluZyBwcmludCgpLCBjYXQoKSwgcHJpbnRmKCksDQogICAgICAgICAgICAgICAgc3ByaW5nZigpLCBhbmQgbWVzc2FnZSgpLiBQcm92aWRlcyB1c2UgY2FzZXMgZm9yIGVhY2ggYW5kDQogICAgICAgICAgICAgICAgd2hlbiB0byBiZXN0IHVzZSB0aGVtLiINCmRhdGU6ICI8c21hbGw+YHIgU3lzLkRhdGUoKWA8L3NtYWxsPiINCmF1dGhvcjogIjxzbWFsbD5NYXJ0aW4gU2NoZWRsYmF1ZXI8L3NtYWxsPiINCmVtYWlsOiAibS5zY2hlZGxiYXVlckBuZXUuZWR1Ig0KYWZmaWxpdGF0aW9uOiAiTm9ydGhlYXN0ZXJuIFVuaXZlcnNpdHkiDQpvdXRwdXQ6IA0KICBib29rZG93bjo6aHRtbF9kb2N1bWVudDI6DQogICAgdG9jOiB0cnVlDQogICAgdG9jX2Zsb2F0OiB0cnVlDQogICAgY29sbGFwc2VkOiBmYWxzZQ0KICAgIG51bWJlcl9zZWN0aW9uczogZmFsc2UNCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlDQogICAgdGhlbWU6IHNwYWNlbGFiDQogICAgaGlnaGxpZ2h0OiB0YW5nbw0KLS0tDQoNCi0tLQ0KdGl0bGU6ICI8c21hbGw+YHIgcGFyYW1zJGNhdGVnb3J5YC5gciBwYXJhbXMkbnVtYmVyYDwvc21hbGw+PGJyLz48c3BhbiBzdHlsZT0nY29sb3I6ICMyRTQwNTM7IGZvbnQtc2l6ZTogMC45ZW0nPmByIHJtYXJrZG93bjo6bWV0YWRhdGEkdGl0bGVgPC9zcGFuPiINCi0tLQ0KDQpgYGB7ciBjb2RlPXhmdW46OnJlYWRfdXRmOChwYXN0ZTAoaGVyZTo6aGVyZSgpLCcvUi9faW5zZXJ0MkRCLlInKSksIGluY2x1ZGUgPSBGQUxTRX0NCmBgYA0KDQojIyBJbnRyb2R1Y3Rpb24NCg0KUHJvdmlkaW5nIG91dHB1dCB0byB0aGUgc2NyZWVuIGlzIGFuIGltcG9ydGFudCBwYXJ0IG9mIGFueSBwcm9ncmFtLiBJbiB0aGlzIGxlc3NvbiB3ZSB3aWxsIHJldmlldyB0aGUgbW9zdCBjb21tb24gdGVjaG5pcXVlcyBmb3IgcHJpbnRpbmcgdG8gdGhlIGNvbnNvbGUgYW5kIGtuaXR0ZWQgSFRNTCBkb2N1bWVudHMuIFRoaXMgbGVzc29uIGRvZXMgbm90IGNvdmVyIGhvdyB0byBvdXRwdXQgcmVzdWx0cyB0byBhIHVzZXIgaW50ZXJmYWNlIG9yIHRoZSB3ZWIuIEZvciB0aGlzIHlvdSBuZWVkIHRvIHVzZSBwYWNrYWdlcyBzdWNoIGFzICoqdGNsdGsqKiBvciAqKnNoaW55KiouDQoNCk91dHB1dHRpbmcgdmFsdWVzIHRvIHRoZSBjb25zb2xlIGlzIGFsc28gaW1wb3J0YW50IGZvciAqYWQgaG9jKiBkZWJ1Z2dpbmcuDQoNClRoZXJlIGFyZSBzZXZlcmFsIGZ1bmN0aW9ucyB0aGF0IGFyZSBjb21tb25seSB1c2VkIHRvIG91dHB1dCByZXN1bHRzIHRvIHRoZSBjb25zb2xlOg0KDQotICAgPGNvZGU+cHJpbnQoKTwvY29kZT4sIG9mdGVuIGNvbWJpbmVkIHdpdGggPGNvZGU+cGFzdGUoKTwvY29kZT4NCi0gICA8Y29kZT5oZWFkKCk8L2NvZGU+IGFuZCA8Y29kZT50YWlsKCk8L2NvZGU+IGZvciBkYXRhZnJhbWVzIGFuZCB2ZWN0b3JzDQotICAgPGNvZGU+c3ByaW50ZigpPC9jb2RlPg0KLSAgIDxjb2RlPmNhdCgpPC9jb2RlPg0KLSAgIDxjb2RlPm1lc3NhZ2UoKTwvY29kZT4NCi0gICA8Y29kZT5rbml0cjo6cmF3X2h0bWwoKTwvY29kZT4NCg0KVG8gYWRkIEhUTUwgdG8gdGhlIGtuaXR0ZWQgZG9jdW1lbnQsIHdlIG9mdGVuIHVzZSA8Y29kZT5rbml0cjo6cmF3X2h0bWwoKTwvY29kZT4uDQoNCkxldCdzIHRha2UgYSBsb29rIGF0IGVhY2ggb2YgdGhlIGZ1bmN0aW9ucyBhbmQgdGhlaXIgbW9zdCBjb21tb24gdXNlIGNhc2VzLiBXZSB3aWxsIHN0YXJ0IHdpdGggdGhlIG1vc3QgY29tbW9ubHkgdXNlZCBmdW5jdGlvbiBmb3IgcHJpbnRpbmcgdG8gdGhlIGNvbnNvbGU6IDxjb2RlPnByaW50KCk8L2NvZGU+Lg0KDQojIyBwcmludCgpDQoNCjxjb2RlPnByaW50KCk8L2NvZGU+IGlzIHRoZSBtb3N0IGNvbW1vbmx5IHVzZWQgZnVuY3Rpb24gZm9yIHByaW50aW5nIG1lc3NhZ2VzIGFuZCB0aGUgdmFsdWUgb2YgUiBvYmplY3RzIGFuZCB2YXJpYWJsZXMgdG8gdGhlIGNvbnNvbGUuIEluIGl0cyBzaW1wbGVzdCBmb3JtLCBqdXN0IHVzaW5nIGEgdmFyaWFibGUgYXMgYSBzdGFuZC1hbG9uZyBzdGF0ZW1lbnQgcHJpbnRzIGl0cyBjb250ZW50cy4gQWx0ZXJuYXRpdmVseSwgY2FsbGluZyA8Y29kZT5wcmludCgpPC9jb2RlPiB3aXRoIGEgc2luZ2xlIGFyZ3VtZW50IHdpbGwgcHJpbnQgdGhlIHZhcmlhYmxlLiBDYWxsaW5nIDxjb2RlPnByaW50KCk8L2NvZGU+IGlzIHByZWZlcmFibGUgYXMgaXQgbWFrZXMgdGhlIGludGVudCBvZiBwcmludGluZyB0aGUgdmFyaWFibGVzIGNsZWFyLiBBcyBhbiBhc2lkZSwgYmUgY2FyZWZ1bCBwcmludGluZyBsYXJnZSBkYXRhZnJhbWVzIG9yIHZlY3RvcnMuIEZvciB0aGF0IHVzZSB0aGUgPGNvZGU+aGVhZCgpPC9jb2RlPiBmdW5jdGlvbiBpbnN0ZWFkLg0KDQpgYGB7ciBjb2xsYXBzZT1UUlVFfQ0KeCA8LSAyMy43Nw0KDQojIGRpc3BsYXkgdGhlIGNvbnRlbnRzIG9mIHRoZSB2YXJpYWJsZQ0KeA0KDQojIGRpc3BsYXkgY29udGVudHMgdXNpbmcgcHJpbnQoKQ0KcHJpbnQoeCkNCmBgYA0KDQpGb3IgbW9yZSBjb21wbGV4IG91dHB1dCBvciB0byBvdXRwdXQgc2V2ZXJhbCB2YXJpYWJsZXMsIGEgc2luZ2xlIGFyZ3VtZW50IGhhcyB0byBiZSBmb3JtZWQgdXNpbmcgdGhlIDxjb2RlPnBhc3RlKCk8L2NvZGU+IGZ1bmN0aW9uLiBUaGUgPGNvZGU+cGFzdGUoKTwvY29kZT4gZnVuY3Rpb24gaXMgdGhlIFIgd2F5IHRvIGNvbmNhdGVuYXRlIHN0cmluZ3MuDQoNCmBgYHtyIGNvbGxhcHNlPUZBTFNFfQ0KeCA8LSAyMy43Nw0KDQojIGRpc3BsYXkgdGhlIGNvbnRlbnRzIG9mIHRoZSB2YXJpYWJsZQ0KeA0KDQojIGRpc3BsYXkgY29udGVudHMgd2l0aCBzdXJyb3VuZGluZyB0ZXh0DQpwcmludChwYXN0ZSgieCA9IiwgeCkpDQpgYGANCg0KIyMgcGFzdGUoKSBhbmQgcGFzdGUwKCkNCg0KVGhlIDxjb2RlPnBhc3RlKCk8L2NvZGU+IGZ1bmN0aW9uIGlzIHVzZWQgdG8gY29uY2F0ZW5hdGUgKGNvbWJpbmUpIHN0cmluZ3MuIEJ5IGRlZmF1bHQsIHRoZSBhcmd1bWVudHMgYXJlIHNlcGFyYXRlZCBieSBhIHNwYWNlLiBUbyBzZXBhcmF0ZSB0aGUgc3RyaW5ncyB3aXRoIGFueSBvdGhlciBzdHJpbmcsIHVzZSB0aGUgYXJndW1lbnQgKnNlcCouIFRvIGdldCByaWQgb2YgYW55IHNlcGFyYXRvciwgZWl0aGVyIHVzZSAqc2VwPSIiKiBvciB0aGUgPGNvZGU+cGFzdGUwKCk8L2NvZGU+IGZ1bmN0aW9uIHdoaWNoIHNldHMgKnNlcD0iIiogYnkgZGVmYXVsdCBidXQgb3RoZXJ3aXNlIHdvcmtzIGxpa2UgPGNvZGU+cGFzdGUoKTwvY29kZT4uIFRoZSBjb2RlIGJlbG93IGlsbHVzdHJhdGVzIHRoZSB2YXJpb3VzIHBhcmFtZXRlcnMuDQoNCmBgYHtyIGNvbW1lbnQ9Rn0NCnggPC0gMTAwLjk5DQpzIDwtIHBhc3RlKCJUaGUgcmVzdWx0IGlzIiwgeCkNCnByaW50KHMpDQoNCnByaW50KHBhc3RlKCJBbW91bnQ6ICIsICIkIiwgeCwgc2VwPSIiKSkNCg0KcHJpbnQocGFzdGUwKCJBbW91bnQ6ICIsICJSJCIsIHJvdW5kKCh4LzUuMTQpLDIpKSkNCmBgYA0KDQojIyBoZWFkKCkgYW5kIHRhaWwoKQ0KDQpUaGUgPGNvZGU+aGVhZCgpPC9jb2RlPiBmdW5jdGlvbiBpcyB0aGUgcHJlZmVycmVkIHdheSB0byBkaXNwbGF5IGEgc3Vic2V0IG9mIGEgZGF0YWZyYW1lIG9yIGEgdmVjdG9yLiBUaGUgZnVuY3Rpb24gdGFrZXMgYW4gb3B0aW9uYWwgYXJndW1lbnQgdGhhdCBkZXRlcm1pbmVzIGhvdyBtYW55IHJvd3Mgd2lsbCBiZSBwcmludGVkOyB0aGUgZGVmYXVsdCBpcyA2LiBUaGUgPGNvZGU+dGFpbCgpPC9jb2RlPiBmdW5jdGlvbiBpcyB0aGUgc2FtZSBhcyA8Y29kZT5oZWFkKCk8L2NvZGU+IGV4Y2VwdCB0aGF0IGl0IHByaW50cyBhdCByb3dzIGF0IHRoZSBlbmQgcmF0aGVyIHRoYW4gYmVnaW5uaW5nIChmcm9tIHRoZSAqdGFpbCogcmF0aGVyIHRoYW4gdGhlICpoZWFkKiBvZiB0aGUgc3RydWN0dXJlKS4NCg0KYGBge3J9DQpkZiA8LSByZWFkLmNzdigiY3VzdG9tZXJ0eG5kYXRhLmNzdiIpDQoNCmhlYWQoZGYsIDMpDQoNCnRhaWwoZGYsMikNCmBgYA0KDQojIyBDb21tZW50IE9wdGlvbiBmb3IgQ29kZSBCbG9ja3MNCg0KTm90ZSB0aGF0IHRoZSBvdXRwdXQgd2l0aGluIGFuIFIgTm90ZWJvb2sgY29kZSBibG9jayAob3IgY29kZSBmZW5jZSkgZGlzcGxheSB0aGUgXCMjIHByZWZpeC4gVGkgaWxsdXN0cmF0ZSBsb29rIGF0IHRoZSBvdXRwdXQgb2YgdGhlIGNvZGUgYmxvY2sgYmVsb3cuDQoNCmBgYHtyfQ0KcHJpbnQoeCkNCmBgYA0KDQpUbyBzdXBwcmVzcyB0aGUgcHJpbnRpbmcgb2YgdGhlIHByZWZpeCwgdXNlIHRoZSBjb2RlIGJsb2NrIG9wdGlvbjoNCg0KYGBge3IgZWNobz1GfQ0Ka25pdHI6OnJhd19odG1sKCc8cHJlPicpDQprbml0cjo6cmF3X2h0bWwocGFzdGUoDQogICcmIzk2OyYjOTY7JiM5Njt7ciBjb21tZW50PSIifScsDQogICcmIzk2OyYjOTY7JiM5NjsnLA0KICBzZXA9Ijxici8+IikpDQprbml0cjo6cmF3X2h0bWwoJzwvcHJlPicpDQpgYGANCg0KVGhlIHJlc3VsdCBpcyBzaG93biBiZWxvdy4gT2YgY291cnNlLCB5b3UgY2FuIHNwZWNpZnkgYW55IG90aGVyIHN0cmluZyBhcyBhIHByZWZpeC4NCg0KYGBge3IgY29tbWVudD0iIn0NCnByaW50KHgpDQpgYGANCg0KIyMgY2F0KCkNCg0KTm90aWNlIHRoYXQgaXQgc3RpbGwgcHJpbnRzIHRoZSByZXN1bHQgYXMgYSB2ZWN0b3Igd2l0aCB0aGUgZWxlbWVudCBudW1iZXIsICplLmcuKiwgYWJvdmUgaXQgcHJpbnRlZCBbMV0gYmVmb3JlLiBUaGlzIGNhbiBiZSBlbGltaW5hdGVkIGJ5IHVzaW5nIDxjb2RlPmNhdCgpPC9jb2RlPiByYXRoZXIgdGhhbiA8Y29kZT5wcmludCgpPC9jb2RlPi4NCg0KYGBge3IgY29tbWVudD0iIn0NCmNhdCh4KQ0KYGBgDQoNClVubGlrZSA8Y29kZT5wcmludCgpPC9jb2RlPiwgPGNvZGU+Y2F0KCk8L2NvZGU+IHRha2VzIGEgdmFyaWFibGUgbnVtYmVyIG9mIGFyZ3VtZW50LCBzbyB0aGUgbmVlZCBmb3IgPGNvZGU+cGFzdGUoKTwvY29kZT4gaXMgbm90IGFzIGNvbW1vbi4NCg0KYGBge3IgY29tbWVudD0iIn0NCmNhdCgneCA9JywgeCkNCmBgYA0KDQpSYXRoZXIgdGhhbiBwcmludGluZyB0aGUgY29udGVudHMgdG8gdGhlIGNvbnNvbGUgKG9yIGFzIHBhcnQgb2YgYSBrbml0dGVkIGRvY3VtZW50KSwgPGNvZGU+Y2F0KCk8L2NvZGU+IGNhbiBhbHNvIGRpcmVjdCB0aGUgb3V0cHV0IHRvIGEgZmlsZSB1c2luZyB0aGUgYXJndW1lbnQgKmZpbGUqLiBUaGlzIGlzIHBhcnRpY3VsYXJseSB1c2VmdWwgZm9yIGxvZyBtZXNzYWdlcyBpbiBSIFNjcmlwdHMuIFVzZSB0aGUgYXJndW1lbnQgKmFwcGVuZD1UKiB0byBhcHBlbmQgc3Vic2VxdWVudCBvdXRwdXRzIHRvIHRoZSBzYW1lIGZpbGUuIEFsc28sIGJ5IGRlZmF1bHQsIHN1YnNlcXVlbnQgb3V0cHV0IGlzIHRvIHRoZSBzYW1lIGxpbmUsIHVubGVzcyB0aGUgYXJndW1lbnQgKmZpbGw9VCogaXMgc3BlY2lmaWVkIG9yIHRoZSBzZXBhcmF0b3IgaXMgc2V0IHRvIGEgbmV3bGluZSBjaGFyYWN0ZXIgd2l0aCAqc2VwPSIqXG4iLiBUaGUgY29kZSBiZWxvdyBpbGx1c3RyYXRlcyB0aGVzZSBmZWF0dXJlcy4NCg0KYGBge3IgY29tbWVudD0iIn0NCmNhdCgneCA9JywgeCwgZmlsZSA9ICJsb2cudHh0IiwgYXBwZW5kID0gVCwgc2VwPSJcbiIpDQpgYGANCg0KTm90aWNlIHRoYXQgYnkgZGVmYXVsdCB0aGUgYXJndW1lbnRzIGFyZSBzZXBhcmF0ZWQgYnkgYSBzcGFjZSwgYWx0aG91Z2ggdGhlIGFyZ3VtZW50ICpzZXAqIGNhbiBtb2RpZnkgdGhhdCB0byBhIG5ld2xpbmUgb3IgYW5vdGhlciBzZXBhcmF0b3IgY2hhcmFjdGVyIHN1Y2ggYXMgIiwiIGZvciBDU1YgZmlsZXMuDQoNCiMjIHNwcmludGYoKQ0KDQpUaGUgPGNvZGU+c3ByaW50ZigpPC9jb2RlPiBmdW5jdGlvbiBwcm92aWRlcyBvdXRwdXQgZm9ybWF0dGluZyBzaW1pbGFyIHRvIHRoZSBmdW5jdGlvbiB3aXRoIHRoZSBzYW1lIG5hbWUgaW4gQy4gVGhlIHJlc3VsdCBpcyAib3V0cHV0IiB0byBhIHN0cmluZyB3aGljaCBjYW4gdGhlbiBiZSBlY2hvZWQgdG8gdGhlIGNvbnNvbGUgdXNpbmcgZWl0aGVyIDxjb2RlPnByaW50KCk8L2NvZGU+IG9yIDxjb2RlPmNhdCgpPC9jb2RlPi4NCg0KVGhlIGZvcm1hdCBzdHJpbmcgb3B0aW9ucyBhcmUgdGhlIHNhbWUgYXMgaW4gQy4NCg0KYGBge3J9DQp4IDwtIDg3LjM0DQpjIDwtICJSJCINCnMgPC0gc3ByaW50ZigiVGhlIGFtb3VudCBpcyAlcyVmIiwgYywgeCkNCg0KcHJpbnQocykNCmBgYA0KDQpUaGUgZmlyc3QgYXJndW1lbnQgaXMgdGhlIGZvcm1hdCBzdHJpbmcgd2hpY2ggY29udGFpbnMgdGV4dCBwbHVzIGZvcm1hdHRpbmcgb3B0aW9ucy4gVGhlIHJlbWFpbmluZyBhcmd1bWVudHMgYXJlIGFkZGVkIHRvIHRoZSBmaXJzdCBhcmd1bWVudCBpbiBwbGFjZSBvZiB0aGUgZm9ybWF0dGluZyBvcHRpb25zLiBTbywgaW4gdGhlIGV4YW1wbGUgYWJvdmUsIHRoZSB2YWx1ZSBvZiAqYyogaXMgYWRkZWQgd2hlcmUgKiVzKiBhcHBlYXIgd2hpbGUgdGhlIHZhbHVlIG9mICp4KiBpcyBhZGRlZCB0byB3aGVyZSAqJWYqIGFwcGVhcnMuIFRoZSB2YWx1ZSBpcyBmb3JtYXR0ZWQgYWNjb3JkaW5nIHRvIHRoZSBmb3JtYXQsICplLmcuKiwgKiVzKiBpcyBmb3Igc3RyaW5ncyBhbmQgKiVmKiBpcyBmb3IgZmxvYXRpbmcgcG9pbnQgbnVtYmVycy4gT3RoZXIgZm9ybWF0cyBpbmNsdWRlIColZSogZm9yIHNjaWVudGlmaWMgbm90YXRpb24sIColZCogZm9yIGludGVnZXJzLCBhbW9uZyBvdGhlcnMuIFRoZSBjb2RlIGJlbG93IGlsbHVzdHJhdGVzIHRoaXMgd2l0aCBleGFtcGxlcy4gTm90ZSB0aGUgZm9ybWF0IHNwZWNpZmljYXRpb24gZm9yIGZsb2F0aW5nIHBvaW50IG51bWJlcnMuDQoNCmBgYHtyfQ0KeCA8LSA5Ny45NTggICAgICAgICAjIGZsb2F0aW5nIHBvaW50IG51bWJlcg0KaSA8LSAxMDI0ICAgICAgICAgICAgIyBpbnRlZ2VyDQpzIDwtICdicm93biBjb3cnICAgICAjIHN0cmluZy90ZXh0DQoNCnByaW50KA0KICBzcHJpbnRmKCIlZCAvICUuMmYgLyAlcyIsIGksIHgsIHMpDQopDQpgYGANCg0KIyMgbWVzc2FnZSgpDQoNClRoZSBvdXRwdXQgb2YgdGhlIDxjb2RlPm1lc3NhZ2UoKTwvY29kZT4gZnVuY3Rpb24gd2l0aGluIGFuIFIgY29kZSBibG9jayAoY29kZSBmZW5jZSkgY2FuIGJlIHN1cHByZXNzZWQgd2l0aCB0aGUgYXJndW1lbnQgKm1lc3NhZ2UgPSBGKi4NCg0KSW4gdGhlIGJsb2NrIGJlbG93LCB0aGUgYXJndW1lbnQgdGhlIG1lc3NhZ2UgYXJndW1lbnQgaXMgVFJVRSwgc28gbWVzc2FnZXMgYXJlIGRpc3BsYXllZC4NCg0KYGBge3IgZWNobz1GfQ0Ka25pdHI6OnJhd19odG1sKCI8cHJlPiIpDQprbml0cjo6cmF3X2h0bWwocGFzdGUoDQogICImIzk2OyYjOTY7JiM5Njt7ciBtZXNzYWdlPVR9IiwNCiAgIi4uLiIsDQogICImIzk2OyYjOTY7JiM5NjsiLA0KICBzZXA9Ijxici8+IikpDQprbml0cjo6cmF3X2h0bWwoIjwvcHJlPiIpDQpgYGANCg0KYGBge3IgbWVzc2FnZT1UUlVFfQ0KeCA8LSAxMi43Nw0KbWVzc2FnZSgiZGVidWcgaW5mbyAvIHggPSAiLCB4KQ0KYGBgDQoNCk9uIHRoZSBvdGhlciBoYW5kLCBpbiB0aGUgYmxvY2sgYmVsb3csIHRoZSBhcmd1bWVudCBpcyBub3cgc2V0IHRvICpGQUxTRSosIHNvIG91dHB1dCBmcm9tIHRoZSA8Y29kZT5tZXNzYWdlKCk8L2NvZGU+IGlzIHN1cHByZXNzZWQuIFRoaXMgY2FuIGJlIHVzZWZ1bCBmb3IgZGVidWcgbWVzc2FnZSB0aGF0IGNhbiBiZSB0dXJuZWQgb2ZmIGZvciBhIGJsb2NrIG9yIHRoZSBlbnRpcmUgZG9jdW1lbnQgd2l0aCB0aGUgPGNvZGU+a25pdHI6Om9wdF9jaHVua3MoKTwvY29kZT4gZnVuY3Rpb24uDQoNCmBgYHtyIG1lc3NhZ2U9RkFMU0V9DQp4IDwtIDEyLjc3DQptZXNzYWdlKCJkZWJ1ZyBpbmZvIC8geCA9ICIsIHgpDQpgYGANCg0KTm90aWNlIHRoYXQgdGhlcmUgaXMgbm8gbWVzc2FnZSBiZWluZyBwcmludGVkLg0KDQojIyBrbml0cjo6cmF3X2h0bWwoKQ0KDQpUaGlzIGZ1bmN0aW9uIG9ubHkgYXBwbGllcyB3aGVuIGtuaXR0aW5nIGEgZG9jdW1lbnQgdG8gSFRNTCwgYWx0aG91Z2ggdGhlcmUgYXJlIHNpbWlsYXIgZnVuY3Rpb25zIGZvciBMYVRlWCBhbmQgUERGIG91dHB1dCBmb3JtYXRzLiBSZWNhbGwgdGhhdCBrbml0dGluZyBhIGRvY3VtZW50IHByb2R1Y2VzIGFuIEhUTUwgZmlsZS4gVXNpbmcgPGNvZGU+a25pdDo6cmF3X2h0bWwoKTwvY29kZT4gb25lIGNhbiBhZGQgYW55IGFyYml0cmFyeSBIVE1MIHNlcXVlbmNlIHRvIHRoZSByZXN1bHQgZG9jdW1lbnQgYXQgdGhlIGxvY2F0aW9uIGluIHRoZSBkb2N1bWVudCB3aGVyZSB0aGUgZnVuY3Rpb24gaXMgY2FsbGVkLg0KDQpgYGB7cn0NCmtuaXRyOjpyYXdfaHRtbCgiPHByZT5SYW5kb20gPGI+SFRNTDwvYj4uPC9wcmU+IikNCmBgYA0KDQpUaGlzIGNhbiBiZSBwYXJ0aWN1bGFybHkgdXNlZnVsIGZvciBzcGVjaWFsIGZvcm1hdHRpbmcsIGNvbXB1dGVkIEhUTUwsIG9yIHNwZWNpYWwgY2hhcmFjdGVycy4NCg0KIyMgU3VtbWFyeQ0KDQpSIHByb3ZpZGVzIHNldmVyYWwgd2F5cyB0byBkaXNwbGF5IHJlc3VsdHMgb24gdGhlIGNvbnNvbGUuIFRoZSBtb3N0IGNvbW1vbiBmdW5jdGlvbnMgYXJlICpwcmludCogd2l0aCAqcGFzdGUwKiBhbmQgKmNhdCosIHdoaWxlIGRlYnVnIGFuZCBvdGhlciBpbmZvcm1hdGlvbiBtZXNzYWdlcyBhcmUgZGlzcGxheWVkIHVzaW5nICptZXNzYWdfZSouIEZvciBpbmplY3RpbmcgSFRNTCB3ZSBnZW5lcmFsbHkgdXNlICprbml0cjo6cmF3X2h0bWwqLiBUaGUgKnNwcmludGYqIGZ1bmN0aW9uIGNhbiBiZSB1c2VmdWwgZm9yIHByb2R1Y2luZyBmb3JtYXR0ZWQgc3RyaW5ncy4NCg0KLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCiMjIEZpbGVzICYgUmVzb3VyY2VzDQoNCmBgYHtyIHppcEZpbGVzLCBlY2hvPUZBTFNFfQ0KemlwTmFtZSA9IHNwcmludGYoIkxlc3NvbkZpbGVzLSVzLSVzLnppcCIsIA0KICAgICAgICAgICAgICAgICBwYXJhbXMkY2F0ZWdvcnksDQogICAgICAgICAgICAgICAgIHBhcmFtcyRudW1iZXIpDQoNCnRleHRBTGluayA9IHBhc3RlMCgiQWxsIEZpbGVzIGZvciBMZXNzb24gIiwgDQogICAgICAgICAgICAgICBwYXJhbXMkY2F0ZWdvcnksIi4iLHBhcmFtcyRudW1iZXIpDQoNCiMgZG93bmxvYWRGaWxlc0xpbmsoKSBpcyBpbmNsdWRlZCBmcm9tIF9pbnNlcnQyREIuUg0Ka25pdHI6OnJhd19odG1sKGRvd25sb2FkRmlsZXNMaW5rKCIuIiwgemlwTmFtZSwgdGV4dEFMaW5rKSkNCmBgYA0KDQotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCg0KIyMgUmVmZXJlbmNlcw0KDQpbSG93IHRvIHVzZSBwaXBlcyB0byBjbGVhbiB1cCB5b3VyIFIgY29kZS4gUmJsb2dnZXJzLiBNYXJjaCAyLCAyMDIyXShodHRwczovL3d3dy5yLWJsb2dnZXJzLmNvbS8yMDIyLzAzL2hvdy10by11c2UtcGlwZXMtdG8tY2xlYW4tdXAteW91ci1yLWNvZGUvKQ0KDQojIyBFcnJhdGENCg0KTm9uZSBjb2xsZWN0ZWQgeWV0LiBMZXQgdXMga25vdy4NCg0KYGBgez1odG1sfQ0KPHNjcmlwdCBzcmM9Imh0dHBzOi8vZm9ybS5qb3Rmb3JtLmNvbS9zdGF0aWMvZmVlZGJhY2syLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPg0KICBuZXcgSm90Zm9ybUZlZWRiYWNrKHsNCiAgICBmb3JtSWQ6ICIyMTIxODcwNzI3ODQxNTciLA0KICAgIGJ1dHRvblRleHQ6ICJGZWVkYmFjayIsDQogICAgYmFzZTogImh0dHBzOi8vZm9ybS5qb3Rmb3JtLmNvbS8iLA0KICAgIGJhY2tncm91bmQ6ICIjRjU5MjAyIiwNCiAgICBmb250Q29sb3I6ICIjRkZGRkZGIiwNCiAgICBidXR0b25TaWRlOiAibGVmdCIsDQogICAgYnV0dG9uQWxpZ246ICJjZW50ZXIiLA0KICAgIHR5cGU6IGZhbHNlLA0KICAgIHdpZHRoOiA3MDAsDQogICAgaGVpZ2h0OiA1MDAsDQogICAgaXNDYXJkRm9ybTogZmFsc2UNCiAgfSk7DQo8L3NjcmlwdD4NCmBgYA0KYGBge3IgY29kZT14ZnVuOjpyZWFkX3V0ZjgocGFzdGUwKGhlcmU6OmhlcmUoKSwnL1IvX2RlcGxveUtuaXQuUicpKSwgaW5jbHVkZSA9IEZBTFNFfQ0KYGBgDQo=