From 1db45630ac82ead30e58057f87cfdb82dc4ed6c2 Mon Sep 17 00:00:00 2001 From: mikahanninen Date: Fri, 10 Nov 2023 12:57:36 +0000 Subject: [PATCH] deploy: 8c048fa9b54ecb128a746f2e45182c8470e6917d --- .doctrees/environment.pickle | Bin 1082380 -> 1082380 bytes .doctrees/libraries/http/python.doctree | Bin 309637 -> 309637 bytes latest.json | 146 ++++++++++++------------ libdoc/RPA_Archive.html | 2 +- libdoc/RPA_Assistant.html | 2 +- libdoc/RPA_Browser.html | 2 +- libdoc/RPA_Browser_Playwright.html | 2 +- libdoc/RPA_Browser_Selenium.html | 2 +- libdoc/RPA_Browser_common.html | 2 +- libdoc/RPA_Calendar.html | 2 +- libdoc/RPA_Cloud_AWS.html | 2 +- libdoc/RPA_Cloud_Azure.html | 2 +- libdoc/RPA_Cloud_Google.html | 2 +- libdoc/RPA_Crypto.html | 2 +- libdoc/RPA_Database.html | 2 +- libdoc/RPA_Desktop.html | 2 +- libdoc/RPA_Desktop_Clipboard.html | 2 +- libdoc/RPA_Desktop_OperatingSystem.html | 2 +- libdoc/RPA_Desktop_Windows.html | 2 +- libdoc/RPA_Dialogs.html | 2 +- libdoc/RPA_DocumentAI.html | 2 +- libdoc/RPA_DocumentAI_Base64AI.html | 2 +- libdoc/RPA_DocumentAI_DocumentAI.html | 2 +- libdoc/RPA_DocumentAI_Nanonets.html | 2 +- libdoc/RPA_Email_Exchange.html | 2 +- libdoc/RPA_Email_ImapSmtp.html | 2 +- libdoc/RPA_Excel_Application.html | 2 +- libdoc/RPA_Excel_Files.html | 2 +- libdoc/RPA_FTP.html | 2 +- libdoc/RPA_FileSystem.html | 2 +- libdoc/RPA_HTTP.html | 2 +- libdoc/RPA_Hubspot.html | 2 +- libdoc/RPA_Images.html | 2 +- libdoc/RPA_JSON.html | 2 +- libdoc/RPA_JavaAccessBridge.html | 2 +- libdoc/RPA_MFA.html | 2 +- libdoc/RPA_MSGraph.html | 2 +- libdoc/RPA_Netsuite.html | 2 +- libdoc/RPA_Notifier.html | 2 +- libdoc/RPA_OpenAI.html | 2 +- libdoc/RPA_Outlook_Application.html | 2 +- libdoc/RPA_PDF.html | 2 +- libdoc/RPA_Robocloud_Items.html | 2 +- libdoc/RPA_Robocloud_Secrets.html | 2 +- libdoc/RPA_Robocorp_Process.html | 2 +- libdoc/RPA_Robocorp_Storage.html | 2 +- libdoc/RPA_Robocorp_Vault.html | 2 +- libdoc/RPA_Robocorp_WorkItems.html | 2 +- libdoc/RPA_RobotLogListener.html | 2 +- libdoc/RPA_SAP.html | 2 +- libdoc/RPA_Salesforce.html | 2 +- libdoc/RPA_Slack.html | 2 +- libdoc/RPA_Smartsheet.html | 2 +- libdoc/RPA_Tables.html | 2 +- libdoc/RPA_Tasks.html | 2 +- libdoc/RPA_Twitter.html | 2 +- libdoc/RPA_Windows.html | 2 +- libdoc/RPA_Word_Application.html | 2 +- libdoc/RPA_version.html | 2 +- libraries/http/python.html | 12 +- python.json | 2 +- 61 files changed, 136 insertions(+), 136 deletions(-) diff --git a/.doctrees/environment.pickle b/.doctrees/environment.pickle index c066310f8c478f5ac5250044f55d2ac36a5cf0a6..ee3bd4e51d47fd081e884a7181f1c1cf8a5e4df1 100644 GIT binary patch delta 60173 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

3$rnGFVbB@`yp%Z+Xun4>W{Ew-#*6yMPv+w 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 delta 60505 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%0AD`mAzEN7({4T95YfABU&(+Z^RfCP--iRh@-H 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%0AgEGr}tq!YlK^E7QR% zTV${JbFXZWy|O>{$~5qbXZOnH*eibBD_di)cy+IAjJ@L1y|OL#ibwa#p4cn?-0P{s zp3E!W+$$SmudJ!PvK{t{C-=%`*eibAD_dc&cyX_6guUX!y|NAViU;?~CfFzol=jN**9*rDPv(`)uUGuGSGK-h@!DS5_du7|}6_4$eO|Mt{wO6*h zUh&pm+3$MAS9@i*>y;I>SN6JIanoMe>3YRUdwHK*j^f#9y|T^qiih^fCf6(e*(+OI zuXty#Y;e8eo4vBT^@?Zq%HGy13umwFY`x-?y|SlHWbl`X7ScCcRA zz>V@q=e%`e4h1KR!L?_iZtUVNXZZ@alSw*HEc}iv|>`y41_#ily(ms4y`9*G08)Bm`4@SJjcz zaa~pSP(CX1)TPP+xCs0}SCy*?4TP}naaEn_c2k*QA{B>9q!{&RYB$xK^RMovx@hsH zw6vc}rHbx|_7Ud_s-euvOTNTJx*l&tQ|s<3ne!EQSNS?$dLW;Z2GqhweW$uBr!FiD z4f_)fJJ1Ig0z$Eq5_+g`&1>OT*u95VIID;1tU0-fbo_JVs_dcCxTw25R6eKqJyijx zOM9w(E!i!R)+~hN$(|}zQ+oRQ7!9sXyL+PP<+EI2)UB6NTA(nIqTWN6w|k*uE0cm$ zxPOSCcuOzUiwn5aOZC*!LlSA$c1WMWqCPM$8M;zy6q;$u>_j^B2={PbG)r=y?i^H|68f6z=UD>vXk1@i z!I^zkH=Tch#ZJ^lu$Y^hO1tMHdS7}!6`|#pR_#9F&FrUK+}K6^OkIyzGR<|aC#?F` zrq=yclqRoCq=TbjfwvxV)u+Y%F&l2uvi_=HC?AG;NL&Ba{;+-_JZo8h&jGN0A#O;d z;#HWyZ8t&3{}#GxI{jmIlMWA1BX!PyXjSG(i3SZjV2 z=DY`*KbfxT)3`w@4z*h}NDW4Ee2^NX6}?Usqf|C-;Kf*uAFLXHxp1%=rZwzKr0it0 z@9JQc2S&yaH9|8Ep@(K+wC;{X`5jjT;Y;TsDopd!c7ynN$Eq? zAShZgR1M%-3^l7-nMhe1pzr8pRCV(vd|6pCOgVK8Kcl6CQA1E`(*9w}jf_WzsS%p_ z6;~f0bPiXspf4J(hU!Y3vMSM5Q_fnHjEFgNtvW3lp~fKZ*CW(OBrQj(p<4A7R^5RP zk5tiG!*#3CEw#uEEi!kMibmGuqttLM`G>!(d!v-U^gD?(-lMYU+7m8ZoW9a!SOd0V zv>JlkXGX)tibaG{_5@6V{4pv=6T+#}4BRii3Iqrq8l!q6i5#m6b!jzmSym!C{HFov zyn}08AvFBGpxRY>eb-p-?senL-foaYYeu0mkd5R($#tOa)p4pnlH22CWcn7Ven9Js zgm|e)YS?MK>I34c@gj0+yc!QwP%M(Gi%o;-K!+x&mNe%3pm3{=X%o~~UD?h_bp2_J zblc4^_k$C};J%$88q+6AvUsBE3&lq?YB)*smrRnHygfghbQttMvR8*p=8%u5PPruv+vsn#){_KY)iluc8$wT=Zz z`k8QY490XH$SAjRnsiLubh8g0P0~-L)(a48d&6{93qpIQn~i^hiab!9@Pey~cNpc7 z`M=YnS~;2p(~ln2gVT)r)BsMG+@}gT{RC-PGk%0`Nn*boLt%p*QS^9@!xeDo5F1y5 z#~;$oC2EKkf5qQlbzXMi4ewT0l=&=!RL2>rZ>T8U%{CH47w^Me*r&I`2#dD4ob<^I z6{h+B;(m=GQF3IPE1oWwn5?}NH5RPyGvPl)*PG~VUIp|igR+g=@njv-71w3>hZV){ zpQ-wCB_m7KKu(KFRd=2By(C&X56YrnMwsj)eH;;VsuZIoYCrV%?T4?rnpIJjU8cHf zfuk~sqG|0eS7R#M;fku-F~`bOA1w( z*26&AJx4WU*<*9y%0>3KBr0apt~(DQNCwPRb#y6zP}FR!I@&G9?_(8^ zS>R}4HXhpU%vE`;wc|Wg8tKA$SRNqV@5i_2VUZ#VRI;8(R~MqR()(2`i>5L9l^!sspiz-0aZumYmiK{UW9j;`GD?@%@3%Nn$sv*+j_x6&^Y-iM~pdfrM#xe z6m=Q-isq|$E@H`ijInT1lJ!DXJ&XlOgMF~I#ZNoJ>EJUCHNGiyzmUNYHUvw`dG*UY+AN?5Vb?7HFBAf|A|8!S?7bc@HLNdtKK z*%Moh5&L1BOBSgZU7zt3zeqK!J$v%h>7_Gm9WfC<;8YFBdC1ks*0WRm+P-@fLj+4~ z#D$Grta5c>C4P3%gD4CGl*H`Qi%nCTovfD*-O5!9oncdIL;}AKrzVGHcqL(w1Pi)tsl`BmR;{EL4%Sc?m9@7Vz03n46)JcP5w@2yXPg%SD?XfenAXLPhA@|1fjcr(?Gr z)dOA%THby{b%wTekE*PoI+XpW>V;&%qv%BHFk<^K&X zS*UMu@7ASBe?xZ*VShl(MZ5o|qBZrP7JKOk9{hKaSr_sj|K#b?Fu0bOs5ua#>3#oD zj$5W0p>9i-!SxB_h=0`2EW@Y^<(PjoZ)OEY1-3W*{eb5LWiLSwoqQY*h_Yia*2*XF zU;J33y5mcK6aV#?$_W$UuQdfj*isWtLT@i*f8#M^8c^w~ZO)a;2;}^@D$r8jC)1k! zx{w?PI8k2JLvt?p2Pn;p{FPqpo0fagB~sWW?vfaK$EzA^;j8}MxZ{N%gPOP}&|ne0 zi3b0K0?MCIF=6~6gy)ZBeHJiuc(6g0pJ54ba3^|Icz2TdRG>Z$8y>8v;Dp}@9cLYs zxg1X>q>Gm0K_jxEwB%W=)EA8mZfs6Vp@*mF2W!>}m8`W!rf9F8_77~E7q392MNp-% z^F%XNs6uP_4K^7f+nuhhQTep->EOsJUOU3Kx$ zHNt}8Y0y(@aA-VNek_e&r5e-eD=v)cu3*3W5Y~8qD{8HURS0DwTER>Ic|W*Xm&bR4H27YHx6TN?WI5br~f* zvuWvaSjmPRFt#uE2fG3t5l?A^ZWf{WDS8wC>(7Y!KH!t!5ZXcTL&9626`Um1OcNH< z^?#reBG&MIT<~ryZz)x7(%gtDMXi4tyMcJR__P{`q~kM^tbPXjl6d+SiProiU3*%! zpvupK1Dol1TJS8~y42(u%U;9i^0O)dncdG}ItX_YWo|I5RdE@Eu<1F>Dq%fuS)ado z5D2STGTquvirkB|rswL9J~*qeXgq6=~NbrBItQ>s3!ADH~LO zo%c1}8wansBF#A}d3`B-7!*UNHkiHomX`eET5zOg5PdhQLapeaX7~HWRkd5sY?N-z z+Jp^8JT2ZN{dss3T$q${%xZTg_(hws_7KMBbmkc}bn^|YJ-*wlYH8|!x%>UpOVvZ7 zY&HJIva}57TD?VzJh=tiyLd`}UiHyMp0mUdtJDZlwEuZ?7=EhC+i}aKtcDu@FfIR@ z<$|14_PeXTCA8rm(%MV^z;+Wx^nw(<@&&P_Q!n5UNZRH|4Y)s1V9>}HRg`XFb=vbb zdVTvdm@S)LRCRR@XR3D5w{Bb8`$W3*qH2d@*2I@o7bN$+r25n9=2*!bcnMYIFVj5H zsoMP|ysVN#g;I~UY{lB|&vqdZX0;P(<;$pOBAt9$bwU!qRplco+NwGrS-(|vMsjMa z>WHM_HsOuk1~(!4TBg#T2Qj_h*#={W*fuTpit3>G9aFVqzyB4L0{)g)&Ui^afg zb;Cm3fw`DS8QWC{UEN-~x>rNoRjRvlyHx4=b{LE38bm>_qrcV-$6%^Zw~NYN#fS=P zXevc{;KtSt4>9QKL_`VS1)k0}f;#O`iS+%eFfS1vO~-aX&qLdATZ-Wc!K!y^h-J(O z8@@wYziOQTHzFMT9(Cld%#& z#CWGWK00Ia%&F67PBz_?2+d2SSyPQ#v|<-ddW13`6h0ao>U1Yi&~8;bRLBdo$q04f zvXF3dQV4T#s(!4M?^c14C`l-5Y1>}aI1rnj?(IdF3+HK=_Zrx6>P}Td_^S+eU}*!bMgPJx zS0p#4>ZQ%Ge_@Rzlr5=zn1xHIUc)k0C@)|D`8jjIhoDUyrk*NZ!?Y0I%YWgea;5LQ zhOOfT2j#u4hKKS;m!5ypwLRF49DH4+Gvn&(IAjsV8{CGrw0N&-#*7#Ds$$J}hnBJ- zyd3RLroKMdi-;czP-AJ84?9|+zH3#YwWfRk3U3*^)pN%L-Yf72pPnPBH2ocHRJOl? z`)hcu7$)RnsrsPy#W%3z6w0TmlzjyUthe64agk6yPo<@gg7SGicVlzrx~b$%)hkr^ z-_Wc#;TjRLy-tvO-CKBW3i*t5aRx1VOV!tuAH*VJ>AttH&DwoF1iR9A(L#|vrGlzraiJQESHAs)d&PRZDU-c1z!@Z^IQ!hTE{J z&tWM=`_b7M2&%5h_bhTM3SPb+n;?;|o~HNkAHEHF_d75HAv@D_e-Cfu4s17nb@pftr_naAX%gh;#>MT8oBW}sx3>taeUvLOoBnJAsf49b2N1B?jgSQ#QJx~67R#eQtz#{UOf0K_@3We$jBX_=-^D9Zk= zvbFj)mioFR@y^pfQej-s<$q&Ej5Pm0IMYJ9>OWX1BR%sU)tl4wcU5n#qgPcOh+SSs z*Q5_tMQG2dcU7t;^smw(`B(z#+=Hs-=CI{~Z6gZV@}BCi#YUtBY$=pt)7{N&4IBLz zZf$x65zTEPTRbGSQKEQ#EuqOG9>|6#W|KeoBKQL4M`YJwv z7eQ4Ie1Pd78D<0;9zzuas=I6jmFa>&$8$N$9bsOIIR&4_iG^zV@^CCx*`u+ z6{$-IWL{U37h+KPHRHyQRI@PtRNS*PjRrmj`#$v%j`fA|s5auXkFhd1M`a&lyCL)^ z0uuqR*N(w%IC{7_GtW`%VZ1{*N2Q1HkQVtTQH7~SUHY)SJB|Z{t93_EJ7KN!4?+LW z@esT6F-CsIL%5uwthF2W-9QR@MjTP2web2heL!7#1aEuJk@`dp(v0W-KUxqSypFEy zi`3|_YDKs|} z0!8t0bJiD_%N%Cg92k`IDc*aF(ATu-IF3L$kTzxRKs@SepQ5V5I%ye3DE+y$dL&B) zmg-s5?lUQ{{4>1d$)b-nQYUcSBiS#Z1NiOlyc4Q9`0G!=I)(AG?%S94S5Gu&n9y%p zW7k4U-Xeo5hv3<$WBDN9oUf0n0__ ze1UbQWUGM}@BGz3!IyAlLUE=0->4i%cOkxYT}?Ksk`dfGDo}}a|HV6E$r_!`d(s$M z{~3a(BmQ(9T8#lmd8_{m%NLQ@s%mUetRo;7`;~NM@mHpJGph{^5p&NRlB9I)>6ce> z9XV8X{jXuNQbH;<`U;&j{%d$fA!Jl7@@S#ksyV&$waRteLin_=RVPk6e1pRV{wav3 zL%Lq!=ks~#?r+dJLe5Xu-)^LyMW1oJ=zy0+?u>8oh5+f}Z?O#%@t*10+xGev;v65E z+$Zsb5ndrh{fPJS2nZecdH{LHpH!VSe;~Ibh7O!mNm|*Ebp4b*bsow%BsAR~0E0;X z4!t58hfDWI)8Aqd>RJR4Gr9)#{Si+hs9yRV23-Wlrt9^g2nM1>lk1eJX*@LX;&9YA z_yd7@G(Nza==d4}nxAhh|c zR(<)bR-N~~nQ3LZUdMMI=}t0dp=hhXNb}bXtNw>S!9U>p0UOaHG#WM1 zmDz6TPSlj0mSCa`ye99#n6V3Ncn$}HQt)g3N%7@4catjZSbYwMS|ae4uGzhFrupr+ z%&(C|ZKfILO?x?Lje9hQXpV@PQ95b*lK0Y5Ms-Cp}AEW>+H%1^2Vl5sy_^VNe^{e-vdJ?PL+H~<#u z@6lX-xFB?L;5exVrCr1>vj>%2RIRnxrF7ak24V*;VudUjulgxhL4j!SB|Ou5P|Hi$ zMhX9xmBUHPFX0OnA^(|&+EgeVMckp!I&eDir9PmS^!|JQ> zLz1iM+G4C!u3p94Z^4Ol`nqaFcRlXFw@N}!G3CNJQgLl_G6^p|L%+)0bxk$VTC%hj zf6!vZ&5pW`t$~QQ&d@XYNheaUDx$q~Bi0dpr@$&(JSVkKRBp_vn#MT?&LRXJ-< zo{vMyfIATObgPRY;-Z6}j+7U6kx`8IY1d6aes?lq6f z?%b{pjoc7em_9wynUW$MnF z_fS})Kuem^Ck|soSW~tmXC@u$ijP>k1sk2oXpMJ^5t-DeEhy(5h6J#!L*tq3w#bdC zd5FjOs?4P${ZDZ3!{*IXG=CNN%ofN+wt; zB-pHuMQb{3$=Za1!ik+M17tuUZFVI>?8j>-hM!-`vCB&E>rpry!nGtR_ zuzMDznOvV-TJS)a_2N|+Ihs-6H0tSsT4n0(+k@dK2oY!l$HfRR>@-H`V%pPgCz^*S zv@Y}U*ObPF8w1eTo#Dn%E!CB_MC0+3P{U}Yb@#CPsY(diz{#FesCTB`;l5wP$b+`( z5x8QzDNTql#vwTrVUFc6t3O(UF(cCKfiX0Y3q2iav;e)f%P2xJ*JVsZ@?V!R1qpV9 zlaRdNHqwJy(Wh>sTUaZ$k2#rDKAC8vjAr1DRYnIS&nlxUl2b^uY&pv3Hn*DSty(@m>T|^oVCL}`@R#c0DqaaxDj!@3@87!D<%5iU4Z!C}rKSfKiT6-5 z{^F>zj*$hmLD5EUBz>Zd{z%qG8v~JC(Mds!F;ExzB}|?_rrH@}q=5fbj4=dBMqPLd zQF_{6=VdQh$H4fEuddNU^Z(~Ju=sjLYfZU;it^L`CL)l!<`&Heyk)3QYwE#R@$q3j zqerM{|Cvoc(Fzh8aQec%$&)UI=5KSiG^f7MZ?Wwra7o8hw$J#u|N)yd7&4Ah{E3bVgFp(CCHaiH69RO2--+okRKF z0Z$xVi#L)vsAh~NG@>odP@(8nh?yiYJ6Cmb)-Y#wDy@%0uSw3tEN$fX;$Y-LNy*Y@ z#xM4A)^wMY;_Zw1e8X9MyjfxvZHYsP2X@ru4Qb^=Rj0XQPMJ56Rbw#xNw=Nup_Kk})JqRIJ095q}}RA<0OvP7WcNnrxP`o+=WJHWH`t6F%LQ ze49{pK1g`?Rez91N=U)i6V)gw#Yhd~%PTxDLv<8-urvh^-O`U$^`KXDeg6qUsEF*) zgqwjpyZv_8G1X|MMgGO37enQ#MzU7B7avRWmzL#eXylb~%B9I~`up;|RFF@7glo;? zAHf1hig`OrzjGDdS1YhQ7V-g0QzOScL{UtG3lPqG{tP$Mj0DGT4vI-P`i1fx9G*{f zjhCbwdCWMNj>U{HKGTf!3?qXXV>65)n(-ACWg0E1^1QPa-Oa$MtcZUbpgQT^PtL%x z39^pPGzPJXU71F&FcJPSOD{F+WEn}!$-yA!%H^Qm%h&$F?#-o(- z9Af~Nc09)@u45~+ZyMmUYTlwxZWnHSXYZtgt*}UOQdDc$yOWArW6C*s z%5+5C`04jfYaqrR<52 zInzqM!MWjmng*+UflxFMI zZ2Nqp;kph+0+-dn7^cPNW$Wn=0|}?iZ83Ea$_+1vGK@$%-T{3F;f`j-7DE$%;ncB% z(TMVABEA~plPQRN(h=)kDgM#FRI)Y)e8Wfl`9?V87U#jii`*00dI@$q4-eF~6rT@+ z7Dfda{0${6t|1{bo6}q&2=_^Uwr}&XpJ+>Q1!g;*$=0WL7dj#vg0;nQtqS0bCEKQK z{kr)4kM1-Y*9q1w_b0rF;%H}P zKT7T|p>`O&s_&}1jybzRg6TvLV?bC4*YcFVT%3}(rmNUpJI?wkIGjpxK=$pPXt~Iq zN6Yyu+3MHbF_d{6K^U`+hP!B7FT<$?E<%7`-gf z_+yc|VNs$I>ZCV|YgyW1OMFYG($2;Q@eFErX{O0^tq^T?os zl@WrKMONvCD}~tENaj#lGZIUR%KmtaA>>$u*@?mo&(Ut~#XeYC3nel~d(mdUyXzCe zmer)VzHs=$Q&c<%s#5%_LI}~zTsD0*a{@Kh0nQ|j8SRG(ZqoaGjUk-I_A`cSl?^DH zPw>{=gfCeA3;KOmKeJQfP&amS@HHJcle<6srPQNoj^1~zy@`2w?6+`(p6zec)e7;e zL4R`-wQ!1GP1FE9)u5)+0QgZ+la-@S@3QupE7`UuOt>z?H#94NV@YGZV;Znli>;_i;n8RFSMjIeKw>X9V^; zJBMJoEaVB4FbEal;9-Q|VG(q12s&6;lXLU}`Ga5;XD&sBF-;FjVu*^f>bhd6H2CmP zqn*|~OKY~Gh2;-}p@@RH6g3)!f))5$Pv{R|3(p&!&BL(K$)&@?j9i^z5nLZKBo8+l zXu=ZizeW_>=`R`{$t|}BeMualPNSU57 z@R_-^b&Qc2^aK5OjL`#0!dNU{_zpBrCEAK#RLJsv*;pe}Q$O<$xCF?$HWs5L!e3D3 zFwDVa+iF;=k2J~|hYd{{m5+nhPh*olqjl2Nad2ru`2q8gKUi8_1TURXQw4sZmqukp zn5-i7lRw*uA|pprF2l%qhyC?bm;$0t1^l|mU-QeCjK^js4I$9+ycNc;=nZ~4jhc)< zeKyvpVI3=_QCcx}k!e&~jLldY?Jvf<1s~r{FuI3I-s%)}Kb{qR$06&1XcagG6{fR= zc3KA}VC2&1w+TicU81{%|1;)^IKE4x)f4f=PNU-!;Xou~Obh)`_^QazQKk?}%KCIL zjfy709f)K@+FXkAQzoE%#DS`AOe`&#H)CetTnz8WCrNkbO@=FwEX`YJ@3eWck&04} zOooR^Bi9rx#e|>PLc4AkcLxWDTK7>6MH{AIb_+kJg`VBpCu+WvP6c?ake5$;dKPP5 z7$rK((%bg292F+xKQ4GQ;k7L-kEBSLZ=q`Wmn^AMxCGxYC8?%oWbVN3+3^* zYOyo1Tii7brvMprcp9bz4%Vk*65w2SI&3zBR!_&w$e<(Bjl59FIhu}7H=0RMCDRQF zeH;~+fgXsaWcs2ETIDf%>QX1Q(2MY2JTP*}HPzql{QI!|5{idg6hmJ$!M86P??e9x zb%wtWewv2*B63n)EnLPU)r=4e3I4LF3LGhBP*#bwqO8Q|q%}O~AEBcq*tUw!g}TP+ z%~jylUIyjOfaAo+cr#$38FX+4Ygsqb=%%wQ*X5PXG-APBITKrVQMD#eX*VIN5xz|= zHEaG<)tZABL48XxSQ#|BRBHHKDa>A!Y@q9NFthKK8u4Jqmr2KuDZ_ptgEp6$?bwEI z+xgdOcgt{^Ciz}%p+7xdn;W459})b5UsBA1_3@pFp4VIGIsE=CJb#4pMhm?hj+~A2 z4x#J=g%6?n&o<&5$Xzc-$lV(H9T^&mEiNKZd#4Av6SE%mN-z(a6rlBb?UkpMb% zGVURqvx2uI5z7~fL6m4XFWkw9tL0xVgD)2yZb;N|23BbzzKLTnY07A@4p?^$T{ zW_$-}sN`r#L5r|6=dd^C>IK)%tX*V$5fs#jO3IBEjvM&It{l&+JUUTs%x=QhDS6g) z3#zv4DbF)V zuOdT+M4gy-HEAt!ohz7Oni;sMNpiH&M7K@?*E9*wyrfC1G4E*7lJ#`LpBq56c|nsV zhULf=e=G{yhnO8GuWZulAk$EwbM(vRB z^^$x{h9qz=lc2z@Oo9S;G6^zoWYYP}3z;-B?_<&^a2=E6GcRM(Jh_Xh2$wPG_L_Gw oX=Zhuu%ANJLUqzzCp_IjA((2(gD{TM|A{R2Wg{r!SLVjixJF6yz9W0iF_ zi%T*yVo}YHrh2oN)H*3*)M=PXGhtDOY5VktjLcq}_nW^I#G+gP&3g68d>)e13z8X4 zCaZbP#G*kHT|?}Y40a$E>oJK|#|#V%@9B8|G{9o45QedulO5HCH)q$p#1?9}e0ibn zi2xSU@%Ykc0xf)LF_F(0i#u@nvgZg%zWjKMR9^<)poK3ZZ#rU&OY}J3eu|UvCke5* z-JXY$L=R41D9cC#A8ubR$EZe>7ytS(z9+?t8G(%ZNXn0Au_Px7^J4p|6hN*PQ{sAGL{(iwAW7#S*v_Z}yT}Cq;xZ4O3|*xH3FePk+eB?6rBn`Ab2#YDrvnt54?hkeptS%xE%M z&1)uHy(Uieu~RbGfmp1^Bw8I747h!!hD=l diff --git a/latest.json b/latest.json index 6e22eaa30a..355fdd720e 100644 --- a/latest.json +++ b/latest.json @@ -4,7 +4,7 @@ "name": "RPA.Archive", "doc": "

\n

Archive is a library for operating with ZIP and TAR packages.

\n

Examples

\n
\n*** Settings ***\nLibrary  RPA.Archive\n\n*** Tasks ***\nCreating a ZIP archive\n   Archive Folder With ZIP   ${CURDIR}${/}tasks  tasks.zip   recursive=True  include=*.robot  exclude=/.*\n   @{files}                  List Archive             tasks.zip\n   FOR  ${file}  IN  ${files}\n      Log  ${file}\n   END\n   Add To Archive            .${/}..${/}missing.robot  tasks.zip\n   &{info}                   Get Archive Info\n
\n
\nfrom RPA.Archive import Archive\n\nlib = Archive()\nlib.archive_folder_with_tar('./tasks', 'tasks.tar', recursive=True)\nfiles = lib.list_archive('tasks.tar')\nfor file in files:\n   print(file)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:32", + "generated": "2023-11-10 12:55:59", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -462,7 +462,7 @@ "name": "RPA.Assistant", "doc": "
\n

The Assistant library provides a way to display information to a user\nand request input while a robot is running. It allows building processes\nthat require human interaction. Also it offers capabilities of running\nother robots inside the current one and determine what to display to the\nuser based on his previous responses.

\n

It is not included in the rpaframework package, so in order to use it\nyou have to add rpaframework-assistant with the desired version in your\nconda.yaml file

\n

Some examples of use-cases could be the following:

\n
    \n
  • Displaying generated files after an execution is finished
  • \n
  • Displaying dynamic and user-friendly error messages
  • \n
  • Requesting passwords or other personal information
  • \n
  • Running Keywords based on user's actions
  • \n
  • Displaying dynamic content based on user's actions
  • \n
  • Automating based on files created by the user
  • \n
\n

Workflow

\n

The library is used to create dialogs, i.e. windows, that can be composed\non-the-fly based on the current state of the execution.

\n

The content of the dialog is defined by calling relevant keywords\nsuch as Add text or Add file input. When the dialog is opened\nthe content is generated based on the previous keywords.

\n

Depending on the way the dialog is started, the execution will either\nblock or continue while the dialog is open. During this time the user\ncan freely edit any possible input fields or handle other tasks.

\n

After the user has successfully submitted the dialog, any possible\nentered input will be returned as a result. The user also has the option\nto abort by closing the dialog window forcefully.

\n

Results

\n

Each input field has a required name argument that controls what\nthe value will be called in the result object. Each input name should be\nunique, and must not be called submit as that is reserved for the submit\nbutton value.

\n

A result object is a Robot Framework DotDict, where each key\nis the name of the input field and the value is what the user entered.\nThe data type of each field depends on the input. For instance,\na text input will have a string, a checkbox will have a boolean, and\na file input will have a list of paths.

\n

If the user closed the window before submitting or there was an internal\nerror, the results object returned by Run Dialog or Ask User won't have a "submit"\nkey.

\n

Layouting

\n

By default elements are added to the assistant dialog from top to bottom, with a bit\nof margin around each element to add spaciousness. This margin is added as a\nContainer you can manually use Open Container to override the default\ncontainer. You can use it to set smaller margins.

\n

You can combine layouting elements with each other. Layouting elements need to be\nclosed with the corresponding Close keyword. (So Open Row and then\nClose Row.)

\n

Open Row can be used to layout elements in the same row.

\n

Open Column can be used to layout elements in columns.

\n

Open Stack and multiple Open Container's inside it can be used to set\npositions like Center, Topleft, BottomRight, or coordinate tuples likes (0, 0),\n(100, 100) and such.

\n

Open Container can bse used for absolute positioning inside a Stack, or anywhere\nfor setting background color or margins and paddings.

\n

Open Navbar can be used to make a navigation bar that will stay at the top of\nthe dialog. Its contents won't be cleared when.

\n

Examples

\n
\n*** Keywords ***\nSuccess dialog\n    Add icon      Success\n    Add heading   Your orders have been processed\n    Add files     *.txt\n    Run dialog    title=Success\n\nFailure dialog\n    Add icon      Failure\n    Add heading   There was an error\n    Add text      The assistant failed to login to the Enterprise portal\n    Add link      https://robocorp.com/docs    label=Troubleshooting guide\n    Run dialog    title=Failure\n\nLarge dialog\n    Add heading    A real chonker   size=large\n    Add image      fat-cat.jpeg\n    Run dialog     title=Large    height=1024    width=1024\n\nConfirmation dialog\n    Add icon      Warning\n    Add heading   Delete user ${username}?\n    Add submit buttons    buttons=No,Yes    default=Yes\n    ${result}=    Run dialog\n    IF   $result.submit == "Yes"\n        Delete user    ${username}\n    END\n\nInput form dialog\n    Add heading       Send feedback\n    Add text input    email    label=E-mail address\n    Add text input    message\n    ...    label=Feedback\n    ...    placeholder=Enter feedback here\n    ...    maximum_rows=5\n    ${result}=    Run dialog\n    Send feedback message    ${result.email}  ${result.message}\n
\n
\ndef success_dialog():\n    assistant = Assistant()\n    assistant.add_icon("success")\n    assistant.add_heading("Your orders have been processed")\n    assistant.add_files("*.txt")\n    assistant.run_dialog(title="Success")\n\ndef failure_dialog():\n    assistant = Assistant()\n    assistant.add_icon("failure")\n    assistant.add_heading("There was an error")\n    assistant.add_text("The assistant failed to login to the Enterprise portal")\n    assistant.add_link("https://robocorp.com/docs", label="Troubleshooting guide")\n    assistant.add_files("*.txt")\n    assistant.run_dialog(title="Failure")\n\ndef large_dialog():\n    assistant = Assistant()\n    assistant.add_heading("A real chonker", size="large")\n    assistant.add_image("fat-cat.jpeg")\n    assistant.run_dialog(title="Large", height=1024, width=1024)\n\ndef confirmation_dialog():\n    assistant = Assistant()\n    assistant.add_icon("warning")\n    assistant.add_heading("Delete user ${username}?")\n    assistant.add_submit_buttons(buttons="No, Yes", default="Yes")\n    result = assistant.run_dialog()\n    if result.submit == "Yes":\n        delete_user(username)\n\ndef input_from_dialog():\n    assistant = Assistant()\n    assistant.add_heading("Send feedback")\n    assistant.add_text_input("email", label="E-mail address")\n    assistant.add_text_input("message", label="Feedback", placeholder="Enter feedback here", maximum_rows=5)\n    assistant.add_submit_buttons("Submit", default="Submit")\n    result = assistant.run_dialog()\n    send_feedback_message(result.email, result.message)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:32", + "generated": "2023-11-10 12:56:00", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -2671,7 +2671,7 @@ "name": "RPA.Browser", "doc": "

SeleniumLibrary is a web testing library for Robot Framework.

\n

This document explains how to use keywords provided by SeleniumLibrary. For information about installation, support, and more, please visit the project pages. For more information about Robot Framework, see http://robotframework.org.

\n

SeleniumLibrary uses the Selenium WebDriver modules internally to control a web browser. See http://seleniumhq.org for more information about Selenium in general and SeleniumLibrary README.rst Browser drivers chapter for more details about WebDriver binary installation.

\n\n

Locating elements

\n

All keywords in SeleniumLibrary that need to interact with an element on a web page take an argument typically named locator that specifies how to find the element. Most often the locator is given as a string using the locator syntax described below, but using WebElements is possible too.

\n

Locator syntax

\n

SeleniumLibrary supports finding elements based on different strategies such as the element id, XPath expressions, or CSS selectors. The strategy can either be explicitly specified with a prefix or the strategy can be implicit.

\n

Default locator strategy

\n

By default, locators are considered to use the keyword specific default locator strategy. All keywords support finding elements based on id and name attributes, but some keywords support additional attributes or other values that make sense in their context. For example, Click Link supports the href attribute and the link text and addition to the normal id and name.

\n

Examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Click Elementexample# Match based on id or name.
Click Linkexample# Match also based on link text and href.
Click Buttonexample# Match based on id, name or value.
\n

If a locator accidentally starts with a prefix recognized as explicit locator strategy or implicit XPath strategy, it is possible to use the explicit default prefix to enable the default strategy.

\n

Examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Click Elementname:foo# Find element with name foo.
Click Elementdefault:name:foo# Use default strategy with value name:foo.
Click Element//foo# Find element using XPath //foo.
Click Elementdefault: //foo# Use default strategy with value //foo.
\n

Explicit locator strategy

\n

The explicit locator strategy is specified with a prefix using either syntax strategy:value or strategy=value. The former syntax is preferred because the latter is identical to Robot Framework's named argument syntax and that can cause problems. Spaces around the separator are ignored, so id:foo, id: foo and id : foo are all equivalent.

\n

Locator strategies that are supported by default are listed in the table below. In addition to them, it is possible to register custom locators.

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
StrategyMatch based onExample
idElement id.id:example
namename attribute.name:example
identifierEither id or name.identifier:example
classElement class.class:example
tagTag name.tag:div
xpathXPath expression.xpath://div[@id=\"example\"]
cssCSS selector.css:div#example
domDOM expression.dom:document.images[5]
linkExact text a link has.link:The example
partial linkPartial link text.partial link:he ex
sizzleSizzle selector deprecated.sizzle:div.example
dataElement data-* attributedata:id:my_id
jqueryjQuery expression.jquery:div.example
defaultKeyword specific default behavior.default:example
\n

See the Default locator strategy section below for more information about how the default strategy works. Using the explicit default prefix is only necessary if the locator value itself accidentally matches some of the explicit strategies.

\n

Different locator strategies have different pros and cons. Using ids, either explicitly like id:foo or by using the default locator strategy simply like foo, is recommended when possible, because the syntax is simple and locating elements by id is fast for browsers. If an element does not have an id or the id is not stable, other solutions need to be used. If an element has a unique tag name or class, using tag, class or css strategy like tag:h1, class:example or css:h1.example is often an easy solution. In more complex cases using XPath expressions is typically the best approach. They are very powerful but a downside is that they can also get complex.

\n

Examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Click Elementid:foo# Element with id 'foo'.
Click Elementcss:div#foo h1# h1 element under div with id 'foo'.
Click Elementxpath: //div[@id=\"foo\"]//h1# Same as the above using XPath, not CSS.
Click Elementxpath: //*[contains(text(), \"example\")]# Element containing text 'example'.
\n

NOTE:

\n
    \n
  • The strategy:value syntax is only supported by SeleniumLibrary 3.0 and newer.
  • \n
  • Using the sizzle strategy or its alias jquery requires that the system under test contains the jQuery library.
  • \n
  • Prior to SeleniumLibrary 3.0, table related keywords only supported xpath, css and sizzle/jquery strategies.
  • \n
  • data strategy is conveniance locator that will construct xpath from the parameters. If you have element like <div data-automation=\"automation-id-2\">, you locate the element via data:automation:automation-id-2. This feature was added in SeleniumLibrary 5.2.0
  • \n
\n

Implicit XPath strategy

\n

If the locator starts with // or multiple opening parenthesis in front of the //, the locator is considered to be an XPath expression. In other words, using //div is equivalent to using explicit xpath://div and ((//div)) is equivalent to using explicit xpath:((//div))

\n

Examples:

\n\n\n\n\n\n\n\n\n\n
Click Element//div[@id=\"foo\"]//h1
Click Element(//div)[2]
\n

The support for the (// prefix is new in SeleniumLibrary 3.0. Supporting multiple opening parenthesis is new in SeleniumLibrary 5.0.

\n

Chaining locators

\n

It is possible chain multiple locators together as single locator. Each chained locator must start with locator strategy. Chained locators must be separated with single space, two greater than characters and followed with space. It is also possible mix different locator strategies, example css or xpath. Also a list can also be used to specify multiple locators. This is useful, is some part of locator would match as the locator separator but it should not. Or if there is need to existing WebElement as locator.

\n

Although all locators support chaining, some locator strategies do not abey the chaining. This is because some locator strategies use JavaScript to find elements and JavaScript is executed for the whole browser context and not for the element found be the previous locator. Chaining is supported by locator strategies which are based on Selenium API, like xpath or css, but example chaining is not supported by sizzle or `jquery

\n

Examples:

\n\n\n\n\n\n\n
Click Elementcss:.bar >> xpath://a# To find a link which is present after an element with class \"bar\"
\n

List examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
${locator_list} =Create Listcss:div#div_idxpath://*[text(), \" >> \"]
Page Should Contain Element${locator_list}
${element} =Get WebElementxpath://*[text(), \" >> \"]
${locator_list} =Create Listcss:div#div_id${element}
Page Should Contain Element${locator_list}
\n

Chaining locators in new in SeleniumLibrary 5.0

\n

Using WebElements

\n

In addition to specifying a locator as a string, it is possible to use Selenium's WebElement objects. This requires first getting a WebElement, for example, by using the Get WebElement keyword.

\n\n\n\n\n\n\n\n\n\n\n\n
${elem} =Get WebElementid:example
Click Element${elem}
\n

Custom locators

\n

If more complex lookups are required than what is provided through the default locators, custom lookup strategies can be created. Using custom locators is a two part process. First, create a keyword that returns a WebElement that should be acted on:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Custom Locator Strategy[Arguments]${browser}${locator}${tag}${constraints}
${element}=Execute Javascriptreturn window.document.getElementById('${locator}');
[Return]${element}
\n

This keyword is a reimplementation of the basic functionality of the id locator where ${browser} is a reference to a WebDriver instance and ${locator} is the name of the locator strategy. To use this locator, it must first be registered by using the Add Location Strategy keyword:

\n\n\n\n\n\n\n
Add Location StrategycustomCustom Locator Strategy
\n

The first argument of Add Location Strategy specifies the name of the strategy and it must be unique. After registering the strategy, the usage is the same as with other locators:

\n\n\n\n\n\n
Click Elementcustom:example
\n

See the Add Location Strategy keyword for more details.

\n

Browser and Window

\n

There is different conceptual meaning when SeleniumLibrary talks about windows or browsers. This chapter explains those differences.

\n

Browser

\n

When Open Browser or Create WebDriver keyword is called, it will create a new Selenium WebDriver instance by using the Selenium WebDriver API. In SeleniumLibrary terms, a new browser is created. It is possible to start multiple independent browsers (Selenium Webdriver instances) at the same time, by calling Open Browser or Create WebDriver multiple times. These browsers are usually independent of each other and do not share data like cookies, sessions or profiles. Typically when the browser starts, it creates a single window which is shown to the user.

\n

Window

\n

Windows are the part of a browser that loads the web site and presents it to the user. All content of the site is the content of the window. Windows are children of a browser. In SeleniumLibrary browser is a synonym for WebDriver instance. One browser may have multiple windows. Windows can appear as tabs, as separate windows or pop-ups with different position and size. Windows belonging to the same browser typically share the sessions detail, like cookies. If there is a need to separate sessions detail, example login with two different users, two browsers (Selenium WebDriver instances) must be created. New windows can be opened example by the application under test or by example Execute Javascript keyword:

\n
\nExecute Javascript    window.open()    # Opens a new window with location about:blank\n
\n

The example below opens multiple browsers and windows, to demonstrate how the different keywords can be used to interact with browsers, and windows attached to these browsers.

\n

Structure:

\n
\nBrowserA\n           Window 1  (location=https://robotframework.org/)\n           Window 2  (location=https://robocon.io/)\n           Window 3  (location=https://github.com/robotframework/)\n\nBrowserB\n           Window 1  (location=https://github.com/)\n
\n

Example:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Open Browserhttps://robotframework.org${BROWSER}alias=BrowserA# BrowserA with first window is opened.
Execute Javascriptwindow.open()# In BrowserA second window is opened.
Switch Windowlocator=NEW# Switched to second window in BrowserA
Go Tohttps://robocon.io# Second window navigates to robocon site.
Execute Javascriptwindow.open()# In BrowserA third window is opened.
${handle}Switch Windowlocator=NEW# Switched to third window in BrowserA
Go Tohttps://github.com/robotframework/# Third windows goes to robot framework github site.
Open Browserhttps://github.com${BROWSER}alias=BrowserB# BrowserB with first windows is opened.
${location}Get Location# ${location} is: https://www.github.com
Switch Window${handle}browser=BrowserA# BrowserA second windows is selected.
${location}Get Location# ${location} = https://robocon.io/
@{locations 1}Get Locations# By default, lists locations under the currectly active browser (BrowserA).
@{locations 2}Get Locationsbrowser=ALL# By using browser=ALL argument keyword list all locations from all browsers.
\n

The above example, @{locations 1} contains the following items: https://robotframework.org/, https://robocon.io/ and https://github.com/robotframework/'. The @{locations 2} contains the following items: https://robotframework.org/, https://robocon.io/, https://github.com/robotframework/' and 'https://github.com/.

\n

Timeouts, waits, and delays

\n

This section discusses different ways how to wait for elements to appear on web pages and to slow down execution speed otherwise. It also explains the time format that can be used when setting various timeouts, waits, and delays.

\n

Timeout

\n

SeleniumLibrary contains various keywords that have an optional timeout argument that specifies how long these keywords should wait for certain events or actions. These keywords include, for example, Wait ... keywords and keywords related to alerts. Additionally Execute Async Javascript. Although it does not have timeout, argument, uses a timeout to define how long asynchronous JavaScript can run.

\n

The default timeout these keywords use can be set globally either by using the Set Selenium Timeout keyword or with the timeout argument when importing the library. If no default timeout is set globally, the default is 5 seconds. If None is specified for the timeout argument in the keywords, the default is used. See time format below for supported timeout syntax.

\n

Implicit wait

\n

Implicit wait specifies the maximum time how long Selenium waits when searching for elements. It can be set by using the Set Selenium Implicit Wait keyword or with the implicit_wait argument when importing the library. See Selenium documentation for more information about this functionality.

\n

See time format below for supported syntax.

\n

Page load

\n

Page load timeout is the amount of time to wait for page load to complete until error is raised.

\n

The default page load timeout can be set globally when importing the library with the page_load_timeout argument or by using the Set Selenium Page Load Timeout keyword.

\n

See time format below for supported timeout syntax.

\n

Support for page load is new in SeleniumLibrary 6.1

\n

Selenium speed

\n

Selenium execution speed can be slowed down globally by using Set Selenium speed keyword. This functionality is designed to be used for demonstrating or debugging purposes. Using it to make sure that elements appear on a page is not a good idea. The above-explained timeouts and waits should be used instead.

\n

See time format below for supported syntax.

\n

Time format

\n

All timeouts and waits can be given as numbers considered seconds (e.g. 0.5 or 42) or in Robot Framework's time syntax (e.g. 1.5 seconds or 1 min 30 s). For more information about the time syntax see the Robot Framework User Guide.

\n

Run-on-failure functionality

\n

SeleniumLibrary has a handy feature that it can automatically execute a keyword if any of its own keywords fails. By default, it uses the Capture Page Screenshot keyword, but this can be changed either by using the Register Keyword To Run On Failure keyword or with the run_on_failure argument when importing the library. It is possible to use any keyword from any imported library or resource file.

\n

The run-on-failure functionality can be disabled by using a special value NOTHING or anything considered false (see Boolean arguments) such as NONE.

\n

Boolean arguments

\n

Starting from 5.0 SeleniumLibrary relies on Robot Framework to perform the boolean conversion based on keyword arguments type hint. More details in Robot Framework user guide

\n

Please note SeleniumLibrary 3 and 4 did have own custom methods to covert arguments to boolean values.

\n

EventFiringWebDriver

\n

The SeleniumLibrary offers support for EventFiringWebDriver. See the Selenium and SeleniumLibrary EventFiringWebDriver support documentation for further details.

\n

EventFiringWebDriver is new in SeleniumLibrary 4.0

\n

Thread support

\n

SeleniumLibrary is not thread-safe. This is mainly due because the underlying Selenium tool is not thread-safe within one browser/driver instance. Because of the limitation in the Selenium side, the keywords or the API provided by the SeleniumLibrary is not thread-safe.

\n

Plugins

\n

SeleniumLibrary offers plugins as a way to modify and add library keywords and modify some of the internal functionality without creating a new library or hacking the source code. See plugin API documentation for further details.

\n

Plugin API is new SeleniumLibrary 4.0

\n

Auto closing browser

\n

By default, the browser instances created during a task execution are closed at the end of the task. This can be prevented with the auto_close parameter when importing the library.

\n

The value of the parameter needs to be set to False or any object evaluated as false (see Boolean arguments).

", "version": "6.1.3", - "generated": "2023-11-10 12:47:33", + "generated": "2023-11-10 12:56:00", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -11992,7 +11992,7 @@ "name": "RPA.Browser.Playwright", "doc": "

Browser library is a browser automation library for Robot Framework.

\n

This is the keyword documentation for Browser library. For information about installation, support, and more please visit the project pages. For more information about Robot Framework itself, see robotframework.org.

\n

Browser library uses Playwright Node module to automate Chromium, Firefox and WebKit with a single library.

\n

Table of contents

\n\n

Browser, Context and Page

\n

Browser library works with three different layers that build on each other: Browser, Context and Page.

\n

Browsers

\n

A browser can be started with one of the three different engines Chromium, Firefox or Webkit.

\n

Supported Browsers

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
BrowserBrowser with this engine
chromiumGoogle Chrome, Microsoft Edge (since 2020), Opera
firefoxMozilla Firefox
webkitApple Safari, Mail, AppStore on MacOS and iOS
\n

Since Playwright comes with a pack of builtin binaries for all browsers, no additional drivers e.g. geckodriver are needed.

\n

All these browsers that cover more than 85% of the world wide used browsers, can be tested on Windows, Linux and MacOS. There is no need for dedicated machines anymore.

\n

A browser process is started headless (without a GUI) by default. Run New Browser with specified arguments if a browser with a GUI is requested or if a proxy has to be configured. A browser process can contain several contexts.

\n

Contexts

\n

A context corresponds to a set of independent incognito pages in a browser that share cookies, sessions or profile settings. Pages in two separate contexts do not share cookies, sessions or profile settings. Compared to Selenium, these do not require their own browser process. To get a clean environment a test can just open a new context. Due to this new independent browser sessions can be opened with Robot Framework Browser about 10 times faster than with Selenium by just opening a New Context within the opened browser.

\n

To make pages in the same suite share state, use the same context by opening the context with New Context on suite setup.

\n

The context layer is useful e.g. for testing different user sessions on the same webpage without opening a whole new browser context. Contexts can also have detailed configurations, such as geo-location, language settings, the viewport size or color scheme. Contexts do also support http credentials to be set, so that basic authentication can also be tested. To be able to download files within the test, the acceptDownloads argument must be set to True in New Context keyword. A context can contain different pages.

\n

Pages

\n

A page does contain the content of the loaded web site and has a browsing history. Pages and browser tabs are the same.

\n

Typical usage could be:

\n
\n* Test Cases *\nStarting a browser with a page\n    New Browser    chromium    headless=false\n    New Context    viewport={'width': 1920, 'height': 1080}\n    New Page       https://marketsquare.github.io/robotframework-browser/Browser.html\n    Get Title      ==    Browser\n
\n

The Open Browser keyword opens a new browser, a new context and a new page. This keyword is useful for quick experiments or debugging sessions.

\n

When a New Page is called without an open browser, New Browser and New Context are executed with default values first.

\n

Each Browser, Context and Page has a unique ID with which they can be addressed. A full catalog of what is open can be received by Get Browser Catalog as a dictionary.

\n

Automatic page and context closing

\n

Controls when contexts and pages are closed during the test execution.

\n

If automatic closing level is TEST, contexts and pages that are created during a single test are automatically closed when the test ends. Contexts and pages that are created during suite setup are closed when the suite teardown ends.

\n

If automatic closing level is SUITE, all contexts and pages that are created during the test suite are closed when the suite teardown ends.

\n

If automatic closing level is MANUAL, nothing is closed automatically while the test execution is ongoing.

\n

All browsers are automatically closed, always and regardless of the automatic closing level at the end of the test execution. This will also close all remaining pages and contexts.

\n

Automatic closing can be configured or switched off with the auto_closing_level library import parameter.

\n

See: Importing

\n

Finding elements

\n

All keywords in the library that need to interact with an element on a web page take an argument typically named selector that specifies how to find the element. Keywords can find elements with strict mode. If strict mode is true and locator finds multiple elements from the page, keyword will fail. If keyword finds one element, keyword does not fail because of strict mode. If strict mode is false, keyword does not fail if selector points many elements. Strict mode is enabled by default, but can be changed in library importing or Set Strict Mode keyword. Keyword documentation states if keyword uses strict mode. If keyword does not state that strict mode is used, then strict mode is not applied for the keyword. For more details, see Playwright strict documentation.

\n

Selector strategies that are supported by default are listed in the table below.

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
StrategyMatch based onExample
cssCSS selector.css=.class > \\#login_btn
xpathXPath expression.xpath=//input[@id=\"login_btn\"]
textBrowser text engine.text=Login
idElement ID Attribute.id=login_btn
\n

CSS Selectors can also be recorded with Record selector keyword.

\n

Explicit Selector Strategy

\n

The explicit selector strategy is specified with a prefix using syntax strategy=value. Spaces around the separator are ignored, so css=foo, css= foo and css = foo are all equivalent.

\n

Implicit Selector Strategy

\n

The default selector strategy is css.

\n

If selector does not contain one of the know explicit selector strategies, it is assumed to contain css selector.

\n

Selectors that are starting with // or .. are considered as xpath selectors.

\n

Selectors that are in quotes are considered as text selectors.

\n

Examples:

\n
\n# CSS selectors are default.\nClick  span > button.some_class         # This is equivalent\nClick  css=span > button.some_class     # to this.\n\n# // or .. leads to xpath selector strategy\nClick  //span/button[@class=\"some_class\"]\nClick  xpath=//span/button[@class=\"some_class\"]\n\n# \"text\" in quotes leads to exact text selector strategy\nClick  \"Login\"\nClick  text=\"Login\"\n
\n

CSS

\n

As written before, the default selector strategy is css. See css selector for more information.

\n

Any malformed selector not starting with // or .. nor starting and ending with a quote is assumed to be a css selector.

\n

Note that # is a comment character in Robot Framework syntax and needs to be escaped like \\# to work as a css ID selector.

\n

Examples:

\n
\nClick  span > button.some_class\nGet Text  \\#username_field  ==  George\n
\n

XPath

\n

XPath engine is equivalent to Document.evaluate. Example: xpath=//html/body//span[text()=\"Hello World\"].

\n

Malformed selector starting with // or .. is assumed to be an xpath selector. For example, //html/body is converted to xpath=//html/body. More examples are displayed in Examples.

\n

Note that xpath does not pierce shadow_roots.

\n

Text

\n

Text engine finds an element that contains a text node with the passed text. For example, Click text=Login clicks on a login button, and Wait For Elements State text=\"lazy loaded text\" waits for the \"lazy loaded text\" to appear in the page.

\n

Text engine finds fields based on their labels in text inserting keywords.

\n

Malformed selector starting and ending with a quote (either \" or ') is assumed to be a text selector. For example, Click \"Login\" is converted to Click text=\"Login\". Be aware that these leads to exact matches only! More examples are displayed in Examples.

\n

Insensitive match

\n

By default, the match is case-insensitive, ignores leading/trailing whitespace and searches for a substring. This means text= Login matches <button>Button loGIN (click me)</button>.

\n

Exact match

\n

Text body can be escaped with single or double quotes for precise matching, insisting on exact match, including specified whitespace and case. This means text=\"Login \" will only match <button>Login </button> with exactly one space after \"Login\". Quoted text follows the usual escaping rules, e.g. use \\\" to escape double quote in a double-quoted string: text=\"foo\\\"bar\".

\n

RegEx

\n

Text body can also be a JavaScript-like regex wrapped in / symbols. This means text=/^hello .*!$/i or text=/^Hello .*!$/ will match <span>Hello Peter Parker!</span> with any name after Hello, ending with !. The first one flagged with i for case-insensitive. See https://regex101.com for more information about RegEx.

\n

Button and Submit Values

\n

Input elements of the type button and submit are rendered with their value as text, and text engine finds them. For example, text=Login matches <input type=button value=\"Login\">.

\n

Cascaded selector syntax

\n

Browser library supports the same selector strategies as the underlying Playwright node module: xpath, css, id and text. The strategy can either be explicitly specified with a prefix or the strategy can be implicit.

\n

A major advantage of Browser is that multiple selector engines can be used within one selector. It is possible to mix XPath, CSS and Text selectors while selecting a single element.

\n

Selectors are strings that consists of one or more clauses separated by >> token, e.g. clause1 >> clause2 >> clause3. When multiple clauses are present, next one is queried relative to the previous one's result. Browser library supports concatenation of different selectors separated by >>.

\n

For example:

\n
\nHighlight Elements    \"Hello\" >> ../.. >> .select_button\nHighlight Elements    text=Hello >> xpath=../.. >> css=.select_button\n
\n

Each clause contains a selector engine name and selector body, e.g. engine=body. Here engine is one of the supported engines (e.g. css or a custom one). Selector body follows the format of the particular engine, e.g. for css engine it should be a css selector. Body format is assumed to ignore leading and trailing white spaces, so that extra whitespace can be added for readability. If the selector engine needs to include >> in the body, it should be escaped inside a string to not be confused with clause separator, e.g. text=\"some >> text\".

\n

Selector engine name can be prefixed with * to capture an element that matches the particular clause instead of the last one. For example, css=article >> text=Hello captures the element with the text Hello, and *css=article >> text=Hello (note the *) captures the article element that contains some element with the text Hello.

\n

For convenience, selectors in the wrong format are heuristically converted to the right format. See Implicit Selector Strategy

\n

Examples

\n
\n# queries 'div' css selector\nGet Element    css=div\n\n# queries '//html/body/div' xpath selector\nGet Element    //html/body/div\n\n# queries '\"foo\"' text selector\nGet Element    text=foo\n\n# queries 'span' css selector inside the result of '//html/body/div' xpath selector\nGet Element    xpath=//html/body/div >> css=span\n\n# converted to 'css=div'\nGet Element    div\n\n# converted to 'xpath=//html/body/div'\nGet Element    //html/body/div\n\n# converted to 'text=\"foo\"'\nGet Element    \"foo\"\n\n# queries the div element of every 2nd span element inside an element with the id foo\nGet Element    \\#foo >> css=span:nth-child(2n+1) >> div\nGet Element    id=foo >> css=span:nth-child(2n+1) >> div\n
\n

Be aware that using # as a starting character in Robot Framework would be interpreted as comment. Due to that fact a #id must be escaped as \\#id.

\n

Frames

\n

By default, selector chains do not cross frame boundaries. It means that a simple CSS selector is not able to select an element located inside an iframe or a frameset. For this use case, there is a special selector >>> which can be used to combine a selector for the frame and a selector for an element inside a frame.

\n

Given this simple pseudo html snippet:

\n
\n<iframe id=\"iframe\" src=\"src.html\">\n  #document\n    <!DOCTYPE html>\n    <html>\n      <head></head>\n      <body>\n        <button id=\"btn\">Click Me</button>\n      </body>\n    </html>\n</iframe>\n
\n

Here's a keyword call that clicks the button inside the frame.

\n
\nClick   id=iframe >>> id=btn\n
\n

The selectors on the left and right side of >>> can be any valid selectors. The selector clause directly before the frame opener >>> must select the frame element. Frame selection is the only place where Browser Library modifies the selector, as explained in above. In all cases, the library does not alter the selector in any way, instead it is passed as is to the Playwright side.

\n

WebComponents and Shadow DOM

\n

Playwright and so also Browser are able to do automatic piercing of Shadow DOMs and therefore are the best automation technology when working with WebComponents.

\n

Also other technologies claim that they can handle Shadow DOM and Web Components. However, none of them do pierce shadow roots automatically, which may be inconvenient when working with Shadow DOM and Web Components.

\n

For that reason, the css engine pierces shadow roots. More specifically, every Descendant combinator pierces an arbitrary number of open shadow roots, including the implicit descendant combinator at the start of the selector.

\n

That means, it is not necessary to select each shadow host, open its shadow root and select the next shadow host until you reach the element that should be controlled.

\n

CSS:light

\n

css:light engine is equivalent to Document.querySelector and behaves according to the CSS spec. However, it does not pierce shadow roots.

\n

css engine first searches for elements in the light dom in the iteration order, and then recursively inside open shadow roots in the iteration order. It does not search inside closed shadow roots or iframes.

\n

Examples:

\n
\n<article>\n  <div>In the light dom</div>\n  <div slot='myslot'>In the light dom, but goes into the shadow slot</div>\n  <open mode shadow root>\n      <div class='in-the-shadow'>\n          <span class='content'>\n              In the shadow dom\n              <open mode shadow root>\n                  <li id='target'>Deep in the shadow</li>\n              </open mode shadow root>\n          </span>\n      </div>\n      <slot name='myslot'></slot>\n  </open mode shadow root>\n</article>\n
\n

Note that <open mode shadow root> is not an html element, but rather a shadow root created with element.attachShadow({mode: 'open'}).

\n
    \n
  • Both \"css=article div\" and \"css:light=article div\" match the first <div>In the light dom</div>.
  • \n
  • Both \"css=article > div\" and \"css:light=article > div\" match two div elements that are direct children of the article.
  • \n
  • \"css=article .in-the-shadow\" matches the <div class='in-the-shadow'>, piercing the shadow root, while \"css:light=article .in-the-shadow\" does not match anything.
  • \n
  • \"css:light=article div > span\" does not match anything, because both light-dom div elements do not contain a span.
  • \n
  • \"css=article div > span\" matches the <span class='content'>, piercing the shadow root.
  • \n
  • \"css=article > .in-the-shadow\" does not match anything, because <div class='in-the-shadow'> is not a direct child of article
  • \n
  • \"css:light=article > .in-the-shadow\" does not match anything.
  • \n
  • \"css=article li#target\" matches the <li id='target'>Deep in the shadow</li>, piercing two shadow roots.
  • \n
\n

text:light

\n

text engine open pierces shadow roots similarly to css, while text:light does not. Text engine first searches for elements in the light dom in the iteration order, and then recursively inside open shadow roots in the iteration order. It does not search inside closed shadow roots or iframes.

\n

id, data-testid, data-test-id, data-test and their :light counterparts

\n

Attribute engines are selecting based on the corresponding attribute value. For example: data-test-id=foo is equivalent to css=[data-test-id=\"foo\"], and id:light=foo is equivalent to css:light=[id=\"foo\"].

\n

Element reference syntax

\n

It is possible to get a reference to a Locator by using Get Element and Get Elements keywords. Keywords do not save reference to an element in the HTML document, instead it saves reference to a Playwright Locator. In nutshell Locator captures the logic of how to retrieve that element from the page. Each time an action is performed, the locator re-searches the elements in the page. This reference can be used as a first part of a selector by using a special selector syntax element=. like this:

\n
\n${ref}=    Get Element    .some_class\n           Click          ${ref} >> .some_child     # Locator searches an element from the page.\n           Click          ${ref} >> .other_child    # Locator searches again an element from the page.\n
\n

The .some_child and .other_child selectors in the example are relative to the element referenced by ${ref}. Please note that frame piercing is not possible with element reference.

\n

Assertions

\n

Keywords that accept arguments assertion_operator <AssertionOperator> and assertion_expected can optionally assert that a specified condition holds. Keywords will return the value even when the assertion is performed by the keyword.

\n

Assert will retry and fail only after a specified timeout. See Importing and retry_assertions_for (default is 1 second) for configuring this timeout.

\n

Currently supported assertion operators are:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
OperatorAlternative OperatorsDescriptionValidate Equivalent
==equal, equals, should beChecks if returned value is equal to expected value.value == expected
!=inequal, should not beChecks if returned value is not equal to expected value.value != expected
>greater thanChecks if returned value is greater than expected value.value > expected
>=Checks if returned value is greater than or equal to expected value.value >= expected
<less thanChecks if returned value is less than expected value.value < expected
<=Checks if returned value is less than or equal to expected value.value <= expected
*=containsChecks if returned value contains expected value as substring.expected in value
not containsChecks if returned value does not contain expected value as substring.expected in value
^=should start with, startsChecks if returned value starts with expected value.re.search(f\"^{expected}\", value)
$=should end with, endsChecks if returned value ends with expected value.re.search(f\"{expected}$\", value)
matchesChecks if given RegEx matches minimum once in returned value.re.search(expected, value)
validateChecks if given Python expression evaluates to True.
evaluatethenWhen using this operator, the keyword does return the evaluated Python expression.
\n

Currently supported formatters for assertions are:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
FormatterDescription
normalize spacesSubstitutes multiple spaces to single space from the value
stripRemoves spaces from the beginning and end of the value
case insensitiveConverts value to lower case before comparing
apply to expectedApplies rules also for the expected value
\n

Formatters are applied to the value before assertion is performed and keywords returns a value where rule is applied. Formatter is only applied to the value which keyword returns and not all rules are valid for all assertion operators. If apply to expected formatter is defined, then formatters are then formatter are also applied to expected value.

\n

By default, keywords will provide an error message if an assertion fails. Default error messages can be overwritten with a message argument. The message argument accepts {value}, {value_type}, {expected} and {expected_type} format options. The {value} is the value returned by the keyword and the {expected} is the expected value defined by the user, usually the value in the assertion_expected argument. The {value_type} and {expected_type} are the type definitions from {value} and {expected} arguments. In similar fashion as Python type returns type definition. Assertions will retry until timeout has expired if they do not pass.

\n

The assertion assertion_expected value is not converted by the library and is used as is. Therefore when assertion is made, the assertion_expected argument value and value returned the keyword must have the same type. If types are not the same, assertion will fail. Example Get Text always returns a string and has to be compared with a string, even the returned value might look like a number.

\n

Other Keywords have other specific types they return. Get Element Count always returns an integer. Get Bounding Box and Get Viewport Size can be filtered. They return a dictionary without a filter and a number when filtered. These Keywords do automatic conversion for the expected value if a number is returned.

\n

* < less or greater > With Strings* Comparisons of strings with greater than or less than compares each character, starting from 0 regarding where it stands in the code page. Example: A < Z, Z < a, ac < dc It does never compare the length of elements. Neither lists nor strings. The comparison stops at the first character that is different. Examples: `'abcde' < 'abd', '100.000' < '2' In Python 3 and therefore also in Browser it is not possible to compare numbers with strings with a greater or less operator. On keywords that return numbers, the given expected value is automatically converted to a number before comparison.

\n

The getters Get Page State and Get Browser Catalog return a dictionary. Values of the dictionary can directly asserted. Pay attention of possible types because they are evaluated in Python. For example:

\n
\nGet Page State    validate    2020 >= value['year']                     # Comparison of numbers\nGet Page State    validate    \"IMPORTANT MESSAGE!\" == value['message']  # Comparison of strings\n
\n

The 'then' or 'evaluate' closure

\n

Keywords that accept arguments assertion_operator and assertion_expected can optionally also use then or evaluate closure to modify the returned value with BuiltIn Evaluate. Actual value can be accessed with value.

\n

For example Get Title then 'TITLE: '+value. See Builtin Evaluating expressions for more info on the syntax.

\n

Examples

\n
\n# Keyword    Selector                    Key        Assertion Operator    Assertion Expected\nGet Title                                           equal                 Page Title\nGet Title                                           ^=                    Page\nGet Style    //*[@id=\"div-element\"]      width      >                     100\nGet Title                                           matches               \\\\w+\\\\s\\\\w+\nGet Title                                           validate              value == \"Login Page\"\nGet Title                                           evaluate              value if value == \"some value\" else \"something else\"\n
\n

Implicit waiting

\n

Browser library and Playwright have many mechanisms to help in waiting for elements. Playwright will auto-wait before performing actions on elements. Please see Auto-waiting on Playwright documentation for more information.

\n

On top of Playwright auto-waiting Browser assertions will wait and retry for specified time before failing any Assertions. Time is specified in Browser library initialization with retry_assertions_for.

\n

Browser library also includes explicit waiting keywords such as Wait for Elements State if more control for waiting is needed.

\n

Experimental: Re-using same node process

\n

Browser library integrated nodejs and python. The NodeJS side can be also executed as a standalone process. Browser libraries running on the same machine can talk to that instead of starting new node processes. This can speed execution when running tests parallel. To start node side run on the directory when the Browser package is PLAYWRIGHT_BROWSERS_PATH=0 node Browser/wrapper/index.js PORT.

\n

PORT is the port you want to use for the node process. To execute tests then with pabot for example do ROBOT_FRAMEWORK_BROWSER_NODE_PORT=PORT pabot ...

\n

Experimental: Provide parameters to node process

\n

Browser library is integrated with NodeJSand and Python. Browser library starts a node process, to communicate Playwright API in NodeJS side. It is possible to provide parameters for the started node process by defining ROBOT_FRAMEWORK_BROWSER_NODE_DEBUG_OPTIONS environment variable, before starting the test execution. Example: ROBOT_FRAMEWORK_BROWSER_NODE_DEBUG_OPTIONS=--inspect;robot path/to/tests. There can be multiple arguments defined in the environment variable and arguments must be separated with comma.

\n

Scope Setting

\n

Some keywords which manipulates library settings have a scope argument. With that scope argument one can set the \"live time\" of that setting. Available Scopes are: Global, Suite and Test/Task See Scope. Is a scope finished, this scoped setting, like timeout, will no longer be used.

\n

Live Times:

\n
    \n
  • A Global scope will live forever until it is overwritten by another Global scope. Or locally temporarily overridden by a more narrow scope.
  • \n
  • A Suite scope will locally override the Global scope and live until the end of the Suite within it is set, or if it is overwritten by a later setting with Global or same scope. Children suite does inherit the setting from the parent suite but also may have its own local Suite setting that then will be inherited to its children suites.
  • \n
  • A Test or Task scope will be inherited from its parent suite but when set, lives until the end of that particular test or task.
  • \n
\n

A new set higher order scope will always remove the lower order scope which may be in charge. So the setting of a Suite scope from a test, will set that scope to the robot file suite where that test is and removes the Test scope that may have been in place.

\n

Extending Browser library with a JavaScript module

\n

Browser library can be extended with JavaScript. The module must be in CommonJS format that Node.js uses. You can translate your ES6 module to Node.js CommonJS style with Babel. Many other languages can be also translated to modules that can be used from Node.js. For example TypeScript, PureScript and ClojureScript just to mention few.

\n
\nasync function myGoToKeyword(url, args, page, logger, playwright) {\n  logger(args.toString())\n  playwright.coolNewFeature()\n  return await page.goto(url);\n}\n
\n

Functions can contain any number of arguments and arguments may have default values.

\n

There are some reserved arguments that are not accessible from Robot Framework side. They are injected to the function if they are in the arguments:

\n

page: the playwright Page object.

\n

args: the rest of values from Robot Framework keyword call *args.

\n

logger: callback function that takes strings as arguments and writes them to robot log. Can be called multiple times.

\n

playwright: playwright module (* from 'playwright'). Useful for integrating with Playwright features that Browser library doesn't support with it's own keywords. API docs

\n

also argument name self can not be used.

\n

Example module.js

\n
\nasync function myGoToKeyword(pageUrl, page) {\n  await page.goto(pageUrl);\n  return await page.title();\n}\nexports.__esModule = true;\nexports.myGoToKeyword = myGoToKeyword;\n
\n

Example Robot Framework side

\n
\n* Settings *\nLibrary   Browser  jsextension=${CURDIR}/module.js\n\n* Test Cases *\nHello\n  New Page\n  ${title}=  myGoToKeyword  https://playwright.dev\n  Should be equal  ${title}  Playwright\n
\n

Also selector syntax can be extended with a custom selector using a js module

\n

Example module keyword for custom selector registering

\n
\nasync function registerMySelector(playwright) {\nplaywright.selectors.register(\"myselector\", () => ({\n   // Returns the first element matching given selector in the root's subtree.\n   query(root, selector) {\n      return root.querySelector(a[data-title=\"${selector}\"]);\n    },\n\n    // Returns all elements matching given selector in the root's subtree.\n    queryAll(root, selector) {\n      return Array.from(root.querySelectorAll(a[data-title=\"${selector}\"]));\n    }\n}));\nreturn 1;\n}\nexports.__esModule = true;\nexports.registerMySelector = registerMySelector;\n
\n

Plugins

\n

Browser library offers plugins as a way to modify and add library keywords and modify some of the internal functionality without creating a new library or hacking the source code. See plugin API documentation for further details.

\n

Automatic headless detection is supported when opening a new browser.

", "version": "17.5.2", - "generated": "2023-11-10 12:47:34", + "generated": "2023-11-10 12:56:00", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -24054,7 +24054,7 @@ "name": "RPA.Browser.Selenium", "doc": "

SeleniumLibrary is a web testing library for Robot Framework.

\n

This document explains how to use keywords provided by SeleniumLibrary. For information about installation, support, and more, please visit the project pages. For more information about Robot Framework, see http://robotframework.org.

\n

SeleniumLibrary uses the Selenium WebDriver modules internally to control a web browser. See http://seleniumhq.org for more information about Selenium in general and SeleniumLibrary README.rst Browser drivers chapter for more details about WebDriver binary installation.

\n\n

Locating elements

\n

All keywords in SeleniumLibrary that need to interact with an element on a web page take an argument typically named locator that specifies how to find the element. Most often the locator is given as a string using the locator syntax described below, but using WebElements is possible too.

\n

Locator syntax

\n

SeleniumLibrary supports finding elements based on different strategies such as the element id, XPath expressions, or CSS selectors. The strategy can either be explicitly specified with a prefix or the strategy can be implicit.

\n

Default locator strategy

\n

By default, locators are considered to use the keyword specific default locator strategy. All keywords support finding elements based on id and name attributes, but some keywords support additional attributes or other values that make sense in their context. For example, Click Link supports the href attribute and the link text and addition to the normal id and name.

\n

Examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Click Elementexample# Match based on id or name.
Click Linkexample# Match also based on link text and href.
Click Buttonexample# Match based on id, name or value.
\n

If a locator accidentally starts with a prefix recognized as explicit locator strategy or implicit XPath strategy, it is possible to use the explicit default prefix to enable the default strategy.

\n

Examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Click Elementname:foo# Find element with name foo.
Click Elementdefault:name:foo# Use default strategy with value name:foo.
Click Element//foo# Find element using XPath //foo.
Click Elementdefault: //foo# Use default strategy with value //foo.
\n

Explicit locator strategy

\n

The explicit locator strategy is specified with a prefix using either syntax strategy:value or strategy=value. The former syntax is preferred because the latter is identical to Robot Framework's named argument syntax and that can cause problems. Spaces around the separator are ignored, so id:foo, id: foo and id : foo are all equivalent.

\n

Locator strategies that are supported by default are listed in the table below. In addition to them, it is possible to register custom locators.

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
StrategyMatch based onExample
idElement id.id:example
namename attribute.name:example
identifierEither id or name.identifier:example
classElement class.class:example
tagTag name.tag:div
xpathXPath expression.xpath://div[@id=\"example\"]
cssCSS selector.css:div#example
domDOM expression.dom:document.images[5]
linkExact text a link has.link:The example
partial linkPartial link text.partial link:he ex
sizzleSizzle selector deprecated.sizzle:div.example
dataElement data-* attributedata:id:my_id
jqueryjQuery expression.jquery:div.example
defaultKeyword specific default behavior.default:example
\n

See the Default locator strategy section below for more information about how the default strategy works. Using the explicit default prefix is only necessary if the locator value itself accidentally matches some of the explicit strategies.

\n

Different locator strategies have different pros and cons. Using ids, either explicitly like id:foo or by using the default locator strategy simply like foo, is recommended when possible, because the syntax is simple and locating elements by id is fast for browsers. If an element does not have an id or the id is not stable, other solutions need to be used. If an element has a unique tag name or class, using tag, class or css strategy like tag:h1, class:example or css:h1.example is often an easy solution. In more complex cases using XPath expressions is typically the best approach. They are very powerful but a downside is that they can also get complex.

\n

Examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Click Elementid:foo# Element with id 'foo'.
Click Elementcss:div#foo h1# h1 element under div with id 'foo'.
Click Elementxpath: //div[@id=\"foo\"]//h1# Same as the above using XPath, not CSS.
Click Elementxpath: //*[contains(text(), \"example\")]# Element containing text 'example'.
\n

NOTE:

\n
    \n
  • The strategy:value syntax is only supported by SeleniumLibrary 3.0 and newer.
  • \n
  • Using the sizzle strategy or its alias jquery requires that the system under test contains the jQuery library.
  • \n
  • Prior to SeleniumLibrary 3.0, table related keywords only supported xpath, css and sizzle/jquery strategies.
  • \n
  • data strategy is conveniance locator that will construct xpath from the parameters. If you have element like <div data-automation=\"automation-id-2\">, you locate the element via data:automation:automation-id-2. This feature was added in SeleniumLibrary 5.2.0
  • \n
\n

Implicit XPath strategy

\n

If the locator starts with // or multiple opening parenthesis in front of the //, the locator is considered to be an XPath expression. In other words, using //div is equivalent to using explicit xpath://div and ((//div)) is equivalent to using explicit xpath:((//div))

\n

Examples:

\n\n\n\n\n\n\n\n\n\n
Click Element//div[@id=\"foo\"]//h1
Click Element(//div)[2]
\n

The support for the (// prefix is new in SeleniumLibrary 3.0. Supporting multiple opening parenthesis is new in SeleniumLibrary 5.0.

\n

Chaining locators

\n

It is possible chain multiple locators together as single locator. Each chained locator must start with locator strategy. Chained locators must be separated with single space, two greater than characters and followed with space. It is also possible mix different locator strategies, example css or xpath. Also a list can also be used to specify multiple locators. This is useful, is some part of locator would match as the locator separator but it should not. Or if there is need to existing WebElement as locator.

\n

Although all locators support chaining, some locator strategies do not abey the chaining. This is because some locator strategies use JavaScript to find elements and JavaScript is executed for the whole browser context and not for the element found be the previous locator. Chaining is supported by locator strategies which are based on Selenium API, like xpath or css, but example chaining is not supported by sizzle or `jquery

\n

Examples:

\n\n\n\n\n\n\n
Click Elementcss:.bar >> xpath://a# To find a link which is present after an element with class \"bar\"
\n

List examples:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
${locator_list} =Create Listcss:div#div_idxpath://*[text(), \" >> \"]
Page Should Contain Element${locator_list}
${element} =Get WebElementxpath://*[text(), \" >> \"]
${locator_list} =Create Listcss:div#div_id${element}
Page Should Contain Element${locator_list}
\n

Chaining locators in new in SeleniumLibrary 5.0

\n

Using WebElements

\n

In addition to specifying a locator as a string, it is possible to use Selenium's WebElement objects. This requires first getting a WebElement, for example, by using the Get WebElement keyword.

\n\n\n\n\n\n\n\n\n\n\n\n
${elem} =Get WebElementid:example
Click Element${elem}
\n

Custom locators

\n

If more complex lookups are required than what is provided through the default locators, custom lookup strategies can be created. Using custom locators is a two part process. First, create a keyword that returns a WebElement that should be acted on:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Custom Locator Strategy[Arguments]${browser}${locator}${tag}${constraints}
${element}=Execute Javascriptreturn window.document.getElementById('${locator}');
[Return]${element}
\n

This keyword is a reimplementation of the basic functionality of the id locator where ${browser} is a reference to a WebDriver instance and ${locator} is the name of the locator strategy. To use this locator, it must first be registered by using the Add Location Strategy keyword:

\n\n\n\n\n\n\n
Add Location StrategycustomCustom Locator Strategy
\n

The first argument of Add Location Strategy specifies the name of the strategy and it must be unique. After registering the strategy, the usage is the same as with other locators:

\n\n\n\n\n\n
Click Elementcustom:example
\n

See the Add Location Strategy keyword for more details.

\n

Browser and Window

\n

There is different conceptual meaning when SeleniumLibrary talks about windows or browsers. This chapter explains those differences.

\n

Browser

\n

When Open Browser or Create WebDriver keyword is called, it will create a new Selenium WebDriver instance by using the Selenium WebDriver API. In SeleniumLibrary terms, a new browser is created. It is possible to start multiple independent browsers (Selenium Webdriver instances) at the same time, by calling Open Browser or Create WebDriver multiple times. These browsers are usually independent of each other and do not share data like cookies, sessions or profiles. Typically when the browser starts, it creates a single window which is shown to the user.

\n

Window

\n

Windows are the part of a browser that loads the web site and presents it to the user. All content of the site is the content of the window. Windows are children of a browser. In SeleniumLibrary browser is a synonym for WebDriver instance. One browser may have multiple windows. Windows can appear as tabs, as separate windows or pop-ups with different position and size. Windows belonging to the same browser typically share the sessions detail, like cookies. If there is a need to separate sessions detail, example login with two different users, two browsers (Selenium WebDriver instances) must be created. New windows can be opened example by the application under test or by example Execute Javascript keyword:

\n
\nExecute Javascript    window.open()    # Opens a new window with location about:blank\n
\n

The example below opens multiple browsers and windows, to demonstrate how the different keywords can be used to interact with browsers, and windows attached to these browsers.

\n

Structure:

\n
\nBrowserA\n           Window 1  (location=https://robotframework.org/)\n           Window 2  (location=https://robocon.io/)\n           Window 3  (location=https://github.com/robotframework/)\n\nBrowserB\n           Window 1  (location=https://github.com/)\n
\n

Example:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Open Browserhttps://robotframework.org${BROWSER}alias=BrowserA# BrowserA with first window is opened.
Execute Javascriptwindow.open()# In BrowserA second window is opened.
Switch Windowlocator=NEW# Switched to second window in BrowserA
Go Tohttps://robocon.io# Second window navigates to robocon site.
Execute Javascriptwindow.open()# In BrowserA third window is opened.
${handle}Switch Windowlocator=NEW# Switched to third window in BrowserA
Go Tohttps://github.com/robotframework/# Third windows goes to robot framework github site.
Open Browserhttps://github.com${BROWSER}alias=BrowserB# BrowserB with first windows is opened.
${location}Get Location# ${location} is: https://www.github.com
Switch Window${handle}browser=BrowserA# BrowserA second windows is selected.
${location}Get Location# ${location} = https://robocon.io/
@{locations 1}Get Locations# By default, lists locations under the currectly active browser (BrowserA).
@{locations 2}Get Locationsbrowser=ALL# By using browser=ALL argument keyword list all locations from all browsers.
\n

The above example, @{locations 1} contains the following items: https://robotframework.org/, https://robocon.io/ and https://github.com/robotframework/'. The @{locations 2} contains the following items: https://robotframework.org/, https://robocon.io/, https://github.com/robotframework/' and 'https://github.com/.

\n

Timeouts, waits, and delays

\n

This section discusses different ways how to wait for elements to appear on web pages and to slow down execution speed otherwise. It also explains the time format that can be used when setting various timeouts, waits, and delays.

\n

Timeout

\n

SeleniumLibrary contains various keywords that have an optional timeout argument that specifies how long these keywords should wait for certain events or actions. These keywords include, for example, Wait ... keywords and keywords related to alerts. Additionally Execute Async Javascript. Although it does not have timeout, argument, uses a timeout to define how long asynchronous JavaScript can run.

\n

The default timeout these keywords use can be set globally either by using the Set Selenium Timeout keyword or with the timeout argument when importing the library. If no default timeout is set globally, the default is 5 seconds. If None is specified for the timeout argument in the keywords, the default is used. See time format below for supported timeout syntax.

\n

Implicit wait

\n

Implicit wait specifies the maximum time how long Selenium waits when searching for elements. It can be set by using the Set Selenium Implicit Wait keyword or with the implicit_wait argument when importing the library. See Selenium documentation for more information about this functionality.

\n

See time format below for supported syntax.

\n

Page load

\n

Page load timeout is the amount of time to wait for page load to complete until error is raised.

\n

The default page load timeout can be set globally when importing the library with the page_load_timeout argument or by using the Set Selenium Page Load Timeout keyword.

\n

See time format below for supported timeout syntax.

\n

Support for page load is new in SeleniumLibrary 6.1

\n

Selenium speed

\n

Selenium execution speed can be slowed down globally by using Set Selenium speed keyword. This functionality is designed to be used for demonstrating or debugging purposes. Using it to make sure that elements appear on a page is not a good idea. The above-explained timeouts and waits should be used instead.

\n

See time format below for supported syntax.

\n

Time format

\n

All timeouts and waits can be given as numbers considered seconds (e.g. 0.5 or 42) or in Robot Framework's time syntax (e.g. 1.5 seconds or 1 min 30 s). For more information about the time syntax see the Robot Framework User Guide.

\n

Run-on-failure functionality

\n

SeleniumLibrary has a handy feature that it can automatically execute a keyword if any of its own keywords fails. By default, it uses the Capture Page Screenshot keyword, but this can be changed either by using the Register Keyword To Run On Failure keyword or with the run_on_failure argument when importing the library. It is possible to use any keyword from any imported library or resource file.

\n

The run-on-failure functionality can be disabled by using a special value NOTHING or anything considered false (see Boolean arguments) such as NONE.

\n

Boolean arguments

\n

Starting from 5.0 SeleniumLibrary relies on Robot Framework to perform the boolean conversion based on keyword arguments type hint. More details in Robot Framework user guide

\n

Please note SeleniumLibrary 3 and 4 did have own custom methods to covert arguments to boolean values.

\n

EventFiringWebDriver

\n

The SeleniumLibrary offers support for EventFiringWebDriver. See the Selenium and SeleniumLibrary EventFiringWebDriver support documentation for further details.

\n

EventFiringWebDriver is new in SeleniumLibrary 4.0

\n

Thread support

\n

SeleniumLibrary is not thread-safe. This is mainly due because the underlying Selenium tool is not thread-safe within one browser/driver instance. Because of the limitation in the Selenium side, the keywords or the API provided by the SeleniumLibrary is not thread-safe.

\n

Plugins

\n

SeleniumLibrary offers plugins as a way to modify and add library keywords and modify some of the internal functionality without creating a new library or hacking the source code. See plugin API documentation for further details.

\n

Plugin API is new SeleniumLibrary 4.0

\n

Auto closing browser

\n

By default, the browser instances created during a task execution are closed at the end of the task. This can be prevented with the auto_close parameter when importing the library.

\n

The value of the parameter needs to be set to False or any object evaluated as false (see Boolean arguments).

", "version": "6.1.3", - "generated": "2023-11-10 12:47:34", + "generated": "2023-11-10 12:56:01", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -33375,7 +33375,7 @@ "name": "RPA.Browser.common", "doc": "

Documentation for library RPA.Browser.common.

", "version": "", - "generated": "2023-11-10 12:47:34", + "generated": "2023-11-10 12:56:01", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -33466,7 +33466,7 @@ "name": "RPA.Calendar", "doc": "
\n

Library for handling different operations for date and time\nhandling especially in business days and holiday contexts.

\n

Utilizing pendulum and\nholidays packages.

\n

Library is by default using days from Monday to Friday as business\ndays, but that can be changed by giving list of weekdays to\nSet Business Days keyword. A weekday is given as a integer, the\n0 for Sunday and 6 for Saturday.

\n

Common country holidays are respected when getting next and previous\nbusiness days, but custom holidays can be added into consideration\nusing keyword Add Custom Holidays keyword.

\n

Some dates containing for example month names are in English (en), but\nthe locale of the library can be changed with keyword Set Locale or\nfor specific keyword if that has a locale parameter.

\n
\n", "version": "", - "generated": "2023-11-10 12:47:35", + "generated": "2023-11-10 12:56:01", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -34736,7 +34736,7 @@ "name": "RPA.Cloud.AWS", "doc": "
\n

AWS is a library for operating with Amazon AWS services S3, SQS,\nTextract and Comprehend.

\n

Services are initialized with keywords like Init S3 Client for S3.

\n

AWS authentication

\n

Authentication for AWS is set with key id and access key which can be given to the library\nin three different ways.

\n
    \n
  • Method 1 as environment variables, AWS_KEY_ID and AWS_KEY.
  • \n
  • Method 2 as keyword parameters to Init Textract Client for example.
  • \n
  • Method 3 as Robocorp vault secret. The vault name needs to be given in library init or\nwith keyword Set Robocorp Vault. Secret keys are expected to match environment variable\nnames.
  • \n
\n

Note. Starting from rpaframework-aws 1.0.3 region can be given as environment\nvariable AWS_REGION or include as Robocorp Vault secret with the same key name.

\n

Redshift Data authentication: Depending on the authorization method, use\none of the following combinations of request parameters, which can only\nbe passed via method 2:

\n
\n
    \n
  • Secrets Manager - when connecting to a cluster, specify the Amazon\nResource Name (ARN) of the secret, the database name, and the\ncluster identifier that matches the cluster in the secret. When\nconnecting to a serverless endpoint, specify the Amazon Resource\nName (ARN) of the secret and the database name.
  • \n
  • Temporary credentials - when connecting to a cluster, specify the\ncluster identifier, the database name, and the database user name.\nAlso, permission to call the redshift:GetClusterCredentials\noperation is required. When connecting to a serverless endpoint,\nspecify the database name.
  • \n
\n
\n

Role Assumption: With the use of the STS service client, you are able\nto assume another role, which will return temporary credentials. The\ntemporary credentials will include an access key and session token, see\nkeyword documentation for Assume Role for details of how the\ncredentials are returned. You can use these temporary credentials\nas part of method 2, but you must also include the session token.

\n

Method 1. credentials using environment variable

\n
\n*** Settings ***\nLibrary   RPA.Cloud.AWS\n\n*** Tasks ***\nInit AWS services\n    # NO parameters for client, expecting to get credentials\n    # with AWS_KEY, AWS_KEY_ID and AWS_REGION environment variables\n    Init S3 Client\n
\n

Method 2. credentials with keyword parameter

\n
\n*** Settings ***\nLibrary   RPA.Cloud.AWS   region=us-east-1\n\n*** Tasks ***\nInit AWS services\n    Init S3 Client  aws_key_id=${AWS_KEY_ID}  aws_key=${AWS_KEY}\n
\n

Method 3. setting Robocorp Vault in the library init

\n
\n*** Settings ***\nLibrary   RPA.Cloud.AWS  robocorp_vault_name=aws\n\n*** Tasks ***\nInit AWS services\n    Init S3 Client  use_robocorp_vault=${TRUE}\n
\n

Method 3. setting Robocorp Vault with keyword

\n
\n*** Settings ***\nLibrary   RPA.Cloud.AWS\n\n*** Tasks ***\nInit AWS services\n    Set Robocorp Vault     vault_name=aws\n    Init Textract Client    use_robocorp_vault=${TRUE}\n
\n

Requirements

\n

The default installation depends on boto3 library. Due to the size of the\ndependency, this library is available separate package rpaframework-aws but can\nalso be installed as an optional package for rpaframework.

\n

Recommended installation is rpaframework-aws plus rpaframework package.\nRemember to check latest versions from rpaframework Github repository.

\n
\nchannels:\n  - conda-forge\ndependencies:\n  - python=3.7.5\n  - pip=20.1\n  - pip:\n    - rpaframework==13.0.2\n    - rpaframework-aws==1.0.3\n
\n

Example

\n
\n*** Settings ***\nLibrary   RPA.Cloud.AWS   region=us-east-1\n\n*** Variables ***\n${BUCKET_NAME}        testbucket12213123123\n\n*** Tasks ***\nUpload a file into S3 bucket\n    [Setup]   Init S3 Client\n    Upload File      ${BUCKET_NAME}   ${/}path${/}to${/}file.pdf\n    @{files}         List Files   ${BUCKET_NAME}\n    FOR   ${file}  IN   @{files}\n        Log  ${file}\n    END\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:36", + "generated": "2023-11-10 12:56:02", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -37187,7 +37187,7 @@ "name": "RPA.Cloud.Azure", "doc": "
\n

Azure is a library for operating with Microsoft Azure API endpoints.

\n

List of supported service names:

\n\n

Azure authentication

\n

Authentication for Azure is set with service subscription key which can be given to the library\nin two different ways.

\n
    \n
  • Method 1 as environment variables, either service specific environment variable\nfor example AZURE_TEXTANALYTICS_KEY or with common key AZURE_SUBSCRIPTION_KEY which\nwill be used for all the services.
  • \n
  • Method 2 as Robocorp Vault secret. The vault name needs to be given in library init or\nwith keyword Set Robocorp Vault. Secret keys are expected to match environment variable\nnames.
  • \n
\n

Method 1. subscription key using environment variable

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Azure\n\n*** Tasks ***\nInit Azure services\n    # NO parameters for client, expecting to get subscription key\n    # with AZURE_TEXTANALYTICS_KEY or AZURE_SUBSCRIPTION_KEY environment variable\n    Init Text Analytics Service\n
\n

Method 2. setting Robocorp Vault in the library init

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Azure  robocorp_vault_name=azure\n\n*** Tasks ***\nInit Azure services\n    Init Text Analytics Service  use_robocorp_vault=${TRUE}\n
\n

Method 2. setting Robocorp Vault with keyword

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Azure\n\n*** Tasks ***\nInit Azure services\n    Set Robocorp Vault          vault_name=googlecloud\n    Init Text Analytics Service  use_robocorp_vault=${TRUE}\n
\n

References

\n

List of supported language locales - Azure locale list

\n

List of supported region identifiers - Azure region list

\n

Examples

\n

Robot Framework

\n

This is a section which describes how to use the library in your\nRobot Framework tasks.

\n
\n*** Settings ***\nLibrary  RPA.Cloud.Azure\n\n*** Variables ***\n${IMAGE_URL}   IMAGE_URL\n${FEATURES}    Faces,ImageType\n\n*** Tasks ***\nVisioning image information\n   Init Computer Vision Service\n   &{result}   Vision Analyze  image_url=${IMAGE_URL}  visual_features=${FEATURES}\n   @{faces}    Set Variable  ${result}[faces]\n   FOR  ${face}  IN   @{faces}\n      Log  Age: ${face}[age], Gender: ${face}[gender], Rectangle: ${face}[faceRectangle]\n   END\n
\n

Python

\n

This is a section which describes how to use the library in your\nown Python modules.

\n
\nlibrary = Azure()\nlibrary.init_text_analytics_service()\nlibrary.init_face_service()\nlibrary.init_computer_vision_service()\nlibrary.init_speech_service("westeurope")\n\nresponse = library.sentiment_analyze(\n   text="The rooms were wonderful and the staff was helpful."\n)\nresponse = library.detect_face(\n   image_file=PATH_TO_FILE,\n   face_attributes="age,gender,smile,hair,facialHair,emotion",\n)\nfor item in response:\n   gender = item["faceAttributes"]["gender"]\n   age = item["faceAttributes"]["age"]\n   print(f"Detected a face, gender:{gender}, age: {age}")\n\nresponse = library.vision_analyze(\n   image_url=URL_TO_IMAGE,\n   visual_features="Faces,ImageType",\n)\nmeta = response['metadata']\nprint(\n   f"Image dimensions meta['width']}x{meta['height']} pixels"\n)\n\nfor face in response["faces"]:\n   left = face["faceRectangle"]["left"]\n   top = face["faceRectangle"]["top"]\n   width = face["faceRectangle"]["width"]\n   height = face["faceRectangle"]["height"]\n   print(f"Detected a face, gender:{face['gender']}, age: {face['age']}")\n   print(f"      Face rectangle: (left={left}, top={top})")\n   print(f"      Face rectangle: (width={width}, height={height})")\n\nlibrary.text_to_speech(\n    text="Developer tools for open-source RPA leveraging the Robot Framework ecosystem",\n    neural_voice_style="cheerful",\n    target_file='output.mp3'\n)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:36", + "generated": "2023-11-10 12:56:02", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -38167,7 +38167,7 @@ "name": "RPA.Cloud.Google", "doc": "
\n

Google is a library for operating with Google API endpoints.

\n

Usage requires the following steps:

\n
    \n
  • Create a GCP project
  • \n
  • Enable approriate APIs
  • \n
  • Create credentials (OAuth or service account)
  • \n
  • Install rpaframework-google package
  • \n
\n

Google authentication

\n

Authentication for Google is set with service account JSON file which can be given to the library\nin three different ways or with OAuth2 token, which is used for OAuth authentication.

\n

Methods when using service account:

\n
    \n
  • Method 1 as environment variables, GOOGLE_APPLICATION_CREDENTIALS with path to service account file.
  • \n
  • Method 2 as keyword parameter to Init Storage for example.
  • \n
  • Method 3 as Robocorp vault secret. The vault name and secret key name needs to be given in library init\nor with keyword Set Robocorp Vault. Secret value should contain JSON file contents.
  • \n
\n

Method 1. service account using environment variable

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n\n*** Tasks ***\nInit Google services\n    # NO parameters for Init Vision, expecting to get JSON\n    # with GOOGLE_APPLICATION_CREDENTIALS environment variable\n    Init Vision\n
\n

Method 2. service account with keyword parameter

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n\n*** Tasks ***\nInit Google services\n    Init Speech To Text   /path/to/service_account.json\n
\n

Method 3. setting Robocorp Vault in the library init

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n...       vault_name=googlecloud\n...       vault_secret_key=servicecreds\n\n*** Tasks ***\nInit Google services\n    Init Storage\n
\n

Method 3. setting Robocorp Vault with keyword

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n\n*** Tasks ***\nInit Google services\n    Set Robocorp Vault   vault_name=googlecloud  vault_secret_key=servicecreds\n    Init Storage    use_robocorp_vault=${TRUE}\n
\n

Methods when using OAuth token:

\n
    \n
  • Method 1 as keyword parameter token_file to Init Storage for example.
  • \n
  • Method 2 as Robocorp vault secret. The vault name and secret key name needs to be given in library init\nor with keyword Set Robocorp Vault. Secret value should contain JSON file contents.
  • \n
\n

Method 1. The Google Apps Script and Google Drive services are authenticated using this method.

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n\n*** Variables ***\n@{SCRIPT_SCOPES}     forms   spreadsheets\n\n*** Tasks ***\nInit Google OAuth services\n    Init Apps Script    token_file=oauth_token   ${SCRIPT_SCOPES}\n
\n

Method 2. setting Robocorp Vault in the library init

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n...       vault_name=googlecloud\n...       vault_secret_key=oauth\n...       cloud_auth_type=token\n\n*** Tasks ***\nInit Google services\n    Init Storage\n
\n

Creating and using OAuth token file

\n

The token file can be created using credentials.json by running command:

\n

rpa-google-oauth --credentials <filepath> --service drive or\nrpa-google-oauth --credentials <filepath> --scopes drive.appdata,drive.file,drive.install

\n

This will start web based authentication process, which outputs the token at the end.\nToken could be stored into Robocorp Vault.

\n

Example Vault content.

\n
\n"googlecloud": {\n    "oauth-token": "gANfd123321aabeedYsc"\n}\n
\n

Using the Vault.

\n
\n*** Keywords ***\nSet up Google Drive authentication\n    Set Robocorp Vault   vault_name=googlecloud\n    ...  vault_secret_key=oauth-token\n    ...  cloud_auth_type=token\n    Init Drive\n
\n

Installation

\n

This library, RPA.Cloud.Google is available via rpaframework-google package.

\n

Check the latest package version from PyPI.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary   RPA.Cloud.Google\n\n*** Variables ***\n${SERVICE_ACCOUNT}    ${/}path${/}to${/}service_account.json\n${BUCKET_NAME}            testbucket12213123123\n\n*** Tasks ***\nUpload a file into a new storage bucket\n    [Setup]   Init Storage    ${SERVICE_ACCOUNT}\n    Create Storage Bucket    ${BUCKET_NAME}\n    Upload Storage File      ${BUCKET_NAME}\n    ...   ${/}path${/}to${/}file.pdf\n    ...   myfile.pdf\n    @{files}         List Storage Files   ${BUCKET_NAME}\n    FOR   ${file}  IN   @{files}\n        Log  ${file}\n    END\n
\n

Python

\n
\nfrom RPA.Cloud.Google import Google\n\nlibrary = Google\nservice_account = '/path/to/service_account.json'\n\nlibrary.init_vision(service_account)\nlibrary.init_text_to_speech(service_account)\n\nresponse = library.detect_text('imagefile.png', 'result.json')\nlibrary.synthesize_speech('I want this said aloud', target_file='said.mp3')\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:37", + "generated": "2023-11-10 12:56:02", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -43115,7 +43115,7 @@ "name": "RPA.Crypto", "doc": "
\n

Library for common encryption and hashing operations.

\n

It uses the Fernet\nformat for encryption. More specifically, it uses AES in\nCBC mode with a 128-bit key for encryption and HMAC with SHA256 for\nauthentication.

\n

To use the encryption features, generate a key with the command line\nutility rpa-crypto or with the keyword Generate Key. Store\nthe key in a secure place, such as Robocorp Vault, and load it within\nthe execution before calling encryption/decryption keywords.

\n

Example usage with Robocorp Vault

\n

Create an encryption key with the CLI utility:

\n
\n> rpa-crypto key\nrGx1edA07yz7uD08ChiPSunn8vaauRxw0pAbsal9zjM=\n
\n

Store the key in Robocorp Vault, in this case with the name EncryptionKey.

\n

Load the key from the vault before encryption operations:

\n
\nUse encryption key from vault    EncryptionKey\n${encrypted}=   Encrypt file    orders.xlsx\nAdd work item file    ${encrypted}    name=Orders\n
\n

In another task, this same key can be used to decrypt the file:

\n
\nUse encryption key from vault    EncryptionKey\n${encrypted}=    Get work item file    Orders\n${orders}=   Decrypt file    ${encrypted}\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:38", + "generated": "2023-11-10 12:56:03", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -43602,7 +43602,7 @@ "name": "RPA.Database", "doc": "
\n

Database is a library for handling different database operations.

\n

All database operations are supported. Keywords Query and Get Rows\nreturn values by default in RPA.Table format.

\n

Library is compatible with any Database API Specification 2.0 module.

\n

Workaround for inserting large JSON data for Call Stored Procedure

\n

Workaround is to use instead Query keyword. At the moment there is\nno known fix for the Call Stored Procedure keyword as it fails if\nJSON string is more than 8000 characters long.

\n

Robot Framework

\n
\n${data}=    Load JSON from file    random_data.json\n${json}=    Convert JSON to String    ${data}\n# Single quotes around ${json} string are necessary\nQuery    exec InsertJsonDataToSampleTable '${json}'\n
\n

References:

\n\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary         RPA.Database\n\n*** Tasks ***\nGet Orders From Database\n    Connect To Database  pymysql  tester  user  password  127.0.0.1\n    @{orders}            Query    Select * FROM incoming_orders\n    FOR   ${order}  IN  @{orders}\n        Handle Order  ${order}\n    END\n
\n

Python

\n
\nfrom RPA.Database import Database\nfrom RPA.Robocorp.Vault import FileSecrets\n\nfilesecrets = FileSecrets("secrets.json")\nsecrets = filesecrets.get_secret("databasesecrets")\n\ndb = Database()\ndb.connect_to_database('pymysql',\n                    secrets["DATABASE"],\n                    secrets["USERNAME"],\n                    secrets["PASSWORD"],\n                    '127.0.0.1'\n                    )\norders = db.query("SELECT * FROM incoming_orders")\nfor order in orders:\n    print(order)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:38", + "generated": "2023-11-10 12:56:03", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -44262,7 +44262,7 @@ "name": "RPA.Desktop", "doc": "
\n

Desktop is a cross-platform library for navigating and interacting with\ndesktop environments. It can be used to automate applications through\nthe same interfaces that are available to human users.

\n

The library includes the following features:

\n
    \n
  • Mouse and keyboard input emulation
  • \n
  • Starting and stopping applications
  • \n
  • Finding elements through image template matching
  • \n
  • Scraping text from given regions
  • \n
  • Taking screenshots
  • \n
  • Clipboard management
  • \n
\n
\n

Warning

\n

Windows element selectors are not currently supported, and require the use of RPA.Desktop.Windows

\n
\n

Installation

\n

The basic features such as mouse and keyboard input and application\ncontrol work with a default rpaframework install.

\n

Advanced computer-vision features such as image template matching and\nOCR require an additional library called rpaframework-recognition.

\n

The dependency should be added separately by specifing it in your conda.yaml\nas rpaframework-recognition==5.0.1 for example. If installing recognition\nthrough pip instead of conda, the OCR feature also requires tesseract.

\n

Locating elements

\n

To automate actions on the desktop, a robot needs to interact with various\ngraphical elements such as buttons or input fields. The locations of these\nelements can be found using a feature called locators.

\n

A locator describes the properties or features of an element. This information\ncan be later used to locate similar elements even when window positions or\nstates change.

\n

The currently supported locator types are:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
NameArgumentsDescription
aliasname (str)A custom named locator from the locator database, the default.
imagepath (str)Image of an element that is matched to current screen content.
pointx (int), y (int)Pixel coordinates as absolute position.
offsetx (int), y (int)Pixel coordinates relative to current mouse position.
sizewidth (int), height (int)Region of fixed size, around point or screen top-left
regionleft (int), top (int), right (int), bottom (int)Bounding coordinates for a rectangular region.
ocrtext (str), confidence (float, optional)Text to find from the current screen.
\n

A locator is defined by its type and arguments, divided by a colon.\nSome example usages are shown below. Note that the prefix for alias can\nbe omitted as its the default type.

\n
\nClick       point:50,100\nClick       region:20,20,100,30\n\nMove mouse  image:%{ROBOT_ROOT}/logo.png\nMove mouse  offset:200,0\nClick\n\nClick       alias:SpareBin.Login\nClick       SpareBin.Login\n\nClick       ocr:"Create New Account"\n
\n

You can also pass internal region objects as locators:

\n
\n${region}=  Find Element  ocr:"Customer name"\nClick       ${region}\n
\n

Locator chaining

\n

Often it is not enough to have one locator, but instead an element\nis defined through a relationship of various locators. For this use\ncase the library supports a special syntax, which we will call\nlocator chaining.

\n

An example of chaining:

\n
\n# Read text from area on the right side of logo\nRead text    image:logo.png + offset:600,0 + size:400,200\n
\n

The supported operators are:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
OperatorDescription
then, +Base locator relative to the previous one
and, &&, &Both locators should be found
or, ||, |Either of the locators should be found
not, !The locator should not be found
\n

Further examples:

\n
\n# Click below either label\nClick    (image:name.png or image:email.png) then offset:0,300\n\n# Wait until dialog disappears\nWait for element    not image:cookie.png\n
\n

Named locators

\n

The library supports storing locators in a database, which contains\nall of the required fields and various bits of metadata. This enables\nhaving one source of truth, which can be updated if a website's or applications's\nUI changes. Robot Framework scripts can then only contain a reference\nto a stored locator by name.

\n

The main way to create named locators is with VSCode.

\n

Read more on identifying elements and crafting locators:

\n\n

Keyboard and mouse

\n

Keyboard keywords can emulate typing text, but also pressing various function keys.\nThe name of a key is case-insensitive and spaces will be converted to underscores,\ni.e. the key Page Down and page_down are equivalent.

\n

The following function keys are supported:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
KeyDescription
shiftA generic Shift key. This is a modifier.
shift_lThe left Shift key. This is a modifier.
shift_rThe right Shift key. This is a modifier.
ctrlA generic Ctrl key. This is a modifier.
ctrl_lhe left Ctrl key. This is a modifier.
ctrl_rThe right Ctrl key. This is a modifier.
altA generic Alt key. This is a modifier.
alt_lThe left Alt key. This is a modifier.
alt_rThe right Alt key. This is a modifier.
alt_grThe AltGr key. This is a modifier.
cmdA generic command button (Windows / Command / Super key). This may be a modifier.
cmd_lThe left command button (Windows / Command / Super key). This may be a modifier.
cmd_rThe right command button (Windows / Command / Super key). This may be a modifier.
upAn up arrow key.
downA down arrow key.
leftA left arrow key.
rightA right arrow key.
enterThe Enter or Return key.
spaceThe Space key.
tabThe Tab key.
backspaceThe Backspace key.
deleteThe Delete key.
escThe Esc key.
homeThe Home key.
endThe End key.
page_downThe Page Down key.
page_upThe Page Up key.
caps_lockThe Caps Lock key.
f1 to f20The function keys.
insertThe Insert key. This may be undefined for some platforms.
menuThe Menu key. This may be undefined for some platforms.
num_lockThe Num Lock key. This may be undefined for some platforms.
pauseThe Pause / Break key. This may be undefined for some platforms.
print_screenThe Print Screen key. This may be undefined for some platforms.
scroll_lockThe Scroll Lock key. This may be undefined for some platforms.
\n

When controlling the mouse, there are different types of actions that can be\ndone. Same formatting rules as function keys apply. They are as follows:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
ActionDescription
clickClick with left mouse button
left_clickClick with left mouse button
double_clickDouble click with left mouse button
triple_clickTriple click with left mouse button
right_clickClick with right mouse button
\n

The supported mouse button types are left, right, and middle.

\n

Examples

\n

Both Robot Framework and Python examples follow.

\n

The library must be imported first.

\n
\n*** Settings ***\nLibrary    RPA.Desktop\n
\n
\nfrom RPA.Desktop import Desktop\ndesktop = Desktop()\n
\n

The library can open applications and interact with them through\nkeyboard and mouse events.

\n
\n*** Keywords ***\nWrite entry in accounting\n    [Arguments]    ${entry}\n    Open application    erp_client.exe\n    Click         image:%{ROBOT_ROOT}/images/create.png\n    Type text     ${entry}\n    Press keys    ctrl    s\n    Press keys    enter\n
\n
\ndef write_entry_in_accounting(entry):\n    desktop.open_application("erp_client.exe")\n    desktop.click(f"image:{ROBOT_ROOT}/images/create.png")\n    desktop.type_text(entry)\n    desktop.press_keys("ctrl", "s")\n    desktop.press_keys("enter")\n
\n

Targeting can be currently done using coordinates (absolute or relative),\nbut using template matching is preferred.

\n
\n*** Keywords ***\nWrite to field\n    [Arguments]  ${text}\n    Move mouse   image:input_label.png\n    Move mouse   offset:200,0\n    Click\n    Type text    ${text}\n    Press keys   enter\n
\n
\ndef write_to_field(text):\n    desktop.move_mouse("image:input_label.png")\n    desktop.move_mouse("offset:200,0")\n    desktop.click()\n    desktop.type_text(text)\n    desktop.press_keys("enter")\n
\n

Elements can be found by text too.

\n
\n*** Keywords ***\nClick New\n    Click       ocr:New\n
\n
\ndef click_new():\n    desktop.click('ocr:"New"')\n
\n

It is recommended to wait for the elements to be visible before\ntrying any interaction. You can also pass region objects as locators.

\n
\n*** Keywords ***\nClick New\n    ${region}=  Wait For element  ocr:New\n    Click       ${region}\n
\n
\ndef click_new():\n    region = desktop.wait_for_element("ocr:New")\n    desktop.click(region)\n
\n

Another way to find elements by offsetting from an anchor:

\n
\n*** Keywords ***\nType Notes\n    [Arguments]        ${text}\n    Click With Offset  ocr:Notes  500  0\n    Type Text          ${text}\n
\n
\ndef type_notes(text):\n    desktop.click_with_offset("ocr:Notes", 500, 0)\n    desktop.type_text(text)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:38", + "generated": "2023-11-10 12:56:03", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -45472,7 +45472,7 @@ "name": "RPA.Desktop.Clipboard", "doc": "
\n

DEPRECATED!! Use library RPA.Desktop's clipboard functionality instead.

\n

Clipboard is a library for managing clipboard - copy text to,\npaste text from, and clear clipboard contents.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary    RPA.Desktop.Clipboard\n\n*** Tasks ***\nClipping\n    Copy To Clipboard   Text from Robot to clipboard\n    ${var}=             Paste From Clipboard\n    Clear Clipboard\n
\n

Python

\n
\nfrom RPA.Desktop.Clipboard import Clipboard\n\nclip = Clipboard()\nclip.copy_to_clipboard('Text from Python to clipboard')\ntext = clip.paste_from_clipboard()\nprint(f"clipboard had text: '{text}'")\nclip.clear_clipboard()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:38", + "generated": "2023-11-10 12:56:03", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -45530,7 +45530,7 @@ "name": "RPA.Desktop.OperatingSystem", "doc": "
\n

OperatingSystem is a cross-platform library for managing\ncomputer properties and actions.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary    RPA.Desktop.OperatingSystem\n\n*** Tasks ***\nGet computer information\n    ${boot_time}=   Get Boot Time  as_datetime=${TRUE}\n    ${machine}=     Get Machine Name\n    ${username}=    Get Username\n    &{memory}=      Get Memory Stats\n    Log Many        ${memory}\n
\n

Python

\n
\nfrom RPA.Desktop.OperatingSystem import OperatingSystem\n\ndef get_computer_information():\n    ops = OperatingSystem()\n    print(f"Boot time    : { ops.get_boot_time(as_datetime=True) }"\n          f"Machine name : { ops.get_machine_name() }"\n          f"Username     : { ops.get_username() }"\n          f"Memory       : { ops.get_memory_stats() }")\n\nif __name__ == "__main__":\n    get_computer_information()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:39", + "generated": "2023-11-10 12:56:03", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -45794,7 +45794,7 @@ "name": "RPA.Desktop.Windows", "doc": "
\n

Windows is a library for managing the Windows operating system.

\n

DEPRECATION WARNING! USE RPA.Windows library instead.

\n

For Windows desktop automation Robocorp recommends the RPA.Windows library.

\n

No further updates will be released for this library and new functionality will continue\nto be developed in RPA.Windows library.

\n

Running Windows applications

\n

Windows applications can be started in several ways. The library supports\nthe following keywords:

\n
    \n
  • Open Application (dispatch Office applications)
  • \n
  • Open File (open the file as process which opens the associated application)
  • \n
  • Open Executable (uses pywinauto start)
  • \n
  • Open Using Run Dialog (uses Windows run dialog)
  • \n
  • Open From Search (uses Windows search dialog)
  • \n
\n

Locators

\n

Locator is used to identify the element for interaction - usually for a mouse click.

\n

Locators can investigated for application once it has been opened by calling\nthe keyword get_windows_elements which can store locator information into JSON file\nand screenshot of the element into an image file.

\n

Identifying locator

\n

The element needs to be identified by a unique method, for example, "Three" for button 3\nin the Calculator application. It can be given either as Three or name:Three.

\n

Possible search criterias:

\n
    \n
  • name
  • \n
  • class (class_name)
  • \n
  • type (control_type)
  • \n
  • id (automation_id)
  • \n
  • any if none was defined
  • \n
\n

The current method of inspecting elements on Windows is inspect.exe which is part\nof Windows SDK.

\n

Keyboard

\n

The keyword send_keys can be used to send keys to the active window. The keyword\ntype_keys sends keys to the active window element.

\n

Special key codes are documented on pywinauto\ndocumentation page.

\n

FAQ

\n
    \n
  1. I see error message AttributeError: module 'win32com.gen_py.00020813-0000-0000-C000-000000000046x0x1x9' has no attribute 'CLSIDToClassMap'
  2. \n
\n
    \n
  1. From PowerShell run this command: Remove-Item -path $env:LOCALAPPDATATempgen_py -recurse
  2. \n
\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary          RPA.Desktop.Windows\nSuite Teardown   Close all applications\n\n*** Tasks ***\nOpen Calculator using run dialog\n    ${result}=              Open using run dialog    calc.exe   Calculator\n    ${result}=              Get Window Elements\n    Send Keys               5*2=\n    ${result}=              Get element             partial name:Display is\n    Log Many                ${result}\n    ${result}=              Get element rich text   id:CalculatorResults\n    Should Be Equal As Strings  ${result}  Display is 10\n    ${result}=              Get element rectangle   partial name:Display is\n    ${result}=              Is Element Visible      CalculatorResults\n    ${result}=              Is Element Enabled      partial name:Display is\n
\n

Python

\n
\nfrom RPA.Desktop.Windows import Windows\n\nwin = Windows()\n\ndef open_calculator():\n    win.open_from_search("calc.exe", "Calculator")\n    elements = win.get_window_elements()\n\ndef make_calculations(expression):\n    win.send_keys(expression)\n    result = win.get_element_rich_text('id:CalculatorResults')\n    return int(result.strip('Display is '))\n\nif __name__ == "__main__":\n    open_calculator()\n    exp = '5*2='\n    result = make_calculations(exp)\n    print(f"Calculation result of '{exp}' is '{result}'")\n    win.close_all_applications()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:39", + "generated": "2023-11-10 12:56:04", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -48205,7 +48205,7 @@ "name": "RPA.Dialogs", "doc": "
\n

NOTICE: The Dialogs library is now marked for deprecation.\nThe library will be replaced by Assistant. In order to get more\ninformation about the changes and the migration guide, please\ncheck the release note: https://updates.robocorp.com/release/Gfko5-rpaframework-2200

\n

The Dialogs library provides a way to display information to a user\nand request input while a robot is running. It allows building processes\nthat require human interaction.

\n

Some examples of use-cases could be the following:

\n
    \n
  • Displaying generated files after an execution is finished
  • \n
  • Displaying dynamic and user-friendly error messages
  • \n
  • Requesting passwords or other personal information
  • \n
  • Automating based on files created by the user
  • \n
\n

Workflow

\n

The library is used to create dialogs, i.e. windows, that can be composed\non-the-fly based on the current state of the execution.

\n

The content of the dialog is defined by calling relevant keywords\nsuch as Add text or Add file input. When the dialog is opened\nthe content is generated based on the previous keywords.

\n

Depending on the way the dialog is started, the execution will either\nblock or continue while the dialog is open. During this time the user\ncan freely edit any possible input fields or handle other tasks.

\n

After the user has successfully submitted the dialog, any possible\nentered input will be returned as a result. The user also has the option\nto abort by closing the dialog window forcefully.

\n

Results

\n

Each input field has a required name argument that controls what\nthe value will be called in the result object. Each input name should be\nunique, and must not be called submit as that is reserved for the submit\nbutton value.

\n

A result object is a Robot Framework DotDict, where each key\nis the name of the input field and the value is what the user entered.\nThe data type of each field depends on the input. For instance,\na text input will have a string, a checkbox will have a boolean, and\na file input will have a list of paths.

\n

If the user closed the window before submitting or there was an internal\nerror, the library will raise an exception and the result values will\nnot be available.

\n

Examples

\n
\nSuccess dialog\n    Add icon      Success\n    Add heading   Your orders have been processed\n    Add files     *.txt\n    Run dialog    title=Success\n\nFailure dialog\n    Add icon      Failure\n    Add heading   There was an error\n    Add text      The assistant failed to login to the Enterprise portal\n    Add link      https://robocorp.com/docs    label=Troubleshooting guide\n    Run dialog    title=Failure\n\nLarge dialog\n    Add heading    A real chonker   size=large\n    Add image      fat-cat.jpeg\n    Run dialog     title=Large    height=1024    width=1024\n\nConfirmation dialog\n    Add icon      Warning\n    Add heading   Delete user ${username}?\n    Add submit buttons    buttons=No,Yes    default=Yes\n    ${result}=    Run dialog\n    IF   $result.submit == "Yes"\n        Delete user    ${username}\n    END\n\nInput form dialog\n    Add heading       Send feedback\n    Add text input    email    label=E-mail address\n    Add text input    message\n    ...    label=Feedback\n    ...    placeholder=Enter feedback here\n    ...    rows=5\n    ${result}=    Run dialog\n    Send feedback message    ${result.email}  ${result.message}\n\nDialog as progress indicator\n    Add heading    Please wait while I open a browser\n    ${dialog}=     Show dialog    title=Please wait    on_top=${TRUE}\n    Open available browser    https://robocorp.com\n    Close dialog   ${dialog}\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:40", + "generated": "2023-11-10 12:56:04", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -49492,7 +49492,7 @@ "name": "RPA.DocumentAI", "doc": "
\n

Wrapper library offering generic keywords for initializing, scanning and\nretrieving results as fields from documents (PDF, PNG etc.).

\n

Library requires at the minimum rpaframework version 19.0.0.

\n

This is a helper facade for the following libraries:

\n
    \n
  • RPA.Cloud.Google (requires rpaframework-google)
  • \n
  • RPA.DocumentAI.Base64AI
  • \n
  • RPA.DocumentAI.Nanonets
  • \n
\n

Where the following steps are required:

\n
    \n
  1. Engine initialization: Init Engine
  2. \n
  3. Document scan: Predict
  4. \n
  5. Result retrieval: Get Result
  6. \n
\n

So no matter the engine you're using, the very same keywords can be used, as only\nthe passed parameters will differ (please check the docs on each library for\nparticularities). Once initialized, you can jump between the engines with\nSwitch Engine. Before scanning documents, you must configure the service first,\nwith a model to scan the files with and an API key for authorizing the access.

\n

See Portal example: https://robocorp.com/portal/robot/robocorp/example-document-ai

\n

Example: Robot Framework

\n
\n*** Settings ***\nLibrary    RPA.DocumentAI\n\n*** Tasks ***\nScan Documents\n    Init Engine    base64ai    vault=document_ai:base64ai\n    Init Engine    nanonets    vault=document_ai:nanonets\n\n    Switch Engine   base64ai\n    Predict    invoice.png\n    ${data} =    Get Result\n    Log List    ${data}\n\n    Switch Engine   nanonets\n    Predict    invoice.png      model=858e4b37-6679-4552-9481-d5497dfc0b4a\n    ${data} =    Get Result\n    Log List    ${data}\n
\n

Example: Python

\n
\nfrom RPA.DocumentAI import DocumentAI, EngineName\n\nlib_docai = DocumentAI()\nlib_docai.init_engine(\n    EngineName.GOOGLE, vault="document_ai:serviceaccount", region="eu"\n)\nlib_docai.predict(\n    "invoice.pdf", model="df1d166771005ff4",\n    project_id="complete-agency-347912", region="eu"\n)\nprint(lib_docai.get_result())\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:40", + "generated": "2023-11-10 12:56:04", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -49807,7 +49807,7 @@ "name": "RPA.DocumentAI.Base64AI", "doc": "
\n

Library to support Base64.ai service for intelligent\ndocument processing (IDP).

\n

Library requires at the minimum rpaframework version 19.0.0.

\n

Service supports identifying fields in the documents, which can be given to the\nservice in multiple different file formats and via URL.

\n

Robot Framework example usage

\n
\n*** Settings ***\nLibrary   RPA.DocumentAI.Base64AI\nLibrary   RPA.Robocorp.Vault\n\n*** Tasks ***\nIdentify document\n    ${secrets}=   Get Secret  base64ai-auth\n    Set Authorization  ${secrets}[email-address]   ${secrets}[apikey]\n    ${results}=  Scan Document File\n    ...   ${CURDIR}${/}invoice.pdf\n    ...   model_types=finance/check/usa,finance/invoice/usa\n    # Scan response contains list of detected models in the document\n    FOR  ${result}  IN  @{results}\n        Log To Console  Model: ${result}[model]\n        Log To Console  Field keys: ${{','.join($result['fields'].keys())}}\n        Log To Console  Fields: ${result}[fields]\n        Log To Console  Text (OCR): ${result}[ocr]\n    END\n
\n

Python example usage

\n
\nfrom RPA.DocumentAI.Base64AI import Base64AI\nfrom RPA.Robocorp.Vault import Vault\n\nsecrets = Vault().get_secret("base64ai-auth")\nbaselib = Base64AI()\nbaselib.set_authorization(secrets["email-address"], secrets["apikey"])\nresult = baselib.scan_document_file(\n    "invoice.pdf",\n    model_types="finance/invoice,finance/check/usa",\n)\nfor r in result:\n    print(f"Model: {r['model']}")\n    for key, props in r["fields"].items():\n        print(f"FIELD {key}: {props['value']}")\n    print(f"Text (OCR): {r['ocr']}")\n
\n

Portal example: https://github.com/robocorp/example-idp-base64

\n
\n", "version": "", - "generated": "2023-11-10 12:47:41", + "generated": "2023-11-10 12:56:04", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -50320,7 +50320,7 @@ "name": "RPA.DocumentAI.DocumentAI", "doc": "
\n

Wrapper library offering generic keywords for initializing, scanning and\nretrieving results as fields from documents (PDF, PNG etc.).

\n

Library requires at the minimum rpaframework version 19.0.0.

\n

This is a helper facade for the following libraries:

\n
    \n
  • RPA.Cloud.Google (requires rpaframework-google)
  • \n
  • RPA.DocumentAI.Base64AI
  • \n
  • RPA.DocumentAI.Nanonets
  • \n
\n

Where the following steps are required:

\n
    \n
  1. Engine initialization: Init Engine
  2. \n
  3. Document scan: Predict
  4. \n
  5. Result retrieval: Get Result
  6. \n
\n

So no matter the engine you're using, the very same keywords can be used, as only\nthe passed parameters will differ (please check the docs on each library for\nparticularities). Once initialized, you can jump between the engines with\nSwitch Engine. Before scanning documents, you must configure the service first,\nwith a model to scan the files with and an API key for authorizing the access.

\n

See Portal example: https://robocorp.com/portal/robot/robocorp/example-document-ai

\n

Example: Robot Framework

\n
\n*** Settings ***\nLibrary    RPA.DocumentAI\n\n*** Tasks ***\nScan Documents\n    Init Engine    base64ai    vault=document_ai:base64ai\n    Init Engine    nanonets    vault=document_ai:nanonets\n\n    Switch Engine   base64ai\n    Predict    invoice.png\n    ${data} =    Get Result\n    Log List    ${data}\n\n    Switch Engine   nanonets\n    Predict    invoice.png      model=858e4b37-6679-4552-9481-d5497dfc0b4a\n    ${data} =    Get Result\n    Log List    ${data}\n
\n

Example: Python

\n
\nfrom RPA.DocumentAI import DocumentAI, EngineName\n\nlib_docai = DocumentAI()\nlib_docai.init_engine(\n    EngineName.GOOGLE, vault="document_ai:serviceaccount", region="eu"\n)\nlib_docai.predict(\n    "invoice.pdf", model="df1d166771005ff4",\n    project_id="complete-agency-347912", region="eu"\n)\nprint(lib_docai.get_result())\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:41", + "generated": "2023-11-10 12:56:04", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -50635,7 +50635,7 @@ "name": "RPA.DocumentAI.Nanonets", "doc": "
\n

Library to support Nanonets service for intelligent document processing (IDP).

\n

Library requires at the minimum rpaframework version 19.0.0.

\n

Service supports identifying fields in the documents, which can be given to the\nservice in multiple different file formats and via URL.

\n

Robot Framework example usage

\n
\n*** Settings ***\nLibrary   RPA.DocumentAI.Nanonets\nLibrary   RPA.Robocorp.Vault\n\n*** Tasks ***\nIdentify document\n    ${secrets}=   Get Secret  nanonets-auth\n    Set Authorization    ${secrets}[apikey]\n    ${result}=    Predict File\n    ...  ${CURDIR}${/}files${/}eckero.jpg\n    ...  ${secrets}[receipts-model-id]\n    ${fields}=    Get Fields From Prediction Result    ${result}\n    FOR    ${field}    IN    @{fields}\n        Log To Console    Label:${field}[label] Text:${field}[ocr_text]\n    END\n    ${tables}=    Get Tables From Prediction Result    ${result}\n    FOR    ${table}    IN    @{tables}\n        FOR    ${rows}    IN    ${table}[rows]\n            FOR    ${row}    IN    @{rows}\n                ${cells}=    Evaluate    [cell['text'] for cell in $row]\n                Log To Console    ROW:${{" | ".join($cells)}}\n            END\n        END\n    END\n
\n

Python example usage

\n
\nfrom RPA.DocumentAI.Nanonets import Nanonets\nfrom RPA.Robocorp.Vault import Vault\n\nsecrets = Vault().get_secret("nanonets-auth")\nnanolib = Nanonets()\nnanolib.set_authorization(secrets["apikey"])\nresult = nanolib.predict_file(file_to_scan, secrets["receipts-model-id"])\nfields = nanolib.get_fields_from_prediction_result(result)\nfor field in fields:\n    print(f"Label: {field['label']} Text: {field['ocr_text']}")\ntables = nanolib.get_tables_from_prediction_result(result)\nfor table in tables:\n    rpatable = Tables().create_table(table["rows"])\n    for row in table["rows"]:\n        cells = [cell["text"] for cell in row]\n        print(f"ROW: {' | '.join(cells)}")\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:41", + "generated": "2023-11-10 12:56:04", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -50933,7 +50933,7 @@ "name": "RPA.Email.Exchange", "doc": "
\n

Exchange is a library for sending, reading, and deleting emails.\nExchange is interfacing with Exchange Web Services (EWS).

\n

For more information about server settings, see\nthis Microsoft support article.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary     RPA.Email.Exchange\n...     vault_name=email_oauth_microsoft    vault_token_key=token\n...     tenant=ztzvn.onmicrosoft.com  # your custom tenant here\nTask Setup      Ensure Auth\n\n*** Variables ***\n${ACCOUNT}              ACCOUNT_NAME\n${RECIPIENT_ADDRESS}    RECIPIENT\n${IMAGES}               myimage.png\n${ATTACHMENTS}          C:${/}files${/}mydocument.pdf\n\n*** Keywords ***\nEnsure Auth\n    ${secrets} =    Get Secret    email_oauth_microsoft\n    RPA.Email.Exchange.Authorize    ${ACCOUNT}\n    ...    is_oauth=${True}  # use the OAuth2 auth code flow (required)\n    ...    client_id=${secrets}[client_id]  # app ID\n    ...    client_secret=${secrets}[client_secret]  # app password\n    ...    token=${secrets}[token]  # token dict (access, refresh, scope etc.)\n\n*** Tasks ***\nTask of sending email\n    Send Message  recipients=${RECIPIENT_ADDRESS}\n    ...           subject=Exchange Message from RPA Robot\n    ...           body=<p>Exchange RPA Robot message body<br><img src='myimage.png'/></p>\n    ...           save=${TRUE}\n    ...           html=${TRUE}\n    ...           images=${IMAGES}\n    ...           cc=EMAIL_ADDRESS\n    ...           bcc=EMAIL_ADDRESS\n    ...           attachments=${ATTACHMENTS}\n\nTask of listing messages\n    # Attachments are saved specifically with a keyword Save Attachments\n    ${messages}=    List Messages\n    FOR    ${msg}    IN    @{messages}\n        Log Many    ${msg}\n        ${attachments}=    Run Keyword If    "${msg}[subject]"=="about my orders"\n        ...    Save Attachments\n        ...    ${msg}\n        ...    save_dir=${CURDIR}${/}savedir\n    END\n    # Using save_dir all attachments in listed messages are saved\n    ${messages}=    List Messages\n    ...    INBOX/Problems/sub1\n    ...    criterion=subject:'about my orders'\n    ...    save_dir=${CURDIR}${/}savedir2\n    FOR    ${msg}    IN    @{messages}\n        Log Many    ${msg}\n    END\n\nTask of moving messages\n    Move Messages    criterion=subject:'about my orders'\n    ...    source=INBOX/Processed Purchase Invoices/sub2\n    ...    target=INBOX/Problems/sub1\n
\n

Python

\n
\nfrom RPA.Email.Exchange import Exchange\nfrom RPA.Robocorp.Vault import Vault\n\nvault_name = "email_oauth_microsoft"\nsecrets = Vault().get_secret(vault_name)\nex_account = "ACCOUNT_NAME"\n\nmail = Exchange(\n    vault_name=vault_name,\n    vault_token_key="token",\n    tenant="ztzvn.onmicrosoft.com"\n)\nmail.authorize(\n    username=ex_account,\n    is_oauth=True,\n    client_id=secrets["client_id"],\n    client_secret=secrets["client_secret"],\n    token=secrets["token"]\n)\nmail.send_message(\n    recipients="RECIPIENT",\n    subject="Message from RPA Python",\n    body="RPA Python message body",\n)\n
\n

OAuth2

\n

The OAuth2 flow is the only way of authorizing at the moment as Microsoft disabled\nentirely the usage of passwords, even App Passwords. And since you have to work\nwith tokens now and because this library has the capability to automatically\nrefresh an expired token, please don't forget to initialize the library with the\nfollowing parameters: vault_name, vault_token_key and tenant.

\n

Learn more on how to use the OAuth2 flow in this Portal robot\nexample-oauth-email.

\n

About criterion parameter

\n

Following table shows possible criterion keys that can be used to filter emails.\nThere apply to all keywords which have criterion parameter.

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
KeyEffective search
subjectsubject to match
subject_containssubject to contain
bodybody to match
body_containsbody to contain
sendersender (from) to match
sender_containssender (from) to contain
beforereceived time before this time
afterreceived time after this time
betweenreceived time between start and end
categorycategories to match
category_containscategories to contain
importanceimportance to match
\n

Keys before, after and between at the moment support two\ndifferent timeformats either %d-%m-%Y %H:%M or %d-%m-%Y. These\nkeys also support special string NOW which can be used especially\ntogether with keyword Wait for message  criterion=after:NOW.

\n

When giving time which includes hours and minutes then the whole\ntime string needs to be enclosed into single quotes.

\n
\nbefore:25-02-2022\nafter:NOW\nbetween:'31-12-2021 23:50 and 01-01-2022 00:10'\n
\n

Different criterion keys can be combined.

\n
\nsubject_contains:'new year' between:'31-12-2021 23:50 and 01-01-2022 00:10'\n
\n

Please note that all values in the criterion that contain spaces need\nto be enclosed within single quotes.

\n

In the following example the email subject is going to matched\nonly against new not new year.

\n
\nsubject_contains:new year\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:42", + "generated": "2023-11-10 12:56:05", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -52334,7 +52334,7 @@ "name": "RPA.Email.ImapSmtp", "doc": "
\n

ImapSmtp is a library for sending, reading, and deleting emails.\nImapSmtp is interfacing with SMTP and IMAP protocols.

\n

*About criteria argument*

\n

Various keywords like List Messages and Move Messages have keyword\nargument called criterion which can be used to filter emails according\nto given criteria.

\n

Syntax needs to according to specification and more information about that\ncan be read from https://robocorp.com/docs/development-guide/email/sending-emails-with-gmail-smtp#listing-email-messages-by-criteria

\n

Troubleshooting

\n
    \n
  • \n
    Authentication error with GMail - "Application-specific password required"
    \n

    See: https://support.google.com/mail/answer/185833?hl=en

    \n
    \n
    \n
  • \n
  • \n
    More secure apps (XOAUTH2 protocol) - Use the OAuth2 flow as in this Portal robot:
    \n

    example-oauth-email

    \n

    Make sure to specify a provider (and optionally a tenant) when importing\nthe library and planning to use this flow.

    \n
    \n
    \n
  • \n
\n

Examples

\n

Robot Framework

\n

It is highly recommended to secure your passwords and take care\nthat they are not stored in version control by mistake.\nSee RPA.Robocorp.Vault to see how to store secrets in\nRobocorp Vault.

\n

When sending HTML content with IMG tags, the src filenames must match\nthe base image name given with the images parameter.

\n
\n*** Settings ***\nLibrary     RPA.Email.ImapSmtp   smtp_server=smtp.gmail.com  smtp_port=587\nTask Setup  Authorize  account=${GMAIL_ACCOUNT}  password=${GMAIL_PASSWORD}\n\n*** Variables ***\n${GMAIL_ACCOUNT}        ACCOUNT_NAME\n${GMAIL_PASSWORD}       APP_PASSWORD\n${RECIPIENT_ADDRESS}    RECIPIENT\n${BODY_IMG1}            ${IMAGEDIR}${/}approved.png\n${BODY_IMG2}            ${IMAGEDIR}${/}invoice.png\n${EMAIL_BODY}     <h1>Heading</h1><p>Status: <img src='approved.png' alt='approved image'/></p>\n...               <p>INVOICE: <img src='invoice.png' alt='invoice image'/></p>\n\n*** Tasks ***\nSending email\n    Send Message  sender=${GMAIL_ACCOUNT}\n    ...           recipients=${RECIPIENT_ADDRESS}\n    ...           subject=Message from RPA Robot\n    ...           body=RPA Robot message body\n\nSending HTML Email With Image\n    [Documentation]     Sending email with HTML content and attachment\n    Send Message\n    ...                 sender=${GMAIL_ACCOUNT}\n    ...                 recipients=${RECIPIENT_ADDRESS}\n    ...                 subject=HTML email with body images (2) plus one attachment\n    ...                 body=${EMAIL_BODY}\n    ...                 html=${TRUE}\n    ...                 images=${BODY_IMG1}, ${BODY_IMG2}\n    ...                 attachments=example.png\n
\n

Python

\n
\nfrom RPA.Email.ImapSmtp import ImapSmtp\n\ngmail_account = "ACCOUNT_NAME"\ngmail_password = "APP_PASSWORD"\n\nmail = ImapSmtp(smtp_server="smtp.gmail.com", smtp_port=587)\nmail.authorize(account=gmail_account, password=gmail_password)\nmail.send_message(\n    sender=gmail_account,\n    recipients="RECIPIENT",\n    subject="Message from RPA Python",\n    body="RPA Python message body",\n)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:43", + "generated": "2023-11-10 12:56:05", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -54523,7 +54523,7 @@ "name": "RPA.Excel.Application", "doc": "
\n

Excel.Application is a library for controlling the Excel application.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary             RPA.Excel.Application\nTask Setup          Open Application\nTask Teardown       Quit Application\n\n*** Tasks ***\nManipulate Excel application\n    Open Workbook           workbook.xlsx\n    Set Active Worksheet    sheetname=new stuff\n    Write To Cells          row=1\n    ...                     column=1\n    ...                     value=my data\n    Save Excel\n\nRun Excel Macro\n    Open Workbook   orders_with_macro.xlsm\n    Run Macro       Sheet1.CommandButton1_Click\n\nExport Workbook as PDF\n    Open Workbook           workbook.xlsx\n    Export as PDF           workbook.pdf\n
\n

Python

\n
\nfrom RPA.Excel.Application import Application\n\napp = Application()\n\napp.open_application()\napp.open_workbook('workbook.xlsx')\napp.set_active_worksheet(sheetname='new stuff')\napp.write_to_cells(row=1, column=1, value='new data')\napp.save_excel()\napp.quit_application()\n
\n

Caveats

\n

This library works on a Windows operating system with UI enabled only, and you must\nensure that you open the app first with Open Application before running any\nother relevant keyword which requires to operate on an open app. The application is\nautomatically closed at the end of the task execution, so this can be changed by\nimporting the library with the autoexit=${False} setting.

\n
\n*** Settings ***\nLibrary     RPA.Excel|Outlook|Word.Application    autoexit=${False}\n
\n

If you're running the Process by Control Room through a custom self-hosted Worker\nservice, then please make sure that you enable an RDP session by ticking "Use\nDesktop Connection" under the Step configuration.

\n

If you still encounter issues with opening a document, please ensure that file can\nbe opened first manually and dismiss any alert potentially blocking the process.

\n

Check the documentation below for more info:

\n\n
\n", "version": "", - "generated": "2023-11-10 12:47:43", + "generated": "2023-11-10 12:56:06", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -55179,7 +55179,7 @@ "name": "RPA.Excel.Files", "doc": "
\n

The Excel.Files library can be used to read and write Excel\nfiles without the need to start the actual Excel application.

\n

It supports both legacy .xls files and modern .xlsx files.

\n

Note: To run macros or load password protected worksheets,\nplease use the Excel application library.

\n

Examples

\n

Robot Framework

\n

A common use-case is to load an existing Excel file as a table,\nwhich can be iterated over later in a Robot Framework keyword or task:

\n
\n*** Settings ***\nLibrary    RPA.Tables\nLibrary    RPA.Excel.Files\n\n*** Keywords ***\nRead orders as table\n    Open workbook    ${ORDERS_FILE}\n    ${worksheet}=    Read worksheet   header=${TRUE}\n    ${orders}=       Create table     ${worksheet}\n    [Return]         ${orders}\n    [Teardown]       Close workbook\n
\n

Processing all worksheets in the Excel file and checking row count:

\n
\n*** Settings ***\nLibrary    RPA.Excel.Files\n\n*** Variables ***\n${EXCEL_FILE}   /path/to/excel.xlsx\n\n*** Tasks ***\nRows in the sheet\n    [Setup]      Open Workbook    ${EXCEL_FILE}\n    @{sheets}=   List Worksheets\n    FOR  ${sheet}  IN   @{sheets}\n        ${count}=  Get row count in the sheet   ${sheet}\n        Log   Worksheet '${sheet}' has ${count} rows\n    END\n\n*** Keywords ***\nGet row count in the sheet\n    [Arguments]      ${SHEET_NAME}\n    ${sheet}=        Read Worksheet   ${SHEET_NAME}\n    ${rows}=         Get Length  ${sheet}\n    [Return]         ${rows}\n
\n

Creating a new Excel file with a dictionary:

\n
\n*** Tasks ***\nCreating new Excel\n    Create Workbook  my_new_excel.xlsx\n    FOR    ${index}    IN RANGE    20\n        &{row}=       Create Dictionary\n        ...           Row No   ${index}\n        ...           Amount   ${index * 25}\n        Append Rows to Worksheet  ${row}  header=${TRUE}\n    END\n    Save Workbook\n
\n

Creating a new Excel file with a list:

\n
\n*** Variables ***\n@{heading}   Row No   Amount\n@{rows}      ${heading}\n\n*** Tasks ***\nCreating new Excel\n    Create Workbook  my_new_excel.xlsx\n    FOR    ${index}    IN RANGE   1  20\n        @{row}=         Create List   ${index}   ${index * 25}\n        Append To List  ${rows}  ${row}\n    END\n    Append Rows to Worksheet  ${rows}\n    Save Workbook\n
\n

Python

\n

The library can also be imported directly into Python.

\n
\nfrom RPA.Excel.Files import Files\n\ndef read_excel_worksheet(path, worksheet):\n    lib = Files()\n    lib.open_workbook(path)\n    try:\n        return lib.read_worksheet(worksheet)\n    finally:\n        lib.close_workbook()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:44", + "generated": "2023-11-10 12:56:06", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -57111,7 +57111,7 @@ "name": "RPA.FTP", "doc": "
\n

FTP library can be used to access an FTP server,\nand interact with files.

\n

The library is based on Python's built-in ftplib.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary    RPA.FTP\n\n*** Variables ***\n${HOST}       127.0.0.1\n${PORT}       27345\n${USER}       user\n${PASS}       12345\n\n*** Tasks ***\nList files on the server directory\n    Connect   ${HOST}  ${PORT}  ${USER}  ${PASS}\n    @{files}  List Files\n    FOR  ${file}  IN  @{files}\n        Log  ${file}\n    END\n
\n

Python

\n
\nfrom RPA.FTP import FTP\n\nlibrary = FTP()\nlibrary.connect('127.0.0.1', 27345, 'user', '12345')\nfiles = library.list_files()\nfor f in files:\n    print(f)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:45", + "generated": "2023-11-10 12:56:06", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -57714,7 +57714,7 @@ "name": "RPA.FileSystem", "doc": "
\n

The FileSystem library can be used to interact with files and directories\non the local computer. It can inspect and list files, remove and create them,\nread contents from files, and write data out.

\n

It shadows the built-in OperatingSystem library but contains keywords\nwhich are more RPA-oriented.

\n

Examples

\n

Robot Framework

\n

The library allows, for instance, iterating over files and inspecting them.

\n
\n*** Settings ***\nLibrary    RPA.FileSystem\n\n*** Keywords ***\nDelete large files\n    ${files}=    List files in directory    archive/orders/\n    FOR    ${file}  IN  @{FILES}\n        Run keyword if    ${file.size} > 10**8    Remove file    ${file}\n    END\n\nRead process output\n    Start external program\n    Wait until modified    process.log\n    ${output}=  Read file  process.log\n    [Return]    ${output}\n
\n

Python

\n

The library can also be used inside Python.

\n
\nfrom RPA.FileSystem import FileSystem\n\ndef move_to_archive():\n    lib = FileSystem()\n\n    matches = lib.find_files("**/*.xlsx")\n    if matches:\n        lib.create_directory("archive")\n        lib.move_files(matches, "archive")\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:45", + "generated": "2023-11-10 12:56:07", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -59372,7 +59372,7 @@ "name": "RPA.HTTP", "doc": "
\n

The RPA.HTTP library extends functionality of the RequestsLibrary.\nSee that documentation for several examples of how to issue GET\nrequests and utilize the returned result objects.

\n

This extension provides helper keywords to get an HTTP resource on a\nsession. The HTTP Get and Download keywords will initiate a\nsession if one does not exist for the provided URL, or use an existing\nsession. When using these keywords, you do not need to manage\nsessions with Create Session. Session management is still\nrequired if you use the underlying session keywords, e.g.,\n* On Session.

\n
\n", "version": "0.9.5", - "generated": "2023-11-10 12:47:45", + "generated": "2023-11-10 12:56:07", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -59543,10 +59543,10 @@ "name": "retry_method_list", "types": [], "typedocs": {}, - "defaultValue": "['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']", + "defaultValue": "['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "retry_method_list=['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']" + "repr": "retry_method_list=['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']" } ], "doc": "
\n

Create Session: create a HTTP session to a server

\n

url Base url of the server

\n

alias Robot Framework alias to identify the session

\n

headers Dictionary of default headers

\n

cookies Dictionary of cookies

\n

client_certs ['client certificate', 'client key'] PEM files containing the client key and certificate

\n

timeout Connection timeout

\n
\n
proxies Dictionary mapping protocol or protocol and host to the URL of the proxy
\n
(e.g. {'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'})
\n
verify Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.
\n
Defaults to False.
\n
debug Enable http verbosity option more information
\n
https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel
\n
max_retries Number of maximum retries each connection should attempt.
\n
By default it will retry 3 times in case of connection errors only.\nA 0 value will disable any kind of retries regardless of other retry settings.\nIn case the number of retries is reached a retry exception is raised.
\n
\n

disable_warnings Disable requests warning useful when you have large number of testcases

\n
\n
backoff_factor Introduces a delay time between retries that is longer after each retry.
\n
eg. if backoff_factor is set to 0.1\nthe sleep between attemps will be: 0.0, 0.2, 0.4\nMore info here: https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html
\n
retry_method_list List of uppercased HTTP method verbs where retries are allowed.
\n
By default retries are allowed only on HTTP requests methods that are considered to be\nidempotent (multiple requests with the same parameters end with the same state).\neg. set to ['POST', 'GET'] to retry only those kind of requests.
\n
retry_status_list List of integer HTTP status codes that, if returned, a retry is attempted.
\n
eg. set to [502, 503] to retry requests if those status are returned.\nNote that max_retries must be greater than 0.
\n
\n
\n", @@ -59679,10 +59679,10 @@ "name": "retry_method_list", "types": [], "typedocs": {}, - "defaultValue": "['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']", + "defaultValue": "['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "retry_method_list=['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']" + "repr": "retry_method_list=['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']" } ], "doc": "
\n

Create Session: create a HTTP session to a server

\n

url Base url of the server

\n

alias Robot Framework alias to identify the session

\n

headers Dictionary of default headers

\n

cookies Dictionary of cookies

\n
\n
auth A Custom Authentication object to be passed on to the requests library.
\n
http://docs.python-requests.org/en/master/user/advanced/#custom-authentication
\n
\n

timeout Connection timeout

\n
\n
proxies Dictionary mapping protocol or protocol and host to the URL of the proxy
\n
(e.g. {'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'})
\n
verify Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.
\n
Defaults to False.
\n
debug Enable http verbosity option more information
\n
https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel
\n
max_retries Number of maximum retries each connection should attempt.
\n
By default it will retry 3 times in case of connection errors only.\nA 0 value will disable any kind of retries regardless of other retry settings.\nIn case the number of retries is reached a retry exception is raised.
\n
\n

disable_warnings Disable requests warning useful when you have large number of testcases

\n
\n
backoff_factor Introduces a delay time between retries that is longer after each retry.
\n
eg. if backoff_factor is set to 0.1\nthe sleep between attemps will be: 0.0, 0.2, 0.4\nMore info here: https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html
\n
retry_method_list List of uppercased HTTP method verbs where retries are allowed.
\n
By default retries are allowed only on HTTP requests methods that are considered to be\nidempotent (multiple requests with the same parameters end with the same state).\neg. set to ['POST', 'GET'] to retry only those kind of requests.
\n
retry_status_list List of integer HTTP status codes that, if returned, a retry is attempted.
\n
eg. set to [502, 503] to retry requests if those status are returned.\nNote that max_retries must be greater than 0.
\n
\n
\n", @@ -59815,10 +59815,10 @@ "name": "retry_method_list", "types": [], "typedocs": {}, - "defaultValue": "['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']", + "defaultValue": "['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "retry_method_list=['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']" + "repr": "retry_method_list=['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']" } ], "doc": "
\n

Create Session: create a HTTP session to a server

\n

url Base url of the server

\n

alias Robot Framework alias to identify the session

\n

headers Dictionary of default headers

\n

cookies Dictionary of cookies

\n

auth ['DOMAIN', 'username', 'password'] for NTLM Authentication

\n

timeout Connection timeout

\n
\n
proxies Dictionary mapping protocol or protocol and host to the URL of the proxy
\n
(e.g. {'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'})
\n
verify Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.
\n
Defaults to False.
\n
debug Enable http verbosity option more information
\n
https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel
\n
max_retries Number of maximum retries each connection should attempt.
\n
By default it will retry 3 times in case of connection errors only.\nA 0 value will disable any kind of retries regardless of other retry settings.\nIn case the number of retries is reached a retry exception is raised.
\n
\n

disable_warnings Disable requests warning useful when you have large number of testcases

\n
\n
backoff_factor Introduces a delay time between retries that is longer after each retry.
\n
eg. if backoff_factor is set to 0.1\nthe sleep between attemps will be: 0.0, 0.2, 0.4\nMore info here: https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html
\n
retry_method_list List of uppercased HTTP method verbs where retries are allowed.
\n
By default retries are allowed only on HTTP requests methods that are considered to be\nidempotent (multiple requests with the same parameters end with the same state).\neg. set to ['POST', 'GET'] to retry only those kind of requests.
\n
retry_status_list List of integer HTTP status codes that, if returned, a retry is attempted.
\n
eg. set to [502, 503] to retry requests if those status are returned.\nNote that max_retries must be greater than 0.
\n
\n
\n", @@ -59951,10 +59951,10 @@ "name": "retry_method_list", "types": [], "typedocs": {}, - "defaultValue": "['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']", + "defaultValue": "['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "retry_method_list=['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']" + "repr": "retry_method_list=['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']" } ], "doc": "
\n

Create Session: create a HTTP session to a server

\n

url Base url of the server

\n

alias Robot Framework alias to identify the session

\n

headers Dictionary of default headers

\n

cookies Dictionary of cookies

\n

auth ['DOMAIN', 'username', 'password'] for NTLM Authentication

\n

timeout Connection timeout

\n
\n
proxies Dictionary mapping protocol or protocol and host to the URL of the proxy
\n
(e.g. {'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'})
\n
verify Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.
\n
Defaults to False.
\n
debug Enable http verbosity option more information
\n
https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel
\n
max_retries Number of maximum retries each connection should attempt.
\n
By default it will retry 3 times in case of connection errors only.\nA 0 value will disable any kind of retries regardless of other retry settings.\nIn case the number of retries is reached a retry exception is raised.
\n
\n

disable_warnings Disable requests warning useful when you have large number of testcases

\n
\n
backoff_factor Introduces a delay time between retries that is longer after each retry.
\n
eg. if backoff_factor is set to 0.1\nthe sleep between attemps will be: 0.0, 0.2, 0.4\nMore info here: https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html
\n
retry_method_list List of uppercased HTTP method verbs where retries are allowed.
\n
By default retries are allowed only on HTTP requests methods that are considered to be\nidempotent (multiple requests with the same parameters end with the same state).\neg. set to ['POST', 'GET'] to retry only those kind of requests.
\n
retry_status_list List of integer HTTP status codes that, if returned, a retry is attempted.
\n
eg. set to [502, 503] to retry requests if those status are returned.\nNote that max_retries must be greater than 0.
\n
\n
\n", @@ -60087,10 +60087,10 @@ "name": "retry_method_list", "types": [], "typedocs": {}, - "defaultValue": "['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']", + "defaultValue": "['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "retry_method_list=['OPTIONS', 'GET', 'PUT', 'HEAD', 'TRACE', 'DELETE']" + "repr": "retry_method_list=['TRACE', 'HEAD', 'OPTIONS', 'DELETE', 'GET', 'PUT']" } ], "doc": "
\n

Create Session: create a HTTP session to a server

\n

alias Robot Framework alias to identify the session

\n

url Base url of the server

\n

headers Dictionary of default headers

\n

cookies Dictionary of cookies

\n

auth List of username & password for HTTP Basic Auth

\n

timeout Connection timeout

\n
\n
proxies Dictionary mapping protocol or protocol and host to the URL of the proxy
\n
(e.g. {'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'})
\n
\n

verify Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.

\n
\n
debug Enable http verbosity option more information
\n
https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel
\n
max_retries Number of maximum retries each connection should attempt.
\n
By default it will retry 3 times in case of connection errors only.\nA 0 value will disable any kind of retries regardless of other retry settings.\nIn case the number of retries is reached a retry exception is raised.
\n
\n

disable_warnings Disable requests warning useful when you have large number of testcases

\n
\n
backoff_factor Introduces a delay time between retries that is longer after each retry.
\n
eg. if backoff_factor is set to 0.1\nthe sleep between attemps will be: 0.0, 0.2, 0.4\nMore info here: https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html
\n
retry_method_list List of uppercased HTTP method verbs where retries are allowed.
\n
By default retries are allowed only on HTTP requests methods that are considered to be\nidempotent (multiple requests with the same parameters end with the same state).\neg. set to ['POST', 'GET'] to retry only those kind of requests.
\n
retry_status_list List of integer HTTP status codes that, if returned, a retry is attempted.
\n
eg. set to [502, 503] to retry requests if those status are returned.\nNote that max_retries must be greater than 0.
\n
\n
\n", @@ -61469,7 +61469,7 @@ "name": "RPA.Hubspot", "doc": "
\n

Hubspot is a library for accessing HubSpot using REST API. It\nextends hubspot-api-client.

\n

Current features of this library focus on retrieving CRM object data\nfrom HubSpot via API. For additional information, see\nUnderstanding the CRM.

\n
\n

Using Date Times When Searching

\n

When using date times with the Hubspot API, you must provide\nthem as Unix-style epoch timestamps (with milliseconds), which can be obtained\nusing the DateTime library's Convert Date with the\nargument result_format=epoch. The resulting timestamp string\nwill be a float, but the API only accepts integers, so you must\nmultiply the resulting timestamp by 1,000 and then round it to\nthe nearest integar to include in API calls (i.e., the resulting\ninteger sent to the API must have 13 digits as of March 18, 2022).

\n

Robot framework example usage:

\n
\n*** Settings ***\nLibrary     DateTime\nLibrary     RPA.Hubspot\nTask Setup  Authorize Hubspot\n\n*** Tasks ***\nSearch with date\n    ${yesterday}=    Get current date    increment=-24h   result_format=epoch\n    ${yesterday_hs_ts}=    Evaluate    round(${yesterday} * 1000)\n    ${deals}=    Search for objects    DEALS\n    ...    hs_lastmodifieddate    GTE    ${yesterday_hs_ts}\n
\n

Python example usage

\n
\nfrom robot.libraries.DateTime import get_current_date, subtract_time_from_date\nfrom RPA.Hubspot import Hubspot\nfrom RPA.Robocorp.Vault import Vault\n\nsecrets = Vault().get_secret("hubspot")\n\nhs = Hubspot(hubspot_apikey=secrets["api_key"])\nyesterday = round(\n    subtract_time_from_date(get_current_date(), "24h", result_format="epoch") * 1000\n)\ndeals = hs.search_for_objects("DEALS", "hs_lastmodifieddate", "GTE", yesterday)\nprint(deals)\n
\n
\n
\n

Batch Inputs

\n

When retrieving information, the library automatically batches requests\nthat are provided as lists, see Get object keyword for an example,\nbut when wishing to create or update many objects, the library provides\na batching system.

\n

In order to start a batch, you must first call the Create new batch\nkeyword. This initializes a new batch to accept inputs. If a batch\nalready exists when you call this keyword, it will be lost and a new\nblank one will be started.

\n

Once the batch has been initialized, you can add inputs one at a time with\nAdd input to batch or many at a time with Extend batch with inputs.

\n

In order to finally send the batch to HubSpot, you must call\nExecute batch. The final keyword will return the created or updated\nobjects from HubSpot. New object IDs can be obtained from the id\nproperty, see the SimplePublicObject reference.

\n

Robot framework example:

\n
\n*** Settings ***\nLibrary         RPA.Hubspot\nLibrary         RPA.Robocorp.Vault\nTask Setup      Authorize Hubspot\n\n*** Tasks ***\nCreate objects via batch\n    Create new batch\n    Add input to batch    name=Nokia    country=Finland\n    Add input to batch    name=Google    country=USA\n    ${new_companies}=    Execute batch\n    Log    The first new company added has the new id ${{$new_companies[0].id}}\n\n*** Keywords ***\nAuthorize Hubspot\n    ${secrets}=    Get secret    hubspot\n    Auth with api key    ${secrets}[API_KEY]\n
\n

Python example:

\n

NOTE: When executing a batch input in Python, you can directly import the\nBatchInputFactory class to use to create your batch input before\nexecuting the batch.

\n
\nfrom RPA.Hubspot import Hubspot, BatchInputFactory, BatchMode\nfrom RPA.Robocorp.Vault import RobocorpVault\n\n\nvault = RobocorpVault()\nsecrets = vault.get_secret("hubspot")\n\nhs = Hubspot(secrets["API_KEY"])\n\nbatch = BatchInputFactory(BatchMode.UPDATE, "company")\nbatch.extend_inputs(\n    [\n        {"name": "Nokia's New Name", "city": "Espoo"},\n        {"name": "Alphabet", "city": "Mountain View"},\n    ],\n    ["1001", "1002"],\n)\nhs.batch_input = batch\nupdated_companies = hs.execute_batch()\nprint(\n    "Companies have been updated:\\\\n" +\n    "\\\\n".join([str(c) for c in updated_companies])\n)\n
\n
\n
\n

Information Caching

\n

This library loads custom object schemas and pipelines into memory\nthe first time when keywords using them are called. These cached versions\nare recalled unless the use_cache is set to False, where available.

\n
\n
\n

Custom Object Types

\n

All keywords that request a parameter of object_type can accept\ncustom object type names as long as they are properly configured in\nHubSpot. The system will lookup the custom object ID using the\nprovided name against the configured name or one of the configured\nlabels (e.g., "singular" and "plural" types of the name).

\n
\n
\n

HubSpot Object Reference

\n

This section describes the types of objects returned by this Library\nand their associated attributes. These attributes can be accessed via\ndot-notation as described in the Attribute Access section below.

\n
\n

Attribute Access

\n

Keywords return native Python Hubspot objects, rather than common Robot\nFramework types. These types have sets of defined attributes allowing\nfor dot-notation access of object properties. Properties (e.g.,\nthose configured in Hubspot settings for each object) will be\naccessible in a Python dictionary attached to the properties attribute\nof the returned object. See the Attribute Definitions section for\ndetails of that associated attributes for all types returned by this\nlibrary.

\n

Example usage retrieving the city property of a Company object:

\n

Robot framework example:

\n
\n*** Settings ***\nLibrary         RPA.Hubspot\nLibrary         RPA.Robocorp.Vault\n\nTask Setup      Authorize Hubspot\n\n*** Variables ***\n${ACCOUNT_NOKIA}    6818764598\n\n*** Tasks ***\nObtain city information from Hubspot\n    ${account}=    Get object    COMPANY    ${ACCOUNT_NOKIA}\n    Log    The city for account number ${ACCOUNT_NOKIA} is ${account.properties}[city]\n\n*** Keywords ***\nAuthorize Hubspot\n    ${secrets}=    Get secret    hubspot\n    Auth with api key    ${secrets}[API_KEY]\n
\n

Python example:

\n
\nfrom RPA.Hubspot import Hubspot\nfrom RPA.Robocorp.Vault import RobocorpVault\n\nvault = RobocorpVault()\nsecrets = vault.get_secret("hubspot")\n\nhs = Hubspot(secrets["API_KEY"])\nnokia_account_id = "6818764598"\naccount = hs.get_object("COMPANY", nokia_account_id)\nprint(f"The city for account number {nokia_account_id} is {account.properties['city']}")\n
\n
\n
\n

Attribute Definitions

\n

This library can return various types of objects, whose attributes\nare only accessible via dot-notation. The below reference describes\nthe attributes available on these objects.

\n
\n

SimplePublicObject

\n

An object in HubSpot. The object itself does not describe what type\nit represents.

\n
\n
id : str
\n
The HubSpot ID of the object.
\n
properties : Dict[str, str]
\n
A dictionary representing all returned properties associated\nto this object. Properties must be accessed as via standard\ndictionary subscription, e.g., properties["name"].
\n
created_at : datetime
\n
The timestamp when this object was created in HubSpot.
\n
updated_at : datetime
\n
The last modified timestamp for this object.
\n
archived : bool
\n
Whether this object is archived.
\n
archived_at : datetime
\n
The timestamp when this object was archived.
\n
\n
\n
\n

SimplePublicObjectWithAssociations

\n

An object in HubSpot including associations to other objects. The\nobject itself does not describe what type it represents.

\n
\n
id : str
\n
The HubSpot ID of the object.
\n
properties : Dict[str, str]
\n
A dictionary representing all returned properties associated\nto this object. Properties must be accessed as via standard\ndictionary subscription, e.g., properties["name"].
\n
created_at : datetime
\n
The timestamp when this object was created in HubSpot.
\n
updated_at : datetime
\n
The last modified timestamp for this object.
\n
archived : bool
\n
Whether this object is archived.
\n
archived_at : datetime
\n
The timestamp when this object was archived.
\n
associations : Dict[str, CollectionResponseAssociatedId]
\n
A dictionary whose key will be the requested association type, e.g.,\ncompanies and associated value will be a container object\nwith all the associations. See CollectionResponseAssociatedId.
\n
\n
\n
\n

AssociatedId

\n

The ID of an associated object, as well as the type of association.

\n
\n
id : str
\n
The ID of the associated HubSpot object.
\n
type : str
\n
The type of association, e.g., deals_to_companies.
\n
\n
\n
\n

CollectionResponseAssociatedId

\n

A container object for a collection of AssociatedId objects returned\nby the API.

\n
\n
results : List[AssociatedId]
\n
The list of AssociatedId objects returned by the API.
\n
paging : Paging
\n
Used by this library to assist with retreiving multi-page\nAPI responses.
\n
\n
\n
\n

Pipeline

\n

A pipeline represents the steps objects travel through within HubSpot.

\n
\n
id : str
\n
The HubSpot ID for the pipeline. All accounts start with one\npipeline with the id default.
\n
label : str
\n
The human-readabel label for the pipeline.
\n
stages : List[PipelineStage]
\n
A list of PipelineStage objects in the order the object would\nfollow through the pipeline.
\n
created_at : datetime
\n
The timestamp when this pipeline was created in HubSpot.
\n
updated_at : datetime
\n
The last modified timestamp for this pipeline.
\n
archived : bool
\n
Whether this pipeline is archived.
\n
display_order : int
\n
The place in the list of pipelines where this pipeline is shown\nin the HubSpot UI.
\n
\n
\n
\n

PipelineStage

\n

A pipeline stage is one of the various stages defined in a Pipeline.

\n
\n
id : str
\n
The HubSpot ID of the stage.
\n
label : str
\n
The human-readabel label for the stage.
\n
metadata : Dict[str, str]
\n
A dictionary of additional data associated with ths stage, such\nas probability.
\n
created_at : datetime
\n
The timestamp when this stage was created in HubSpot.
\n
updated_at : datetime
\n
The last modified timestamp for this stage.
\n
archived : bool
\n
Whether this stage is archived.
\n
archived_at : datetime
\n
The timestamp when this stage was archived.
\n
\n
\n
\n

PublicOwner

\n

An owner in HubSpot. Owners of companies and deals are responsible\nfor driving a sale to close or similar.

\n
\n
id : str
\n
The HubSpot ID of the owner.
\n
email : str
\n
The owner's email address in HubSpot.
\n
first_name : str
\n
The owner's first name.
\n
last_name : str
\n
The owner's last name.
\n
user_id : int
\n
The associated user ID if the owner is a HubSpot user.
\n
created_at : datetime
\n
The timestamp when this owner was created in HubSpot.
\n
updated_at : datetime
\n
The last modified timestamp for this owner.
\n
archived : bool
\n
Whether this owner is archived.
\n
teams : List[PublicTeam]
\n
A list of teams the owner is in. See PublicTeam.
\n
\n
\n
\n

PublicTeam

\n

A team of owners in HubSpot

\n
\n
id : str
\n
The HubSpot ID of the Team.
\n
name : str
\n
The Team's name.
\n
membership : str
\n
One of PRIMARY, SECONDARY, or CHILD.
\n
\n
\n
\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:47", + "generated": "2023-11-10 12:56:08", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -62645,7 +62645,7 @@ "name": "RPA.Images", "doc": "
\n

Images is a library for general image manipulation.\nFor image-based desktop automation, use the RPA.Desktop library.

\n

Coordinates

\n

The coordinates used in the library are pairs of x and y values that\nrepresent pixels. The upper left corner of the image or screen\nis (0, 0). The x-coordinate increases towards the right, and the y-coordinate\nincreases towards the bottom.

\n

Regions are represented as tuples of (left, top, right, bottom). For example,\na 400 by 200-pixel region in the upper left corner would be (0, 0, 400, 200).

\n

Template matching

\n

Template matching refers to an operation where the (potential) location of\na smaller image is searched from a larger image. It can be used for verifying\ncertain conditions or locating UI elements for desktop or web automation.

\n

Requirements

\n

The default installation depends on Pillow\nlibrary, which is used for general image manipulation operations.

\n

For more robust and faster template matching, the library can use a combination\nof NumPy and OpenCV.\nThey can be installed by opting in to the recognition dependency:

\n

pip install rpaframework rpaframework-recognition

\n

Examples

\n

Robot Framework

\n

The Images library can be imported and used directly in Robot Framework,\nfor instance, for capturing screenshots or verifying something on the screen.

\n

Desktop automation based on images should be done using the corresponding\ndesktop library, i.e. RPA.Desktop.

\n
\n*** Settings ***\nLibrary    RPA.Images\n\n*** Keywords ***\nShould show success\n    [Documentation]    Raises ImageNotFoundError if success image is not on screen\n    Find template on screen    ${CURDIR}${/}success.png\n\nSave screenshot to results\n    [Documentation]    Saves screenshot of desktop with unique name\n    ${timestamp}=      Get current date    result_format=%H%M%S\n    Take screenshot    filename=${OUTPUT_DIR}${/}desktop_${timestamp}.png\n
\n

Python

\n
\nfrom RPA.Images import Images\n\ndef draw_matches_on_image(source, template):\n    matches = lib.find_template_in_image(source, template)\n    for match in matches:\n        lib.show_region_in_image(source, match)\n\n    source.save("matches.png")\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:47", + "generated": "2023-11-10 12:56:08", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -62832,7 +62832,7 @@ "name": "RPA.JSON", "doc": "
\n

JSON is a library for manipulating JSON files and strings.

\n

JSON is a common data interchange format inspired by a subset of\nthe Javascript programming language, but these days is a de facto\nstandard in modern web APIs and is language agnostic.

\n
\n

Serialization

\n

The term serialization refers to the process of converting\nRobot Framework or Python types to JSON or the other way around.

\n

Basic types can be easily converted between the domains,\nand the mapping is as follows:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
JSONPython
objectdict
arraylist
stringstr
number (int)int
number (real)float
trueTrue
falseFalse
nullNone
\n
\n
\n

About JSONPath

\n

Reading and writing values from/to JSON serializable objects is done\nusing JSONPath. It's a syntax designed to quickly and easily refer to\nspecific elements in a JSON structure. The specific flavor used in this\nlibrary is based on jsonpath-ng.

\n

Compared to Python's normal dictionary access, JSONPath expressions can\ntarget multiple elements through features such as conditionals and wildcards,\nwhich can simplify many JSON-related operations. It's analogous to XPath\nfor XML structures.

\n
\n

Syntax example

\n

For this example consider the following structure:

\n
\n{\n  "clients": [\n    {\n      "name": "Johnny Example",\n      "email": "john@example.com",\n      "orders": [\n          {"address": "Streetroad 123", "price": 103.20},\n          {"address": "Streetroad 123", "price": 98.99}\n      ]\n    },\n    {\n      "name": "Jane Example",\n      "email": "jane@example.com",\n      "orders": [\n          {"address": "Waypath 321", "price": 22.00},\n          {"address": "Streetroad 123", "price": 2330.01}\n      ]\n    }\n  ]\n}\n
\n

In the simplest case JSONPath can replace nested access:

\n
\n*** Tasks ***\nNested access\n    # First order of first client, with direct dictionary access\n    ${value}=    Set variable    ${json}["clients"][0]["orders"][0]\n\n    # JSONPath access\n    ${value}=    Get value from JSON    ${json}    $.clients[0].orders[0]\n
\n

But the power comes from complicated expressions:

\n
\n*** Tasks ***\nComplicated expressions\n    # Find delivery addresses for all orders\n    ${prices}=        Get values from JSON    $..address\n\n    # Find orders that cost over 100\n    ${expensives}=    Get values from JSON    $..orders[?(@.price>100)]\n
\n
\n
\n

Supported Expressions

\n

The supported syntax elements are:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
ElementDescription
$Root object/element
@Current object/element inside expressions
. or []Child operator
..Recursive descendant operator
`parent`Parent operator, see functions
*Wilcard, any element
,Select multiple fields
[n]Array index
[a:b:c]Array slice (start, end, step)
[a,b]Union of indices or names
[?()]Apply a filter expression
()Script expression
[\\\\field]Sort descending by field, cannot be combined with\nfilters.
[/field]Sort ascending by field, cannot be combined with\nfilters.
`str()`Convert value to string, see functions
`sub()`Regex substitution function, see functions
`len`Calculate value's length, see functions
`split()`String split function, see functions
+ - * /Arithmetic functions, see functions
\n
\n

Functions

\n

This library allows JSON path expressions to include certain functions\nwhich can provide additional benefit to users. These functions are\ngenerally encapsulated in backticks (`). Some functions require\nyou to pass arguments similar to a Python function.

\n

For example, let's say a JSON has nodes on the JSON path\n$.books[*].genres which are represented as strings of genres with\ncommas separating each genre. So for one book, this node might have a\nvalue like horror,young-adult. You can return a list of first genre\nfor each book by using the split function like so:

\n
\n*** Task ***\nGet genres\n    ${genres}=  Get values from JSON    $.books[*].genres.`split(,, 0, -1)`\n
\n

Each functions parameters are defined here:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
FunctionUsage
str()No parameters, but parenthesis are required
sub(/regex/, repl)The regex pattern must be provided in regex\nand the replacement value provided in repl
lenNo parameters and no parenthesis
split(char, segment, max_split)Separator character provided as char, which\nindex from the resulting array to be returns\nprovided as segment, and maximum number of\nsplits to perform provided as max_split,\n-1 for all splits.
parentNo parameters, no parenthesis
\n

Arithmetic Functions

\n

JSON Path can be written and combined to concatenate string values\nor perform arithmetic functions on numerical values. Each JSONPath\nexpression used must return the same type, and when performing\nsuch functions between returned lists, each list must be the same\nlength. An example is included in documentation for the keyword\nGet values from JSON.

\n
\n
\n

Additional Information

\n

There are a multitude of different script expressions\nin addition to the elements listed above, which can\nbe seen in the aforementioned article.

\n

For further library usage examples, see the individual keywords.

\n
\n
\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:47", + "generated": "2023-11-10 12:56:08", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -63488,7 +63488,7 @@ "name": "RPA.JavaAccessBridge", "doc": "
\n

Java application UI automation library using Java Access Bridge technology.

\n

The library utilizes java-access-bridge-wrapper package to interact with\nJava UI. Currently only the 64-bit Windows OS is supported.

\n

Inspecting elements

\n

We have built an Assistant for working with Java application's element structure and Java locators.\nThe Assistant provides copy-paste-able locators for each element and also allows testing locators against\nselected application.

\n

If our tools fail to pick the locator from your target application, there is always the\nAccess Bridge Explorer from Google that enables you to see the raw view. Please note that\nAccess Bridge Explorer repository has been archived on July 27, 2022 and is no longer actively\nmaintained.

\n

The Accessibility Insights for Windows can show element properties if application framework\nsupports Windows UI Automation (UIA), see more at using Accessibility Insights. Then the recommended\nlibrary would be RPA.Windows library.

\n

Steps to enable

\n
\n
    \n
  1. Enable the Java Access Bridge in Windows
  2. \n
  3. Set environment variable RC_JAVA_ACCESS_BRIDGE_DLL as an absolute path to WindowsAccessBridge-64.dll.\nIt is also possible to give DLL location as library initialization parameter access_bridge_path.
  4. \n
\n
\nC:\\path\\to\\java\\bin\\jabswitch -enable\nset RC_JAVA_ACCESS_BRIDGE_DLL=C:\\path\\to\\Java\\bin\\WindowsAccessBridge-64.dll\n
\n
\n*** Settings ***\nLibrary   RPA.JavaAccessBridge   access_bridge_path=C:\\path\\to\\Java\\bin\\WindowsAccessBridge-64.dll\n
\n
\n

About Java wrapper callbacks and actions

\n

There might be a compatibility issue with callbacks and actions on target Java application. Possible reasons:

\n
    \n
  • target application is executed with 32-bit Java
  • \n
  • target application does not support callbacks and/or actions
  • \n
\n

Workaround for this situation is to initialize JavaAccessBridge library with parameter ignore_callbacks=True.\nThen application's element information is still accessible and any actions on those elements can be performed\nwith RPA.Desktop library. Keep in mind that you can still manuall refresh an element with Refresh Element.

\n

Note. There are still keywords, for example. Call Element Action, which will cause error if used in this\nsituation.

\n
\n*** Settings ***\nLibrary   RPA.JavaAccessBridge   ignore_callbacks=True\n
\n

Controlling the Java window

\n

Keyword for this purpose is Select Window. Window selection is based on the title parameter, which can be\ngiven as a regular expressions to match the correct window. The keyword brings the window into focus and initially\nreads window's element structure.

\n

Locating elements

\n

To automate actions on the Java application, the robot needs locations to various elements\nusing a feature called Java locators. Locator describes properties of an element.

\n

At the moment library contains basic level support for locators.

\n

The common locator types are name and role.

\n

To identify element with more than one property and can be used, for example:

\n
\n
\nrole:push button and name:Clear\n
\n
\n

To address element within parent element > can be used, for example:

\n
\n
\nname:Find Purchase Orders > name:NumberField\n
\n
\n

Some keywords accept element as an parameter in place of locator.

\n

New locator type strict has been added in rpaframework==12.5.0. Currently\nproperty values of string type have been evaluated with startsWith which\ncan match several property values. With strict set in the locator string,\nall locator on the right side of this definition will be matched using\nstrict (equal matching), example:

\n
\n
\n# without strict, name can be 'Type', 'Type1', 'Type of'...\nGet Elements   role:push button and name:Type\n# name must be equal to 'Type'\nGet Elements  role:push button and strict:True and name:Type\n
\n
\n

Keyword Get Elements has extra parameter strict, which when set to\nTrue forces all locator value matches to be strict, example:

\n
\n
\n# without strict, name can be 'Type', 'Type1', 'Type of'...\nGet Elements  role:push button and name:Type\n# name must be equal to 'Type' and role must be equal to 'text'\nGet Elements  role:text and name:Type  strict=True\n
\n
\n

About JavaElement object

\n

The JavaElement was added in rpaframework==12.3.0 for easy access into\nContextNode objects which have been returned by Get Elements keyword.

\n

Keyword Get Elements still returns ContextNode objects, but with parameter\njava_elements=True the keyword returns JavaElement objects instead (they\nstill contain reference to ContextNode object via node property, e.g.\nJavaObject.node).

\n

Properties and methods included in the JavaElement:

\n
    \n
  • name: str
  • \n
  • role: str
  • \n
  • description: str
  • \n
  • states: list # list presentation of states (string)
  • \n
  • ancestry: int # you can set the maximum depth based on this
  • \n
  • checked: bool
  • \n
  • selected: bool
  • \n
  • visible: bool
  • \n
  • enabled: bool
  • \n
  • showing: bool
  • \n
  • focusable: bool
  • \n
  • states_string: str
  • \n
  • x: int # left coordinate of the element
  • \n
  • y: int # top coordinate of the element
  • \n
  • width: int
  • \n
  • height: int
  • \n
  • node: ContextNode # original ContextNode
  • \n
  • row: int # table row, -1 if element is not member of table
  • \n
  • col: int # table column, -1 if element is not member of table
  • \n
  • text: str # text content of the element
  • \n
  • column_count: int # table column count
  • \n
  • visible_children: list # visible children elements of this element
  • \n
  • visible_children_count: int
  • \n
  • index_in_parent: int # position in the parent
  • \n
  • click() # method for clicking element center
  • \n
  • type_text() # method for typing text into element (if possible)
  • \n
\n

Interacting with elements

\n

By default application elements are interacted with Actions supported by the element.\nMost common example is click action supported by an button element.

\n

But because application and technology support for the actions might be limited, it is also\npossible to opt for interaction elements by their coordinates by giving keyword parameter\naction=False if parameter is available.

\n

Examples

\n

robotframework

\n
\n*** Settings ***\nLibrary   RPA.JavaAccessBridge\nLibrary   Process\n\n*** Tasks ***\nWrite text into Swing application\n    Start Process    java -jar BasicSwing.jar\n    ...              shell=${TRUE}\n    ...              cwd=${CURDIR}\n    Select Window    Chat Frame\n    Type Text    role:text\n    ...          text for the textarea\n    Type Text    role:text\n    ...          text for the input field\n    ...          index=1\n    ...          clear=${TRUE}\n    Click Element    role:push button and name:Send\n
\n

Python

\n
\nfrom RPA.JavaAccessBridge import JavaAccessBridge\nimport subprocess\n\njab = JavaAccessBridge()\n\nsubprocess.Popen(\n    ["java", "-jar", "BasicSwing.jar"],\n    shell=True,\n    cwd=".",\n    close_fds=True\n)\njab.select_window("Chat Frame")\njab.type_text(\n    "role:text",\n    "text for the textarea",\n    enter=True\n)\njab.type_text(\n    "role:text",\n    "text for the input field",\n    index=1,\n    clear=True\n)\njab.click_element("role:push button and name:Send")\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:47", + "generated": "2023-11-10 12:56:08", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -64802,7 +64802,7 @@ "name": "RPA.MFA", "doc": "
\n

RPA.MFA is a library intended mainly for generating one-time passwords (OTP)\nand not only, as OAuth2 support was introduced lately.

\n

Library requires at the minimum rpaframework version 19.4.0.

\n

Based on the pyotp and\nrequests_oauthlib packages. It\nprovides support for both MFA with the * OTP related keywords and OAuth2\n"Authorization Code Flow" with the * OAuth * related keywords.

\n

In the below example the mfa secret we are reading from the Robocorp\nVault is the passcode generated by the Authenticator service. The passcode\nvalue is stored into the Vault with key otpsecret.

\n

Passcode is typically a long string (16-32 characters), which is provided\nin a form of QR image, but it can be obtained by requesting access to a string.

\n

Note that same code can be used to add a mobile phone as a duplicate authentication\ndevice at the same time when the same code is added into the Vault.

\n

Robot framework example usage:

\n
\n*** Settings ***\nLibrary     RPA.MFA\nLibrary     RPA.Robocorp.Vault\n\n*** Tasks ***\nGenerate time based code\n    ${secrets}=    Get Secret   mfa\n    ${code}=    Get Time Based OTP    ${secrets}[otpsecret]\n
\n

Python example usage

\n
\nfrom RPA.MFA import MFA\nfrom RPA.Robocorp.Vault import Vault\n\n\ndef main():\n    secrets = Vault().get_secret("mfa")\n    code = MFA().get_time_based_otp(secrets["otpsecret"])\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:47", + "generated": "2023-11-10 12:56:09", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -65352,7 +65352,7 @@ "name": "RPA.MSGraph", "doc": "
\n

The MSGraph library wraps the O365 package, giving robots\nthe ability to access the Microsoft Graph API programmatically.

\n

OAuth Configuration

\n

Graph's API primarily authenticates via the OAuth 2.0 authorization code grant\nflow or OpenID Connect. This library exposes the OAuth 2.0 flow for robots to\nauthenticate on behalf of users. A user must complete an initial authentication\nflow with the help of our OAuth Graph Example Bot.

\n

For best results, register an app in Azure AD and configure it as so:

\n
    \n
  • The type is "Web App".
  • \n
  • Redirect URI should be https://login.microsoftonline.com/common/oauth2/nativeclient
  • \n
  • The app should be a multi-tenant app.
  • \n
  • Accounts in any organizational directory is checked.
  • \n
  • Has relevant permissions enabled, check the Microsoft Graph permissions reference\nfor a list of permissions available to MS Graph apps.
  • \n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:48", + "generated": "2023-11-10 12:56:09", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -66699,7 +66699,7 @@ "name": "RPA.Netsuite", "doc": "
\n

Netsuite is a library for accessing Netsuite using NetSuite SOAP web service SuiteTalk.\nThe library extends the netsuitesdk library.

\n

More information available at NetSuite SOAP webservice SuiteTalk.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary     RPA.Netsuite\nLibrary     RPA.Excel.Files\nLibrary     RPA.Tables\nTask Setup  Authorize Netsuite\n\n*** Tasks ***\nGet data from Netsuite and Store into Excel files\n    ${accounts}=        Get Accounts   account_type=_expense\n    ${accounts}=        Create table    ${accounts}\n    Create Workbook\n    Append Rows To Worksheet  ${accounts}\n    Save Workbook       netsuite_accounts.xlsx\n    Close Workbook\n    ${bills}=           Get Vendor Bills\n    ${bills}=           Create table    ${bills}\n    Create Workbook\n    Append Rows To Worksheet  ${bills}\n    Save Workbook       netsuite_bills.xlsx\n    Close Workbook\n\n\n*** Keywords ***\nAuthorize Netsuite\n    ${secrets}=     Get Secret   netsuite\n    Connect\n    ...        account=${secrets}[ACCOUNT]\n    ...        consumer_key=${secrets}[CONSUMER_KEY]\n    ...        consumer_secret=${secrets}[CONSUMER_KEY]\n    ...        token_key=${secrets}[CONSUMER_SECRET]\n    ...        token_secret=${secrets}[TOKEN_KEY]\n
\n

Python

\n
\nfrom RPA.Netsuite import Netsuite\n\nns = Netsuite()\nns.connect()\naccounts = ns.get_accounts()\ncurrencies = ns.get_currencies()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:48", + "generated": "2023-11-10 12:56:09", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -67248,7 +67248,7 @@ "name": "RPA.Notifier", "doc": "
\n

Notifier is a library interfacting with different notification providers.

\n

Supported providers

\n
    \n
  • email
  • \n
  • gmail
  • \n
  • pushover
  • \n
  • slack
  • \n
  • telegram
  • \n
  • twilio
  • \n
\n

Providers not supported yet via specific keywords

\n
    \n
  • gitter
  • \n
  • join
  • \n
  • mailgun
  • \n
  • pagerduty
  • \n
  • popcornnotify
  • \n
  • pushbullet
  • \n
  • simplepush
  • \n
  • statuspage
  • \n
  • zulip
  • \n
\n

There is a keyword Generic Notify which can be used\nto call above services, for example.

\n
\nGeneric Notify\n    provider_name=gitter\n    message=Hello from Robot\n    token=TOKEN\n    room_id=ID_OF_THE_GITTER_ROOM\n
\n

Parameters for different providers can be read from the\nNotifiers documents (link below).

\n

Read more at https://notifiers.readthedocs.io/en/latest/

\n

About kwargs

\n

The **kwargs is a term for any extra named parameters, which\ncan be included in the same way as already named arguments,\ne.g. Notify Email could be called with subject=my email subject\nwhich will be passed through **kwargs.

\n

Notifier documentation contains information about all possible\narguments that different providers support.

\n

Robot Framework

\n
\n&{account}=    Create Dictionary\n...    host=smtp.office365.com\n...    username=ACCOUNT_USERNAME\n...    password=ACCOUNT_PASSWORD\nNotify Email\n...    to=RECIPIENT_EMAIL\n...    from_=SENDER_ADDRESS            # passed via kwargs\n...    subject=Hello from the Robot    # passed via kwargs\n...    message=Hello from the Robot\n...    &{account}                      # passed via kwargs\n
\n
\nnotifier = Notifier()\naccount = {\n    "host": "smtp.office365.com",\n    "username": "EMAIL_USERNAME",\n    "password": "EMAIL_PASSWORD"\n}\nnotifier.email_notify(\n    to="RECIPIENT_EMAIL",\n    from_="SENDER_EMAIL",\n    subject="Hello from the Python Robot",\n    message="Hello from the Python RObot",\n    **account\n)\n
\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary  RPA.Notifier\n\n*** Variables ***\n${SLACK_WEBHOOK}   https://hooks.slack.com/services/WEBHOOKDETAILS\n${CHANNEL}         notification-channel\n\n*** Tasks ***\nLets notify\n   Notify Slack   message from robot  channel=${CHANNEL}  webhook_url=${SLACK_WEBHOOK}\n
\n

Python

\n
\nfrom RPA.Notifier import Notifier\n\nlibrary = Notifier()\n\nslack_attachments = [\n   {\n      "title": "attachment 1",\n      "fallback": "liverpool logo",\n      "image_url": "https://upload.wikimedia.org/wikipedia/fi/thumb/c/cd/Liverpool_FC-n_logo.svg/1200px-Liverpool_FC-n_logo.svg.png",\n   }\n]\n\nlibrary.notify_slack(\n   message='message for the Slack',\n   channel="notification-channel",\n   webhook_url=slack_webhook_url,\n   attachments=slack_attachments,\n)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:48", + "generated": "2023-11-10 12:56:09", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -67846,7 +67846,7 @@ "name": "RPA.OpenAI", "doc": "
\n

Library to support OpenAI and Azure OpenAI services.

\n

Library is not included in the rpaframework package, so in order to use it\nyou have to add rpaframework-openai with the desired version in your\nconda.yaml file.

\n

Robot Framework example usage

\n
\n*** Settings ***\nLibrary    RPA.Robocorp.Vault\nLibrary    RPA.OpenAI\n\n*** Tasks ***\nCreate a text completion\n    ${secrets}   Get Secret   secret_name=OpenAI\n    Authorize To OpenAI   api_key=${secrets}[key]\n    ${completion}    Completion Create\n    ...     Write a tagline for an ice cream shop\n    ...     temperature=0.6\n    Log   ${completion}\n
\n

Python example usage

\n
\nfrom RPA.Robocorp.Vault import Vault\nfrom RPA.OpenAI import OpenAI\n\nsecrets = Vault().get_secret("OpenAI")\nbaselib = OpenAI()\nbaselib.authorize_to_openai(secrets["key"])\n\nresult = baselib.completion_create(\n    Create a tagline for icecream shop',\n    temperature=0.6,\n)\nprint(result)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:48", + "generated": "2023-11-10 12:56:09", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -68409,7 +68409,7 @@ "name": "RPA.Outlook.Application", "doc": "
\n

Outlook.Application is a library for controlling the Outlook application.

\n

About Email Filtering

\n

Emails can be filtered according to specification set by Restrict method of the\nItem class https://docs.microsoft.com/en-us/office/vba/api/outlook.items.restrict.

\n

Couple of examples:

\n
\nGet Emails\n...   email_filter=[Subject]='test email'\n\nMove Emails\n...   email_filter=[SenderEmailAddress]='hello@gmail.com'\n
\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary                 RPA.Outlook.Application\nTask Setup              Open Application\nSuite Teardown          Quit Application\n\n*** Variables ***\n${RECIPIENT}            address@domain.com\n\n*** Tasks ***\nSend message\n    Send Message       recipients=${RECIPIENT}\n    ...                subject=This is the subject\n    ...                body=This is the message body\n    ..                 attachments=approved.png\n
\n

Python

\n
\nfrom RPA.Outlook.Application import Application\n\ndef send_message():\n    app = Application()\n    app.open_application()\n    app.send_message(\n        recipients='EMAILADDRESS_1, EMAILADDRESS_2',\n        subject='email subject',\n        body='email body message',\n        attachments='../orders.csv'\n
\n

For more information, see: https://docs.microsoft.com/en-us/previous-versions/office/developer/office-2007/bb219950(v=office.12)

\n

Caveats

\n

This library works on a Windows operating system with UI enabled only, and you must\nensure that you open the app first with Open Application before running any\nother relevant keyword which requires to operate on an open app. The application is\nautomatically closed at the end of the task execution, so this can be changed by\nimporting the library with the autoexit=${False} setting.

\n
\n*** Settings ***\nLibrary     RPA.Excel|Outlook|Word.Application    autoexit=${False}\n
\n

If you're running the Process by Control Room through a custom self-hosted Worker\nservice, then please make sure that you enable an RDP session by ticking "Use\nDesktop Connection" under the Step configuration.

\n

If you still encounter issues with opening a document, please ensure that file can\nbe opened first manually and dismiss any alert potentially blocking the process.

\n

Check the documentation below for more info:

\n\n
\n", "version": "", - "generated": "2023-11-10 12:47:49", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -69078,7 +69078,7 @@ "name": "RPA.PDF", "doc": "
\n

PDF is a library for managing PDF documents.

\n

It can be used to extract text from PDFs, add watermarks to pages, and\ndecrypt/encrypt documents.

\n

Merging and splitting PDFs is supported by Add Files To PDF keyword. Read\nthe keyword documentation for examples.

\n

There is also limited support for updating form field values. (check\nSet Field Value and Save Field Values for more info)

\n

The input PDF file can be passed as an argument to the keywords, or it can be\nomitted if you first call Open PDF. A reference to the current active PDF will\nbe stored in the library instance and can be changed by using the Switch To PDF\nkeyword with another PDF file path, therefore you can asynchronously work with\nmultiple PDFs.

\n
\n

Attention!

\n

Keep in mind that this library works with text-based PDFs, and it can't\nextract information from an image-based (scan) PDF file. For accurate\nresults, you have to use specialized external services wrapped by the\nRPA.DocumentAI library.

\n
\n

Portal example with video recording demo for parsing PDF invoices:\nhttps://github.com/robocorp/example-parse-pdf-invoice

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary    RPA.PDF\nLibrary    String\n\n*** Tasks ***\nExtract Data From First Page\n    ${text} =    Get Text From PDF    report.pdf\n    ${lines} =     Get Lines Matching Regexp    ${text}[${1}]    .+pain.+\n    Log    ${lines}\n\nGet Invoice Number\n    Open Pdf    invoice.pdf\n    ${matches} =  Find Text    Invoice Number\n    Log List      ${matches}\n\nFill Form Fields\n    Switch To Pdf    form.pdf\n    ${fields} =     Get Input Fields   encoding=utf-16\n    Log Dictionary    ${fields}\n    Set Field Value    Given Name Text Box    Mark\n    Save Field Values    output_path=${OUTPUT_DIR}${/}completed-form.pdf\n    ...                  use_appearances_writer=${True}\n
\n
\nfrom RPA.PDF import PDF\nfrom robot.libraries.String import String\n\npdf = PDF()\nstring = String()\n\ndef extract_data_from_first_page():\n    text = pdf.get_text_from_pdf("report.pdf")\n    lines = string.get_lines_matching_regexp(text[1], ".+pain.+")\n    print(lines)\n\ndef get_invoice_number():\n    pdf.open_pdf("invoice.pdf")\n    matches = pdf.find_text("Invoice Number")\n    for match in matches:\n        print(match)\n\ndef fill_form_fields():\n    pdf.switch_to_pdf("form.pdf")\n    fields = pdf.get_input_fields(encoding="utf-16")\n    for key, value in fields.items():\n        print(f"{key}: {value}")\n    pdf.set_field_value("Given Name Text Box", "Mark")\n    pdf.save_field_values(\n        output_path="completed-form.pdf",\n        use_appearances_writer=True\n    )\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:49", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -70733,7 +70733,7 @@ "name": "RPA.Robocloud.Items", "doc": "
\n

A library for interacting with Control Room work items.

\n

Work items are used for managing data that go through multiple\nsteps and tasks inside a process. Each step of a process receives\ninput work items from the previous step, and creates output work items for\nthe next step.

\n

Item structure

\n

A work item's data payload is JSON and allows storing anything that is\nserializable. This library by default interacts with payloads that\nare a dictionary of key-value pairs, which it treats as individual\nvariables. These variables can be exposed to the Robot Framework task\nto be used directly.

\n

In addition to the data section, a work item can also contain files,\nwhich are stored by default in Robocorp Control Room. Adding and using\nfiles with work items requires no additional setup from the user.

\n

Loading inputs

\n

The library automatically loads the first input work item, if the\nlibrary input argument autoload is truthy (default).

\n

After an input has been loaded its payload and files can be accessed\nthrough corresponding keywords, and optionally these values can be modified.

\n

E-mail triggering

\n

Since a process can be started in Control Room by sending an e-mail, a body\nin Text/JSON/YAML/HTML format can be sent as well and this gets attached to the\ninput work item with the rawEmail payload variable. This library automatically\nparses the content of it and saves into parsedEmail the dictionary\ntransformation of the original e-mail.

\n

If "Parse email" Control Room configuration option is enabled (recommended), then\nyour e-mail is automatically parsed in the work item under the email payload\nvariable, which is a dictionary containing a body holding the final parsed form\nof the interpreted e-mail body. The payload variable parsedEmail is still\navailable for backwards compatibility reasons and holds the very same body inside\nthe parsedEmail[Body].

\n

E-mail attachments will be added into the work item as files. Read more on:\nhttps://robocorp.com/docs/control-room/attended-or-unattended/email-trigger

\n

Example:

\n

After starting the process by sending an e-mail with a body like:

\n
\n{\n    "message": "Hello world!"\n}\n
\n

The robot can use the parsed e-mail body's dictionary:

\n
\n*** Tasks ***\nUsing Parsed Emails\n    ${mail} =    Get Work Item Variable    email\n    Set Work Item Variables    &{mail}[body]\n    ${message} =     Get Work Item Variable     message\n    Log    ${message}    # will print "Hello world!"\n
\n

The behaviour can be disabled by loading the library with\nauto_parse_email=${None} or altered by providing to it a dictionary with one\n"key: value" where the key is usually "email.text" (deprecated "rawEmail", the\nvariable set by Control Room, which acts as source for the parsed (deprecated raw)\ne-mail data) and the value can be "email.body" (deprecated "parsedEmail", where the\nparsed e-mail data gets stored into), value which can be customized and retrieved\nwith Get Work Item Variable.

\n

Creating outputs

\n

It's possible to create multiple new work items as an output from a\ntask. With the keyword Create Output Work Item a new empty item\nis created as a child for the currently loaded input.

\n

All created output items are sent into the input queue of the next\nstep in the process.

\n

Active work item

\n

Keywords that read or write from a work item always operate on the currently\nactive work item. Usually that is the input item that has been automatically\nloaded when the execution started, but the currently active item is changed\nwhenever the keywords Create Output Work Item or Get Input Work Item\nare called. It's also possible to change the active item manually with the\nkeyword Set current work item.

\n

Saving changes

\n

While a work item is loaded automatically when a suite starts, changes are\nnot automatically reflected back to the source. The work item will be modified\nlocally and then saved when the keyword Save Work Item is called.\nThis also applies to created output work items.

\n

It is recommended to defer saves until all changes have been made to prevent\nleaving work items in a half-modified state in case of failures.

\n

Local Development

\n

While Control Room is the default implementation, it can also be replaced\nwith a custom adapter. The selection is based on either the default_adapter\nargument for the library, or the RPA_WORKITEMS_ADAPTER environment\nvariable. The library has a built-in alternative adapter called FileAdapter for\nstoring work items to disk.

\n

The FileAdapter uses a local JSON file for input work items.\nIt's a list of work items, each of which has a data payload and files.

\n

An example of a local file with one work item:

\n
\n[\n    {\n        "payload": {\n            "variable1": "a-string-value",\n            "variable2": ["a", "list", "value"]\n        },\n        "files": {\n            "file1": "path/to/file.ext"\n        }\n    }\n]\n
\n

Output work items (if any) are saved to an adjacent file\nwith the same name, but with the extension .output.json. You can specify\nthrough the "RPA_OUTPUT_WORKITEM_PATH" env var a different path and name for this\nfile.

\n

Simulating the Cloud with Robocorp Code VSCode Extension

\n

If you are developing in VSCode with the Robocorp Code extension, you can\nutilize the built in local development features described in the\nDeveloping with work items locally section of the\nUsing work items development guide.

\n

Examples

\n

Robot Framework

\n

In the following example a task creates an output work item,\nand attaches some variables to it.

\n
\n*** Settings ***\nLibrary    RPA.Robocorp.WorkItems\n\n*** Tasks ***\nSave variables to Control Room\n    Create Output Work Item\n    Set work item variables    user=Dude    mail=address@company.com\n    Save Work Item\n
\n

In the next step of the process inside a different robot, we can use\npreviously saved work item variables. Also note how the input work item is\nloaded implicitly when the suite starts.

\n
\n*** Settings ***\nLibrary    RPA.Robocorp.WorkItems\n\n*** Tasks ***\nUse variables from Control Room\n    Set task variables from work item\n    Log    Variables are now available: s${user}, ${mail}\n
\n

Python

\n

The library can also be used through Python, but it does not implicitly\nload the first work item.

\n
\nimport logging\nfrom RPA.Robocorp.WorkItems import WorkItems\n\ndef list_variables(item_id):\n    library = WorkItems()\n    library.get_input_work_item()\n\n    variables = library.get_work_item_variables()\n    for variable, value in variables.items():\n        logging.info("%s = %s", variable, value)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -71091,10 +71091,10 @@ "name": "default", "types": [], "typedocs": {}, - "defaultValue": "", + "defaultValue": "", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "default=" + "repr": "default=" } ], "doc": "
\n

Return a single variable value from the work item,\nor default value if defined and key does not exist.

\n

If key does not exist and default is not defined, raises KeyError.

\n\n\n\n\n\n\n\n\n\n
param name:Name of variable
param default:Default value if key does not exist
\n

Robot Framework Example:

\n
\n*** Tasks ***\nUsing a work item\n    ${username}=    Get work item variable    username    default=guest\n
\n

Python Example:

\n
\nfrom RPA.Robocorp.WorkItems import WorkItems\n\nwi = WorkItems()\nwi.get_input_work_item()\ncustomers = wi.get_work_item_variable("customers")\nprint(customers)\n
\n
\n", @@ -71540,7 +71540,7 @@ "name": "RPA.Robocloud.Secrets", "doc": "
\n

Vault is a library for interacting with secrets stored in the Robocorp\nControl Room Vault (by default) or file-based secrets, which can be taken\ninto use by setting some environment variables.

\n

Robocorp Vault relies on environment variables, which are normally set\nautomatically by the Robocorp Work Agent or Assistant when a run is\ninitialized by the Robocorp Control Room. When developing robots locally\nin VSCode, you can use the Robocorp Code Extension to set these\nvariables automatically as well.

\n

Alternatively, you may set these environment variable manually using\nrcc or directly in some other fashion. The specific variables which\nmust exist are:

\n
    \n
  • RC_API_SECRET_HOST: URL to Robocorp Vault API
  • \n
  • RC_API_SECRET_TOKEN: API Token for Robocorp Vault API
  • \n
  • RC_WORKSPACE_ID: Control Room Workspace ID
  • \n
\n

File-based secrets can be set by defining two environment variables.

\n
    \n
  • RPA_SECRET_MANAGER: RPA.Robocorp.Vault.FileSecrets
  • \n
  • RPA_SECRET_FILE: Absolute path to the secrets database file
  • \n
\n

Example content of local secrets file:

\n
\n{\n    "swaglabs": {\n        "username": "standard_user",\n        "password": "secret_sauce"\n    }\n}\n
\n

OR

\n
\nswaglabs:\n    username: standard_user\n    password: secret_sauce\n
\n

Examples of Using Secrets in a Robot

\n

Robot Framework

\n
\n*** Settings ***\nLibrary    Collections\nLibrary    RPA.Robocorp.Vault\n\n*** Tasks ***\nReading secrets\n    ${secret}=    Get Secret  swaglabs\n    Log Many      ${secret}\n\nModifying secrets\n    ${secret}=          Get Secret      swaglabs\n    ${level}=           Set Log Level   NONE\n    Set To Dictionary   ${secret}       username    nobody\n    Set Log Level       ${level}\n    Set Secret          ${secret}\n
\n

Python

\n
\nfrom RPA.Robocorp.Vault import Vault\n\nVAULT = Vault()\n\ndef reading_secrets():\n    print(f"My secrets: {VAULT.get_secret('swaglabs')}")\n\ndef modifying_secrets():\n    secret = VAULT.get_secret("swaglabs")\n    secret["username"] = "nobody"\n    VAULT.set_secret(secret)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -71660,7 +71660,7 @@ "name": "RPA.Robocorp.Process", "doc": "
\n

A library for interacting with Control Room (CR) Process API endpoints.

\n

See Unattended processes for information about process run, step run and work\nitem states.

\n

See APIs and webhooks for information about Control Room APIs.

\n

The Process API endpoint is defined by RC_API_PROCESS_HOST environment variable,\nwhich is available during Robocorp Workforce Agent runs.

\n

Examples

\n

Robot Framework

\n

In the following example a task creates two input work items,\nand starts a process with those items. This results in 2 different\nprocess runs in the Control Room.

\n
\n*** Settings ***\nLibrary    RPA.Robocorp.Process\nLibrary    RPA.Robocorp.Vault\n\n*** Keywords ***\nInitialize Process Library\n    ${secrets}=  Get Secret  ProcessAPI\n    Set Credentials\n    ...   ${secrets}[workspace_id]\n    ...   ${secrets}[process_id]\n    ...   ${secrets}[apikey]\n\n*** Tasks ***\nStart process with work items\n    [Setup]   Initialize Process Library\n    &{item1}=  Create Dictionary  fname=Mark  lname=Monkey\n    &{item2}=  Create Dictionary  fname=John  lname=Doe\n    @{items}=  Create List  ${item1}   ${item2}\n    Start Process  work_items=${items}  batch=True\n
\n

Robot Framework

\n

In the following example a task creates work item with files.\nTo include files in a work item, the item needs to be created\nbefore starting the process (note. different start keyword than above).

\n

In this example I am using same keywords and settings from above example.

\n
\n*** Tasks ***\nStart process with work items\n    [Setup]   Initialize Process Library\n    &{data}=  Create Dictionary  fname=Mark  lname=Monkey\n    @{files}=  Create List\n    ...   ${CURDIR}${/}workdata.xlsx\n    ...   ${CURDIR}${/}other.csv\n    ${item_id}=    Create Input Work Item\n    ...   payload=${data}\n    ...   files=${files}\n    Start Configured Process\n    ...  config_type=work_items\n    ...  extra_info=${item_id}\n
\n

Download from process runs artifacts all matching files

\n
\n*** Settings ***\nLibrary      RPA.Robocorp.Process\nLibrary      RPA.Robocorp.Vault\nLibrary      RPA.HTTP\nTask Setup   Set Control Room Variables\n\n*** Keywords ***\nDownload Artifacts Matching\n    [Arguments]   ${filematch}\n    @{workitems}=   List Process Work Items\n    FOR  ${item}  IN  @{workitems}\n        @{artifacts}=   List Run Artifacts\n        ...  process_run_id=${item}[processRunId]\n        ...  step_run_id=${item}[activityRunId]\n        FOR  ${artifact}  IN  @{artifacts}\n            IF  "${filematch}"  IN   "${artifact}[fileName]"\n                ${download_link}=   Get Robot Run Artifact\n                ...  process_run_id=${item}[processRunId]\n                ...  step_run_id=${item}[activityRunId]\n                ...  artifact_id=${artifact}[id]\n                ...  filename=${artifact}[fileName]\n                Download\n                ...  url=${download_link}\n                ...  target=%{ROBOT_ARTIFACTS}${/}${artifact}[fileName]\n                ...  overwrite=${TRUE}\n                ...  stream=${TRUE}\n            END\n        END\n    END\n
\n

Python

\n

List work items in Control Room and retry failed items.

\n
\nfrom RPA.Robocorp.Process import Process\nfrom RPA.Robocorp.Vault import Vault\n\nsecrets = Vault().get_secret("ProcessAPI")\nprocess = Process(\n    secrets["workspace_id"],\n    secrets["process_id"],\n    secrets["apikey"]\n)\n\n\ndef retry_failed_items():\n    items = process.list_process_work_items()\n    for item in items:\n        if item["state"] == "FAILED":\n            print("FAILED work item: %s" % item["id"])\n            result = process.retry_work_item(item["id"])\n            print(result)\n\nif __name__ == "__main__":\n    retry_failed_items()\n
\n

Download from process runs artifacts all ".xlsx" files

\n
\nfrom RPA.Robocorp.Process import Process\nfrom RPA.HTTP import HTTP\n\ndef download_artifacts_matching(filematch=".xlsx"):\n    work_items = process.list_process_work_items()\n    for item in work_items:\n        artifacts = process.list_run_artifacts(\n            process_run_id=item["processRunId"],\n            step_run_id=item["activityRunId"]\n        )\n        for artifact in artifacts:\n            if filematch in artifact["fileName"]:\n                download_link = process.get_robot_run_artifact(\n                    process_run_id=item["processRunId"],\n                    step_run_id=item["activityRunId"],\n                    artifact_id=artifact["id"],\n                    filename=artifact["fileName"]\n                )\n                target_filepath = os.path.join(\n                    os.getenv("ROBOT_ARTIFACTS"),\n                    f"{artifact['fileName']}"\n                )\n                HTTP().download(\n                    url=download_link,\n                    target_file=target_filepath,\n                    overwrite=True,\n                    stream=True\n                )\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -72835,7 +72835,7 @@ "name": "RPA.Robocorp.Storage", "doc": "
\n

Control Room Asset Storage library operating with the cloud built-in key-value\nstore.

\n

Library requires at the minimum rpaframework version 24.0.0.

\n

Usage

\n
\n*** Tasks ***\nManage Assets\n    @{assets} =    List Assets\n    Log List    ${assets}\n\n    Set Text Asset    my-asset    My string asset value\n    ${value} =      Get Text Asset       my-asset\n    Log     Asset value: ${value}\n\n    Delete Asset    my-asset\n
\n
\nimport logging\nfrom RPA.Robocorp.Storage import Storage\n\nstorage = Storage()\n\ndef manage_assets():\n    assets = storage.list_assets()\n    logging.info(assets)\n\n    storage.set_text_asset("my-asset", "My string asset value")\n    value = storage.get_text_asset("my-asset")\n    logging.info("Asset value: %s", value)\n\n    storage.delete_asset("my-asset")\n
\n

Caveats

\n

Currently, there's no local file adapter support, therefore you need to be linked\nto Control Room and connected to a Workspace in VSCode before being able to develop\nlocally robots using this functionality.

\n

While the content type can be controlled (during bytes and file setting), it is\ncurrently disabled in this version of the library for simplicity reasons.

\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -73264,7 +73264,7 @@ "name": "RPA.Robocorp.Vault", "doc": "
\n

Vault is a library for interacting with secrets stored in the Robocorp\nControl Room Vault (by default) or file-based secrets, which can be taken\ninto use by setting some environment variables.

\n

Robocorp Vault relies on environment variables, which are normally set\nautomatically by the Robocorp Work Agent or Assistant when a run is\ninitialized by the Robocorp Control Room. When developing robots locally\nin VSCode, you can use the Robocorp Code Extension to set these\nvariables automatically as well.

\n

Alternatively, you may set these environment variable manually using\nrcc or directly in some other fashion. The specific variables which\nmust exist are:

\n
    \n
  • RC_API_SECRET_HOST: URL to Robocorp Vault API
  • \n
  • RC_API_SECRET_TOKEN: API Token for Robocorp Vault API
  • \n
  • RC_WORKSPACE_ID: Control Room Workspace ID
  • \n
\n

File-based secrets can be set by defining two environment variables.

\n
    \n
  • RPA_SECRET_MANAGER: RPA.Robocorp.Vault.FileSecrets
  • \n
  • RPA_SECRET_FILE: Absolute path to the secrets database file
  • \n
\n

Example content of local secrets file:

\n
\n{\n    "swaglabs": {\n        "username": "standard_user",\n        "password": "secret_sauce"\n    }\n}\n
\n

OR

\n
\nswaglabs:\n    username: standard_user\n    password: secret_sauce\n
\n

Examples of Using Secrets in a Robot

\n

Robot Framework

\n
\n*** Settings ***\nLibrary    Collections\nLibrary    RPA.Robocorp.Vault\n\n*** Tasks ***\nReading secrets\n    ${secret}=    Get Secret  swaglabs\n    Log Many      ${secret}\n\nModifying secrets\n    ${secret}=          Get Secret      swaglabs\n    ${level}=           Set Log Level   NONE\n    Set To Dictionary   ${secret}       username    nobody\n    Set Log Level       ${level}\n    Set Secret          ${secret}\n
\n

Python

\n
\nfrom RPA.Robocorp.Vault import Vault\n\nVAULT = Vault()\n\ndef reading_secrets():\n    print(f"My secrets: {VAULT.get_secret('swaglabs')}")\n\ndef modifying_secrets():\n    secret = VAULT.get_secret("swaglabs")\n    secret["username"] = "nobody"\n    VAULT.set_secret(secret)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:10", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -73384,7 +73384,7 @@ "name": "RPA.Robocorp.WorkItems", "doc": "
\n

A library for interacting with Control Room work items.

\n

Work items are used for managing data that go through multiple\nsteps and tasks inside a process. Each step of a process receives\ninput work items from the previous step, and creates output work items for\nthe next step.

\n

Item structure

\n

A work item's data payload is JSON and allows storing anything that is\nserializable. This library by default interacts with payloads that\nare a dictionary of key-value pairs, which it treats as individual\nvariables. These variables can be exposed to the Robot Framework task\nto be used directly.

\n

In addition to the data section, a work item can also contain files,\nwhich are stored by default in Robocorp Control Room. Adding and using\nfiles with work items requires no additional setup from the user.

\n

Loading inputs

\n

The library automatically loads the first input work item, if the\nlibrary input argument autoload is truthy (default).

\n

After an input has been loaded its payload and files can be accessed\nthrough corresponding keywords, and optionally these values can be modified.

\n

E-mail triggering

\n

Since a process can be started in Control Room by sending an e-mail, a body\nin Text/JSON/YAML/HTML format can be sent as well and this gets attached to the\ninput work item with the rawEmail payload variable. This library automatically\nparses the content of it and saves into parsedEmail the dictionary\ntransformation of the original e-mail.

\n

If "Parse email" Control Room configuration option is enabled (recommended), then\nyour e-mail is automatically parsed in the work item under the email payload\nvariable, which is a dictionary containing a body holding the final parsed form\nof the interpreted e-mail body. The payload variable parsedEmail is still\navailable for backwards compatibility reasons and holds the very same body inside\nthe parsedEmail[Body].

\n

E-mail attachments will be added into the work item as files. Read more on:\nhttps://robocorp.com/docs/control-room/attended-or-unattended/email-trigger

\n

Example:

\n

After starting the process by sending an e-mail with a body like:

\n
\n{\n    "message": "Hello world!"\n}\n
\n

The robot can use the parsed e-mail body's dictionary:

\n
\n*** Tasks ***\nUsing Parsed Emails\n    ${mail} =    Get Work Item Variable    email\n    Set Work Item Variables    &{mail}[body]\n    ${message} =     Get Work Item Variable     message\n    Log    ${message}    # will print "Hello world!"\n
\n

The behaviour can be disabled by loading the library with\nauto_parse_email=${None} or altered by providing to it a dictionary with one\n"key: value" where the key is usually "email.text" (deprecated "rawEmail", the\nvariable set by Control Room, which acts as source for the parsed (deprecated raw)\ne-mail data) and the value can be "email.body" (deprecated "parsedEmail", where the\nparsed e-mail data gets stored into), value which can be customized and retrieved\nwith Get Work Item Variable.

\n

Creating outputs

\n

It's possible to create multiple new work items as an output from a\ntask. With the keyword Create Output Work Item a new empty item\nis created as a child for the currently loaded input.

\n

All created output items are sent into the input queue of the next\nstep in the process.

\n

Active work item

\n

Keywords that read or write from a work item always operate on the currently\nactive work item. Usually that is the input item that has been automatically\nloaded when the execution started, but the currently active item is changed\nwhenever the keywords Create Output Work Item or Get Input Work Item\nare called. It's also possible to change the active item manually with the\nkeyword Set current work item.

\n

Saving changes

\n

While a work item is loaded automatically when a suite starts, changes are\nnot automatically reflected back to the source. The work item will be modified\nlocally and then saved when the keyword Save Work Item is called.\nThis also applies to created output work items.

\n

It is recommended to defer saves until all changes have been made to prevent\nleaving work items in a half-modified state in case of failures.

\n

Local Development

\n

While Control Room is the default implementation, it can also be replaced\nwith a custom adapter. The selection is based on either the default_adapter\nargument for the library, or the RPA_WORKITEMS_ADAPTER environment\nvariable. The library has a built-in alternative adapter called FileAdapter for\nstoring work items to disk.

\n

The FileAdapter uses a local JSON file for input work items.\nIt's a list of work items, each of which has a data payload and files.

\n

An example of a local file with one work item:

\n
\n[\n    {\n        "payload": {\n            "variable1": "a-string-value",\n            "variable2": ["a", "list", "value"]\n        },\n        "files": {\n            "file1": "path/to/file.ext"\n        }\n    }\n]\n
\n

Output work items (if any) are saved to an adjacent file\nwith the same name, but with the extension .output.json. You can specify\nthrough the "RPA_OUTPUT_WORKITEM_PATH" env var a different path and name for this\nfile.

\n

Simulating the Cloud with Robocorp Code VSCode Extension

\n

If you are developing in VSCode with the Robocorp Code extension, you can\nutilize the built in local development features described in the\nDeveloping with work items locally section of the\nUsing work items development guide.

\n

Examples

\n

Robot Framework

\n

In the following example a task creates an output work item,\nand attaches some variables to it.

\n
\n*** Settings ***\nLibrary    RPA.Robocorp.WorkItems\n\n*** Tasks ***\nSave variables to Control Room\n    Create Output Work Item\n    Set work item variables    user=Dude    mail=address@company.com\n    Save Work Item\n
\n

In the next step of the process inside a different robot, we can use\npreviously saved work item variables. Also note how the input work item is\nloaded implicitly when the suite starts.

\n
\n*** Settings ***\nLibrary    RPA.Robocorp.WorkItems\n\n*** Tasks ***\nUse variables from Control Room\n    Set task variables from work item\n    Log    Variables are now available: s${user}, ${mail}\n
\n

Python

\n

The library can also be used through Python, but it does not implicitly\nload the first work item.

\n
\nimport logging\nfrom RPA.Robocorp.WorkItems import WorkItems\n\ndef list_variables(item_id):\n    library = WorkItems()\n    library.get_input_work_item()\n\n    variables = library.get_work_item_variables()\n    for variable, value in variables.items():\n        logging.info("%s = %s", variable, value)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:11", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -73781,10 +73781,10 @@ "name": "default", "types": [], "typedocs": {}, - "defaultValue": "", + "defaultValue": "", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "default=" + "repr": "default=" } ], "doc": "
\n

Return a single variable value from the work item,\nor default value if defined and key does not exist.

\n

If key does not exist and default is not defined, raises KeyError.

\n\n\n\n\n\n\n\n\n\n
param name:Name of variable
param default:Default value if key does not exist
\n

Robot Framework Example:

\n
\n*** Tasks ***\nUsing a work item\n    ${username}=    Get work item variable    username    default=guest\n
\n

Python Example:

\n
\nfrom RPA.Robocorp.WorkItems import WorkItems\n\nwi = WorkItems()\nwi.get_input_work_item()\ncustomers = wi.get_work_item_variable("customers")\nprint(customers)\n
\n
\n", @@ -74234,7 +74234,7 @@ "name": "RPA.RobotLogListener", "doc": "
\n

RobotLogListener is a library for controlling logging during\na Robot Framework execution using the listener API.

\n

About keyword parameters

\n

Parameters names and keywords for keywords Mute Run On Failure and Register Protected Keywords\ndo not need to be full names of keywords, ie. all keywords matching even partially will be affected.\nRun Keyword would match all BuiltIn library keywords (17 keywords in RF 3.2.1) and of course all\nRun Keyword named keywords in any resource and/or library file which are imported would be matched also.

\n

Mute Run On Failure

\n

This keyword is to be used specifically with RPA.Browser.Selenium library, which extends\nSeleniumLibrary. Normally most of the SeleniumLibrary keywords execute run_on_failure\nbehaviour, which can be set at library initialization. By default this behaviour is running\nCapture Page Screenshot keyword on failure.

\n

In the example task Check the official website below the keyword Run Keyword is muted and when\nit runs the keyword Element Should Be Visible then those failures do not create page screenshots\ninto log file.

\n

It is also possible to change default failure behaviour by giving parameter\noptional_keyword_to_run for Mute Run On Failure, see task Check the official website with error log.\nThis optional keyword would be then executed on failure. Due to the underlying SeleniumLibrary\nimplementation this keyword can't have arguments.

\n

Example of using Mute Run On Failure without and with optional keyword to run.

\n
\n*** Settings ***\nLibrary         RPA.Browser.Selenium\nLibrary         RPA.RobotLogListener\nTask Setup      Set Task Variable   ${TRIES}   1\nTask Teardown   Close All Browsers\n\n*** Tasks ***\nCheck the official website\n   Mute Run On Failure   Run Keyword\n   Open Available Browser   https://www.robocorp.com\n   Check for visible element\n   Capture Page Screenshot\n\nCheck the official website with error log\n   Mute Run On Failure   Run Keyword  optional_keyword_to_run=Log tries\n   Open Available Browser   https://www.robocorp.com\n   Check for visible element\n   Capture Page Screenshot\n\n*** Keywords ***\nCheck for visible element\n   FOR  ${idx}  IN RANGE  1   20\n      Set Task Variable   ${TRIES}   ${idx}\n      ${status}   Run Keyword And Return Status   Element Should Be Visible  id:xyz\n      Exit For Loop If   '${status}' == 'PASS'\n      Sleep  2s\n   END\n\nLog tries\n   Log  Checked element visibility ${TRIES} times\n
\n

Register Protected Keywords

\n

This keyword is used to totally disable logging for named keywords. In the example below\nthe keyword This will not output is protected and it will not be logging into Robot Framework\nlog files.

\n

Robot Framework

\n
\n*** Settings ***\nLibrary         RPA.RobotLogListener\n\n*** Tasks ***\nProtecting keywords\n   This will not output        # will output because called before register\n   Register Protected Keywords    This will not output\n   This will not output        # is now registered\n   This will output\n\n*** Keywords ***\nThis will not output\n   Log   1\n\nThis will output\n   Log   2\n
\n

Python

\n
\nfrom robot.libraries.BuiltIn import BuiltIn, RobotNotRunningError\nfrom RPA.RobotLogListener import RobotLogListener\n\ntry:\n   BuiltIn().import_library("RPA.RobotLogListener")\nexcept RobotNotRunningError:\n   pass\n\nclass CustomLibrary:\n\n   def __init__(self):\n      listener = RobotLogListener()\n      listener.register_protected_keywords(\n            ["CustomLibrary.special_keyword"]\n      )\n\n   def special_keyword(self):\n      print('will not be written to log')\n      return 'not shown in the log'\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:50", + "generated": "2023-11-10 12:56:11", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -74448,7 +74448,7 @@ "name": "RPA.SAP", "doc": "

This library wraps the upstream SapGuiLibrary.

\n

The SapGuiLibrary is a library that enables users to create tests for the Sap Gui application

\n

The library uses the Sap Scripting Engine, therefore Scripting must be enabled in Sap in order for this library to work.

\n

Opening a connection / Before running tests

\n

First of all, you have to make sure the Sap Logon Pad is started. You can automate this process by using the AutoIT library or the Process Library.

\n

After the Sap Login Pad is started, you can connect to the Sap Session using the keyword connect to session.

\n

If you have a successful connection you can use Open Connection to open a new connection from the Sap Logon Pad or Connect To Existing Connection to connect to a connection that is already open.

\n

Locating or specifying elements

\n

You need to specify elements starting from the window ID, for example, wnd[0]/tbar[1]/btn[8]. In some cases the SAP ID contains backslashes. Make sure you escape these backslashes by adding another backslash in front of it.

\n

Screenshots (on error)

\n

The SapGUILibrary offers an option for automatic screenshots on error. Default this option is enabled, use keyword disable screenshots on error to skip the screenshot functionality. Alternatively, this option can be set at import.

", "version": "1.1", - "generated": "2023-11-10 12:47:51", + "generated": "2023-11-10 12:56:11", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -75672,7 +75672,7 @@ "name": "RPA.Salesforce", "doc": "
\n

Salesforce is a library for accessing Salesforce using REST API.\nThe library extends simple-salesforce library.

\n

More information available at Salesforce REST API Developer Guide.

\n

Dataloader

\n

The keyword execute_dataloader_import can be used to mimic\nSalesforce Dataloader import behaviour.

\n

input_object can be given in different formats. Below is an example where\ninput is in RPA.Table format in method a and list format in method b.

\n
\n*** Settings ***\nLibrary     RPA.Salesforce\nLibrary     RPA.Database\nTask Setup  Authorize Salesforce\n\n*** Tasks ***\n# Method a\n${orders}=        Database Query Result As Table\n...               SELECT * FROM incoming_orders\n${status}=        Execute Dataloader Insert\n...               ${orders}  ${mapping_dict}  Tilaus__c\n# Method b\n${status}=        Execute Dataloader Insert\n...               ${WORKDIR}${/}orders.json  ${mapping_dict}  Tilaus__c\n
\n

Example file orders.json

\n
\n[\n    {\n        "asiakas": "0015I000002jBLIQA2"\n    },\n    {\n        "asiakas": "0015I000002jBLDQA2"\n    },\n]\n
\n

mapping_object describes how the input data fields are mapped into Salesforce\nobject attributes. In the example, the mapping defines that asiakas attribute in the\ninput object is mapped into Tilaaja__c attribute of Tilaus__c custom Salesforce object.

\n
\n{\n    "Tilaus__c": {\n        "asiakas": "Tilaaja__c"\n    },\n}\n
\n

Object type could be, for example, Tilaus__c.

\n

Salesforce object operations

\n

Following operations can be used to manage Salesforce objects:

\n
    \n
  • Get Salesforce Object By Id
  • \n
  • Create Salesforce Object
  • \n
  • Update Salesforce Object
  • \n
  • Upsert Salesforce Object
  • \n
  • Delete Salesforce Object
  • \n
  • Get Salesforce Object Metadata
  • \n
  • Describe Salesforce Object
  • \n
\n

There are two ways to set the Salesforce domain. You can set the domain at time of\nlibrary import or using the Set Domain keyword.

\n

There are several ways to declare a domain at time of library import:

\n
\n*** Settings ***\nLibrary     RPA.Salesforce    sandbox=${TRUE}\n
\n

Or using the domain to your Salesforce My domain:

\n
\n*** Settings ***\nLibrary     RPA.Salesforce    domain="robocorp"\n
\n

The domain can also be set using the keyword Set Domain:

\n
\n*** Settings ***\nLibrary     RPA.Salesforce\n\n*** Tasks ***\n# Sets the domain for a sandbox environment\nSet Domain    sandbox\n\n# Sets the domain to a Salseforce My domain\nSet Domain    robocorp\n\n# Sets to domain to the default of 'login'\nSet Domain\n
\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary     RPA.Salesforce\nTask Setup  Authorize Salesforce\n\n*** Variables ***\n${ACCOUNT_NOKIA}    0015I000002jBLDQA2\n\n*** Tasks ***\nChange account details in Salesforce\n    &{account}=      Get Salesforce Object By Id   Account  ${ACCOUNT_NOKIA}\n    &{update_obj}=   Create Dictionary   Name=Nokia Ltd  BillingStreet=Nokia bulevard 1\n    ${result}=       Update Salesforce Object  Account  ${ACCOUNT_NOKIA}  ${update_obj}\n\n*** Keywords ***\nAuthorize Salesforce\n    ${secrets}=     Get Secret   salesforce\n    Auth With Token\n    ...        username=${secrets}[USERNAME]\n    ...        password=${secrets}[PASSWORD]\n    ...        api_token=${secrets}[API_TOKEN]\n
\n

Python

\n
\nimport pprint\nfrom RPA.Salesforce import Salesforce\nfrom RPA.Robocorp.Vault import FileSecrets\n\npp = pprint.PrettyPrinter(indent=4)\nfilesecrets = FileSecrets("secrets.json")\nsecrets = filesecrets.get_secret("salesforce")\n\nsf = Salesforce()\nsf.auth_with_token(\n    username=secrets["USERNAME"],\n    password=secrets["PASSWORD"],\n    api_token=secrets["API_TOKEN"],\n)\nnokia_account_id = "0015I000002jBLDQA2"\naccount = sf.get_salesforce_object_by_id("Account", nokia_account_id)\npp.pprint(account)\nbilling_information = {\n    "BillingStreet": "Nokia Bulevard 1",\n    "BillingCity": "Espoo",\n    "BillingPostalCode": "01210",\n    "BillingCountry": "Finland",\n}\nresult = sf.update_salesforce_object("Account", nokia_account_id, billing_information)\nprint(f"Update result: {result}")\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:51", + "generated": "2023-11-10 12:56:11", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -76731,7 +76731,7 @@ "name": "RPA.Slack", "doc": "
\n

RPA Framework library for Slack operations.

\n
\n", "version": "", - "generated": "2023-11-10 12:47:51", + "generated": "2023-11-10 12:56:11", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -76919,7 +76919,7 @@ "name": "RPA.Smartsheet", "doc": "
\n

Smartsheet is a library for accessing Smartsheet using the\nSmartsheet API 2.0. It extends smartsheet-python-sdk.

\n
\n

Getting started

\n

To use this library, you need to have a Smartsheet account and an API token.\nYou can get your API token from the Smartsheet Developer Portal.\nThis library currently only supports raw token authentication. Once\nobtained, you can configure the access token using the Set Access Token\nkeyword or via the access_token argument in the library import.

\n
\n
\n

Working on a sheet

\n

The library supports working on a single sheet at a time. To select a sheet\nto work on, use the Select Sheet keyword. This will set the sheet as\nthe active sheet for all subsequent operations. Some operations\nupdate the sheet, but this will not necessarily be reflected in the active\nsheet. To refresh the active sheet, use the Refresh Sheet keyword.

\n
\n
\n

Native Smartsheet objects

\n

You can retrieve the native Smartsheet object from many keywords by\nspecifying the native argument. The default will return a more\ncommon Python object, such as a dictionary or list. The native object\nis a class from the smartsheet-python-sdk library and will have\nadditional methods and attributes. The most important attributes\navailable for most native objects are (some may be unavailable\nfor some objects):

\n
    \n
  • id: the unique identifier of the object
  • \n
  • name: the name of the object
  • \n
  • title: the title of a column
  • \n
  • permalink: the URL to the object
  • \n
\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:52", + "generated": "2023-11-10 12:56:11", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -78236,7 +78236,7 @@ "name": "RPA.Tables", "doc": "
\n

Tables is a library for manipulating tabular data inside Robot Framework.

\n

It can import data from various sources and apply different operations to it.\nCommon use-cases are reading and writing CSV files, inspecting files in\ndirectories, or running tasks using existing Excel data.

\n

Import types

\n

The data a table can be created from can be of two main types:

\n
    \n
  1. An iterable of individual rows, like a list of lists, or list of dictionaries
  2. \n
  3. A dictionary of columns, where each dictionary value is a list of values
  4. \n
\n

For instance, these two input values:

\n
\ndata1 = [\n    {"name": "Mark", "age": 58},\n    {"name": "John", "age": 22},\n    {"name": "Adam", "age": 67},\n]\n\ndata2 = {\n    "name": ["Mark", "John", "Adam"],\n    "age":  [    58,     22,     67],\n}\n
\n

Would both result in the following table:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
IndexNameAge
0Mark58
1John22
2Adam67
\n

Indexing columns and rows

\n

Columns can be referred to in two ways: either with a unique string\nname or their position as an integer. Columns can be named either when\nthe table is created, or they can be (re)named dynamically with keywords.\nThe integer position can always be used, and it starts from zero.

\n

For instance, a table with columns "Name", "Age", and "Address" would\nallow referring to the "Age" column with either the name "Age" or the\nnumber 1.

\n

Rows do not have a name, but instead only have an integer index. This\nindex also starts from zero. Keywords where rows are indexed also support\nnegative values, which start counting backwards from the end.

\n

For instance, in a table with five rows, the first row could be referred\nto with the number 0. The last row could be accessed with either 4 or\n-1.

\n

Examples

\n

Robot Framework

\n

The Tables library can load tabular data from various other libraries\nand manipulate it inside Robot Framework.

\n
\n*** Settings ***\nLibrary    RPA.Tables\n\n*** Keywords ***\nFiles to Table\n    ${files}=    List files in directory    ${CURDIR}\n    ${files}=    Create table    ${files}\n    Filter table by column    ${files}    size  >=  ${1024}\n    FOR    ${file}    IN    @{files}\n        Log    ${file}[name]\n    END\n    Write table to CSV    ${files}    ${OUTPUT_DIR}${/}files.csv\n
\n

Python

\n

The library is also available directly through Python, where it\nis easier to handle multiple different tables or do more bespoke\nmanipulation operations.

\n
\nfrom RPA.Tables import Tables\n\nlibrary = Tables()\norders = library.read_table_from_csv(\n    "orders.csv", columns=["name", "mail", "product"]\n)\n\ncustomers = library.group_table_by_column(rows, "mail")\nfor customer in customers:\n    for order in customer:\n        add_cart(order)\n    make_order()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:52", + "generated": "2023-11-10 12:56:12", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -79904,7 +79904,7 @@ "name": "RPA.Tasks", "doc": "
\n

Tasks is a library for controlling task execution during a Robot Framework run.

\n

It allows conditional branching between tasks, loops and jumps, and optionally\nvalidating the execution through a schema file. It can also be used to\nvisualize the tasks as a graph.

\n

Execution model

\n

In a typical Robot Framework run, tasks are ordered linearly in a file and\nthey're executed in definition order. Events that happen during\nthe execution can not affect the order and only have the option to fail the task\nor continue as defined.

\n

Using the Tasks library, it's possible to change this model according\nto different states or inputs. The execution will start by running a single\nstart task from the suite, and then according to user-defined keywords or\nschema rules select the next task. Any task which is defined in the same file\ncan be used, and the same task can also be used multiple times during a single\nexecution.

\n

Example:

\n

As an example, the following Robot Framework file describes a process where\na task would have to be executed multiple times before a condition is reached.\nIn a real-world scenario, these tasks would be more complicated, instead of just\nincrementing numbers.

\n
\n*** Settings ***\nLibrary    RPA.Tasks\n\n*** Variables ***\n${CURRENT}    ${1}\n${TARGET}     ${5}\n\n*** Tasks ***\nCheck loop condition\n    Log    I'm trying to count to ${TARGET}\n    Set next task if    ${CURRENT} >= ${TARGET}\n    ...    Target reached\n    ...    Increment current number\n\nThis will not run\n    Fail    This should never run\n\nIncrement current number\n    Set suite variable    ${CURRENT}    ${CURRENT + 1}\n    Log    Number is now ${CURRENT}\n    Jump to task    Check loop condition\n\nTarget reached\n    Log    Those are some good numbers!\n
\n

The execution for this example would go as follows:

\n
    \n
  1. It starts from Check loop condition, as it's the first task in the file.
  2. \n
  3. During the first task, the keyword Set next task if is called, which queues\nup the next task according to a condition.
  4. \n
  5. In the initial state, we have not reached the target number, and will next run\nthe task Increment current number.
  6. \n
  7. The second task executes normally and in the end jumps back to the first\ntask using the keyword Jump to task.
  8. \n
  9. The above sequence is repeated until the condition is met, and we move to\nthe final task of the file. This final task does not schedule further tasks\nand the execution ends.
  10. \n
\n

You can also note the task This will not run, which as the name implies\nis never executed, as no other task schedules or jumps to it.

\n

The console log from the above execution shows how the same task is executed\nmultiple times:

\n
\n==============================================================================\nIncrementing Process\n==============================================================================\n#1   Check loop condition                                             | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#2   Increment current number                                         | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#3   Check loop condition                                             | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#4   Increment current number                                         | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#5   Check loop condition                                             | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#6   Increment current number                                         | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#7   Check loop condition                                             | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#8   Increment current number                                         | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#9   Check loop condition                                             | PASS |\nTransition: Set by keyword\n------------------------------------------------------------------------------\n#10  Target reached                                                   | PASS |\n------------------------------------------------------------------------------\nIncrementing Process:: [/graph_incrementing_process.svg]              | PASS |\n10 critical tasks, 10 passed, 0 failed\n10 tasks total, 10 passed, 0 failed\n==============================================================================\n
\n

Graph

\n

A common way to document a process is through a directed graph. These graphs\nare usually drawn manually and describe the expected higher level steps.\nThe actual implementation itself follows a different path through a graph,\ndepending on inputs or implementation details. This library visualizes this\nexecution graph using the Graphviz tool.

\n

After the execution is finished, it will create a\nDOT file\nand render it as an image. This image will automatically be appended\nto the suite's documentation field.

\n

Requirements

\n

Drawing the graph requires a working installation of\nGraphviz. This can be installed through their\nwebsite or by using Conda.

\n

This requirement is optional for the functioning of this library, and will\ndisplay a warning if the tool is not available. The visualization\ncan be entirely disabled with the graph argument during library\ninitialization.

\n

Schema

\n

There is an option to define a schema file for the suite, which is written in JSON.\nThis file will be used to validate the actual execution and fail it if an unexpected\ntransition between tasks happens. It can also define rules for selecting the next\ntask, which allows separating the task and process definitions.

\n

Example:

\n

The execution-example shown previously used keywords to control\nthe execution. This can also be done using the following schema:

\n
\n{\n    "tasks": {\n        "Check loop condition": {\n            "alias": "check",\n            "start": true,\n            "next": [\n                "increment",\n                "target"\n            ],\n            "actions": [\n                {\n                    "condition": "$CURRENT >= $TARGET",\n                    "task": "target"\n                },\n                {\n                    "condition": "$CURRENT < $TARGET",\n                    "task": "increment"\n                }\n            ]\n        },\n        "Increment current number": {\n            "alias": "increment",\n            "next": [\n                "check"\n            ],\n            "actions": [\n                {\n                    "status": "PASS",\n                    "task": "check"\n                }\n            ]\n        },\n        "Target reached": {\n            "alias": "target",\n            "end": true,\n            "next": []\n        }\n    }\n}\n
\n

This has the added benefit of protecting against implementation errors,\nas the library will validate the start and end tasks, and transitions between\ndifferent tasks.

\n

After this schema has been taken into use, the aforementioned example\nwill reduce to the following:

\n
\n*** Settings ***\nLibrary    RPA.Tasks    schema=counter-schema.json\n\n*** Variables ***\n${CURRENT}    ${1}\n${TARGET}     ${5}\n\n*** Tasks ***\nCheck loop condition\n    Log    I'm trying to count to ${TARGET}\n\nThis will not run\n    Fail    This should never run\n\nIncrement current number\n    Set suite variable    ${CURRENT}    ${CURRENT + 1}\n    Log    Number is now ${CURRENT}\n\nTarget reached\n    Log    Those are some good numbers!\n
\n

Format

\n

The current format is JSON with the following structure:

\n
\n{\n    "tasks": {\n        [name: string]: {\n            "alias": string,\n            "start": boolean,\n            "end": boolean,\n            "next": string[],\n            "actions": action[],\n        }\n    }\n}\n
\n

Each schema is a map of tasks with various properties. The keys must\nmatch the task names in the Robot Framework file definition. All properties\ninside the task are optional.

\n

The available properties and their uses:

\n
    \n
  • \n
    alias: Define a short name for the task, which can be used as a reference
    \n
    inside the schema.
    \n
    \n
  • \n
  • \n
    start: Start task for execution. There can be only one task with this
    \n
    enabled. If not defined, will default to first task in the file.
    \n
    \n
  • \n
  • \n
    end: Valid end task for execution. There can be multiple tasks with this
    \n
    enabled. Fails the execution if this is defined for any task and the\nexecution stops in a non-end task.
    \n
    \n
  • \n
  • \n
    next: List of valid tasks to transition to from this task. Supports
    \n
    alias definitions.
    \n
    \n
  • \n
  • \n
    actions: List of actions that are executed at the end of the task.
    \n
    See section below for details.
    \n
    \n
  • \n
\n

The types of actions:

\n
    \n
  • \n
    exception: Set the next task if a matching exception occurs.
    \n
    Matches the exception message as regex.
    \n
    \n
  • \n
  • \n
    condition: Set the next task if a conditional expression is true.
    \n
    Allows using Robot Framework variables.
    \n
    \n
  • \n
  • \n
    status: Set the next task if the current task's result matches,
    \n
    e.g. PASS or FAIL.
    \n
    \n
  • \n
\n

Examples of actions:

\n
\n[\n    {"exception": ".*ValueError.*", "task": "Invalid input values"},\n    {"condition": "$ATTEMPTS > 10", "task": "Too many attempts"},\n    {"status": "PASS", "task": "Success state"}\n]\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:52", + "generated": "2023-11-10 12:56:12", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -80232,7 +80232,7 @@ "name": "RPA.Twitter", "doc": "
\n

Twitter is a library for accessing Twitter using developer API.\nThe library extends tweepy library.

\n

Authorization credentials can be given as parameters for authorize keyword\nor keyword can read them in as environment variables:

\n
    \n
  • TWITTER_CONSUMER_KEY
  • \n
  • TWITTER_CONSUMER_SECRET
  • \n
  • TWITTER_ACCESS_TOKEN
  • \n
  • TWITTER_ACCESS_TOKEN_SECRET
  • \n
\n

Library usage requires Twitter developer credentials.\nThose can be requested from Twitter developer site

\n

Examples

\n
\n*** Settings ***\nLibrary   RPA.Twitter\n\n*** Tasks ***\nGet user tweets and like them\n    [Setup]   Authorize\n    @{tweets}=   Get User Tweets   username=niinisto   count=5\n    FOR   ${tweet}  IN   @{tweets}\n        Like   ${tweet}\n    END\n
\n
\nfrom RPA.Twitter import Twitter\n\nlibrary = Twitter()\nlibrary.authorize()\ntweets = library.get_user_tweets(username="niinisto", count=5)\nfor tw in tweets:\n    library.like(tw)\ntweets = library.text_search_tweets(query="corona trump")\nfor tw in tweets:\n    print(tw.text)\nuser = library.get_user_profile("niinisto")\nlibrary.follow(user)\nlibrary.tweet("first tweet")\nme = library.get_me()\nprint(me)\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:52", + "generated": "2023-11-10 12:56:12", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -80709,7 +80709,7 @@ "name": "RPA.Windows", "doc": "
\n

The Windows is a library that can be used for Windows desktop automation.

\n

Library is included in the rpaframework package by default, but as shown in the\nbelow example library can be also installed separately without rpaframework.

\n
\nchannels:\n  - conda-forge\ndependencies:\n  - python=3.9.13\n  - pip=22.1.2\n  - pip:\n    - rpaframework-windows==7.0.2 # standalone Windows library (rpaframework includes this library)\n
\n

About terminology

\n

ControlType is a value referred to by locator keys type: or control. Represents type of application\nobject, which can be e.g. Window, Button or ListItem.

\n

Element is an entity of an application structure (e.g. certain button in a window), which can be\nidentified by a locator. (also referred as Control)

\n

WindowsElement is an library container object for the Element. All the keywords returning elements, will in\nfact return WindowsElement`s. The ones accepting `locator or root_element as arguments, will accept\nWindowsElement as an argument value. (locator accepts strings as well)

\n

Structure of the WindowsElement

\n
\nclass WindowsElement:\n    item: Control        # `item` contains object instance of the element\n    locator: str         # `locator` that found this element\n    name: str            # `Name` attribute of the element\n    automation_id: str   # `AutomationId` attribute of the element\n    control_type: str    # `ControlTypeName` attribute of the element\n    class_name: str      # `ClassName` attribute of the element\n    left: int            # element's rectangle left coordinate\n    right: int           # element's rectangle right coordinate\n    top: int             # element's rectangle top coordinate\n    bottom: int          # element's rectangle bottom coordinate\n    width: int           # element's rectangle horizontal width\n    height: int          # element's rectangle vertical height\n    xcenter: int         # element's rectangle center point x coordinate\n    ycenter: int         # element's rectangle center point y coordinate\n
\n

Example of the WindowsElement usage

\n
\n${rows}=    Get Elements    class:DataGridRow\n# ${rows} is a list of `WindowsElement`s\nFOR    ${row}    IN    @{rows}\n    Log To Console   ${row.name}                # access `WindowsElement`\n    Log To Console   ${row.item.AutomationId}   # access `WindowsElement.item` directly\n    Log To Console   ${row.item.Name}           # same as `${row.name}`\nEND\n
\n

Locators

\n

Locators are based on different strategies that can used identify Control object.

\n

Available strategies that can be used for the locators:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
KeyMaps to search property
nameName
classClassName
typeControlType
controlControlType
idAutomationId
automationidAutomationId
regexRegexName
subnameSubName
indexfoundIndex (int)
offsetoffset coordinates (x (int), y (int)) from control center
executabletarget window by its executable name
handletarget window handle (int)
desktopSPECIAL target desktop, no value for the key e.g. desktop:desktop and name:Calculator
processNOT YET SUPPORTED target window by its executable's process id
depthsearchDepth (int) for finding Control (default 8)
pathtarget element by its index-based path traversal (e.g. path:2|3|8|2)
\n

About root element on locators

\n

Locators work on currently active root element. At the start root element is the whole\ndesktop. There are different ways on changing this root element.

\n

Keyword Control Window is the most common method of setting certain system window\nas a root element for further actions using locators. In the absence of a provided\nroot_element parameter, here's how you can control the default root element\nresolving:

\n
\n
    \n
  • Set Anchor: Sets the active anchor window from which the search begins.
  • \n
  • Control Window: Controls and focuses on a window and marks it as the current\nactive window, from which all the subsequent searches will start from in the\nabsence of a set anchor.
  • \n
  • If there's no set anchor nor active window, then the last resort will be the\n"Desktop" element itself.
  • \n
\n
\n

Locators themselves support cascading syntax (denoted by character > in the locator string),\nwhich can denote root element in "parent (root) & child" terms.

\n

For example.

\n
\nClick  id:controls > id:activate\n
\n

On the above example the left side of the > character, id:controls, represents the root element\n(can be called as "parent element" in this case). Right side of the locator string, id:activate,\nrepresents "child" element and it will be searched under the "parent element".

\n

This way element search is more efficient, because search are restricted to certain section of element\ntree which can be quite huge especially on the desktop level and in certain applications.

\n

Keyword examples:

\n
\nControl Window    name:Calculator\nControl Window    Calculator  # will execute search by 'name:Calculator'\nControl Window    executable:Spotify.exe\n
\n

some example locators, and can be omitted ie. space between locator keys means the same thing as and:

\n
\nid:clearButton\ntype:Group and name:"Number pad" > type:Button and index:4\ntype:Group and name:"Number pad" > control:Button index:5\nid:Units1 > name:${unit}\nclass:Button offset:370,0\n
\n

About locator restrictions

\n

Visual locators are not supported in this library and they can't be used in the same chain with these\nWindows locators. Visual locators are supported by the RPA.Desktop library. Locator chaining (image and\nWindows locators) support will be added in the future.

\n

Locator syntax does not yet support OR operation (only AND operations).

\n

About search depth

\n

The library does element search depth by default to the level of 8. This means that locator will look into\n8 levels of elements under element tree of the root element. This can lead into situation where element\ncan't be found. To fix this it is recommended to set root element which can be found within 8 levels OR\ndefining depth in the locator string to a bigger value, e.g. id:deeplyNestedButton depth:16. Useful\nkeywords for setting root element are Control Window, Set Anchor and Get Element.

\n

About the path strategy

\n

When automation IDs and names aren't enough (or not reliable), then you can fallback\nto the positions of elements in a tree. This can be achieved using the path:\nstrategy to specify a list of element positions which indicates how to traverse the\ntree from parent to child beginning with the resolved root.

\n

Example: Calculator > path:2|3|2|8|2 - this locator looks for the "Calculator"\nwindow, then it looks for the 2nd direct child and then it looks for the 3rd one of\nthe previous child and so on until it consumes the path completely. (indexes start\nwith 1)

\n

An alternative way to get the whole tree to explore it yourself would be to use the\nPrint Tree keyword.

\n

Keyboard and mouse

\n

Keys for the keyboard actions are given using uiautomation specification.

\n

Special keys which are given within {} syntax.

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
KeyMaps to action
LBUTTONLeft mouse button
RBUTTONRight mouse button
CANCELControl-break processing
MBUTTONMiddle mouse button (three-button mouse)
XBUTTON1X1 mouse button
XBUTTON2X2 mouse button
BACKBACKSPACE key
TABTAB key
CLEARCLEAR key
RETURNENTER key
ENTERENTER key
SHIFTSHIFT key
CTRLCTRL key
CONTROLCTRL key
ALTALT key
PAUSEPAUSE key
CAPITALCAPS LOCK key
KANAIME Kana mode
HANGUELIME Hanguel mode (maintained for compatibility; use VK_HANGUL)
HANGULIME Hangul mode
JUNJAIME Junja mode
FINALIME final mode
HANJAIME Hanja mode
KANJIIME Kanji mode
ESCESC key
ESCAPEESC key
CONVERTIME convert
NONCONVERTIME nonconvert
ACCEPTIME accept
MODECHANGEIME mode change request
SPACESPACEBAR
PRIORPAGE UP key
PAGEUPPAGE UP key
NEXTPAGE DOWN key
PAGEDOWNPAGE DOWN key
ENDEND key
HOMEHOME key
LEFTLEFT ARROW key
UPUP ARROW key
RIGHTRIGHT ARROW key
DOWNDOWN ARROW key
SELECTSELECT key
PRINTPRINT key
EXECUTEEXECUTE key
SNAPSHOTPRINT SCREEN key
PRINTSCREENPRINT SCREEN key
INSERTINS key
INSINS key
DELETEDEL key
DELDEL key
HELPHELP key
WINLeft Windows key (Natural keyboard)
LWINLeft Windows key (Natural keyboard)
RWINRight Windows key (Natural keyboard)
APPSApplications key (Natural keyboard)
SLEEPComputer Sleep key
NUMPAD0Numeric keypad 0 key
NUMPAD1Numeric keypad 1 key
NUMPAD2Numeric keypad 2 key
NUMPAD3Numeric keypad 3 key
NUMPAD4Numeric keypad 4 key
NUMPAD5Numeric keypad 5 key
NUMPAD6Numeric keypad 6 key
NUMPAD7Numeric keypad 7 key
NUMPAD8Numeric keypad 8 key
NUMPAD9Numeric keypad 9 key
MULTIPLYMultiply key
ADDAdd key
SEPARATORSeparator key
SUBTRACTSubtract key
DECIMALDecimal key
DIVIDEDivide key
F1F1 key
F2F2 key
F3F3 key
F4F4 key
F5F5 key
F6F6 key
F7F7 key
F8F8 key
F9F9 key
F10F10 key
F11F11 key
F12F12 key
F13F13 key
F14F14 key
F15F15 key
F16F16 key
F17F17 key
F18F18 key
F19F19 key
F20F20 key
F21F21 key
F22F22 key
F23F23 key
F24F24 key
NUMLOCKNUM LOCK key
SCROLLSCROLL LOCK key
LSHIFTLeft SHIFT key
RSHIFTRight SHIFT key
LCONTROLLeft CONTROL key
LCTRLLeft CONTROL key
RCONTROLRight CONTROL key
RCTRLRight CONTROL key
LALTLeft MENU key
RALTRight MENU key
BROWSER_BACKBrowser Back key
BROWSER_FORWARDBrowser Forward key
BROWSER_REFRESHBrowser Refresh key
BROWSER_STOPBrowser Stop key
BROWSER_SEARCHBrowser Search key
BROWSER_FAVORITESBrowser Favorites key
BROWSER_HOMEBrowser Start and Home key
VOLUME_MUTEVolume Mute key
VOLUME_DOWNVolume Down key
VOLUME_UPVolume Up key
MEDIA_NEXT_TRACKNext Track key
MEDIA_PREV_TRACKPrevious Track key
MEDIA_STOPStop Media key
MEDIA_PLAY_PAUSEPlay/Pause Media key
LAUNCH_MAILStart Mail key
LAUNCH_MEDIA_SELECTSelect Media key
LAUNCH_APP1Start Application 1 key
LAUNCH_APP2Start Application 2 key
OEM_1Used for miscellaneous characters; it can vary by keyboard.For the US standard keyboard, the ';:' key
OEM_PLUSFor any country/region, the '+' key
OEM_COMMAFor any country/region, the ',' key
OEM_MINUSFor any country/region, the '-' key
OEM_PERIODFor any country/region, the '.' key
OEM_2Used for miscellaneous characters; it can vary by keyboard.
OEM_3Used for miscellaneous characters; it can vary by keyboard.
OEM_4Used for miscellaneous characters; it can vary by keyboard.
OEM_5Used for miscellaneous characters; it can vary by keyboard.
OEM_6Used for miscellaneous characters; it can vary by keyboard.
OEM_7Used for miscellaneous characters; it can vary by keyboard.
OEM_8Used for miscellaneous characters; it can vary by keyboard.
OEM_102Either the angle bracket key or the backslash key on the RT 102-key keyboard
PROCESSKEYIME PROCESS key
PACKETUsed to pass Unicode characters as if they were keystrokes. The VK_PACKET key is the low word of a 32-bit Virtual Key value used for non-keyboard input methods. For more information, see Remark in KEYBDINPUT, SendInput, WM_KEYDOWN, and WM_KeyUp
ATTNAttn key
CRSELCrSel key
EXSELExSel key
EREOFErase EOF key
PLAYPlay key
ZOOMZoom key
NONAMEReserved
PA1PA1 key
OEM_CLEARClear key
\n

Examples.

\n
\nlib = Windows()\n# {Ctrl}, {Delete} ... are special keys' name in SpecialKeyNames.\nlib.send_keys('{Ctrl}a{Delete}{Ctrl}v{Ctrl}s{Ctrl}{Shift}s{Win}e{PageDown}') #press Ctrl+a, Delete, Ctrl+v, Ctrl+s, Ctrl+Shift+s, Win+e, PageDown\nlib.send_keys('{Ctrl}(AB)({Shift}(123))') #press Ctrl+A+B, type '(', press Shift+1+2+3, type ')', if '()' follows a hold key, hold key won't release util ')'\nlib.send_keys('{Ctrl}{a 3}') #press Ctrl+a at the same time, release Ctrl+a, then type 'a' 2 times\nlib.send_keys('{a 3}{B 5}') #type 'a' 3 times, type 'B' 5 times\nlib.send_keys('{{}Hello{}}abc {a}{b}{c} test{} 3}{!}{a} (){(}{)}') #type: '{Hello}abc abc test}}}!a ()()'\nlib.send_keys('0123456789{Enter}')\nlib.send_keys('ABCDEFGHIJKLMNOPQRSTUVWXYZ{Enter}')\nlib.send_keys('abcdefghijklmnopqrstuvwxyz{Enter}')\nlib.send_keys('`~!@#$%^&*()-_=+{Enter}')\nlib.send_keys('[]{{}{}}\\|;:'",<.>/?{Enter}')\n
\n

Using access key of the element (element property -> AccessKey 'alt+s').\nThe (+s) means that previous special key is kept down until closing parenthesis is reached.

\n

On the below example this means that 'ALT' key is pressed down, then '+' and 's' keys are pressed\ndown before they are all released up.

\n
\nSend Keys   keys={Alt}(+s)\n
\n

Mouse clicks can be executed with keywords specific for a type of a click, e.g. Click (normal click),\nDouble Click and Right Click.

\n

How to inspect

\n

Most common, and recommended by Microsoft, inspector tool for Windows is Accessibility Insights that\ncan be installed separately. Other options are tools Inspect Object and UI Automation Verify, which\ncan be accessed by installing Windows SDK.

\n

A more programmatic approach is to run Print Tree    log_as_warnings=${True}\nkeyword and then observe in the logs the found elements structure starting from\nDesktop (or the currently set anchor / active window) as root. (refer to keyword's\ndocumentation\nfor more details)

\n

Recording

\n

The package provides some rudimentary inspecting and recording via the\nwindows-record script, which can be started through the command line (in an\nenvironment containing the rpaframework-windows installation).

\n

Recording inspects elements on mouse click and can be stopped by pressing the\nESC key. Expected console output:

\n
\nC:\\Users\\User\\robots\\> windows-record -v  # or > python -m RPA.Windows -v\nMouse recording started. Use ESC to stop recording.\n\n--------------------------------------------------------------------------------\nCopy-paste the code below into your *** Tasks *** or *** Keywords ***\n--------------------------------------------------------------------------------\n\nControl Window    name:Calculator and type:WindowControl and class:ApplicationFrameWindow  # handle:9569486\nClick    name:Calculator and id:TitleBar and type:WindowControl and class:ApplicationFrameTitleBarWindow and path:1\nClick    name:"Display is 0" and id:CalculatorResults and type:TextControl and path:2|3|2|2\nClick    name:Eight and id:num8Button and type:ButtonControl and class:Button and path:2|3|2|8|9\nClick    name:Nine and id:num9Button and type:ButtonControl and class:Button and path:2|3|2|8|10\nClick    name:Clear and id:clearButton and type:ButtonControl and class:Button and path:2|3|2|5|3\n\n--------------------------------------------------------------------------------\n
\n

Check our Portal example in order to learn more abot the path: strategy in\nlocators and how to record elements displaying their paths:\nhttps://robocorp.com/portal/robot/robocorp/example-windows-element-path

\n

Video recorded demo on how to run the recorder script from VSCode:\nhttps://www.loom.com/share/2807372359f34b9cbe1bc2df9194ec68

\n

Caveats

\n
    \n
  • Make sure your display scaling is set to 100%, otherwise you might encounter\nissues when clicking or interacting with elements. (since offsets and coordinates\nget distorted)
  • \n
  • Disturbing the automation (like interacting with your mouse/keyboard) or having\nother apps obstructing the process interacting with your app of interest will\nmost probably affect the expected behaviour. In order to avoid this, try\ncontrolling the app's main window right before sending clicks or keys. And keep\ntargeting elements through string locators, as interacting with Windows\nelement objects previously retrieved will not work as expected in a future\naltered state of the app (changes under the element structure).
  • \n
\n

Example: Robot Framework

\n

The library must be imported first.

\n
\n*** Settings ***\nLibrary    RPA.Windows\n
\n

Windows Calculator automation task

\n
\n*** Tasks ***\nDo some calculations\n    [Setup]  Windows Run   calc.exe\n\n    Control Window    name:Calculator\n    Click    id:clearButton\n    Send Keys   keys=96+4=\n    ${result} =    Get Attribute    id:CalculatorResults    Name\n    Log To Console    ${result}\n\n    @{buttons} =  Get Elements  type:Group and name:"Number pad" > type:Button\n    FOR  ${button}  IN  @{buttons}\n        Log To Console   ${button}\n    END\n\n    [Teardown]   Close Current Window\n
\n

Example: Python

\n
\nfrom RPA.Windows import Windows\n\nlibrary = Windows()\n\ndef test_do_some_calculations():\n    library.windows_run("calc.exe")\n    try:\n        library.control_window("name:Calculator")\n        library.click("id:clearButton")\n        library.send_keys(keys="96+4=")\n        result = library.get_attribute("id:CalculatorResults", "Name")\n        print(result)\n        buttons = library.get_elements(\n            'type:Group and name:"Number pad" > type:Button'\n        )\n        for button in buttons:\n            print(button)\n    finally:\n        library.close_current_window()\n
\n
\n", "version": "", - "generated": "2023-11-10 12:47:53", + "generated": "2023-11-10 12:56:12", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -82160,10 +82160,10 @@ "typedocs": { "None": "None" }, - "defaultValue": "", + "defaultValue": "", "kind": "POSITIONAL_OR_NAMED", "required": false, - "repr": "validator: Callable | None = " + "repr": "validator: Callable | None = " } ], "doc": "
\n

Set value of the element defined by the locator.

\n

Note: An anchor will work only on element structures where you can\nrely on the stability of that root/child element tree, as remaining the same.\nUsually these kind of structures are tables. (but not restricted to)

\n

Note: It is important to set append=${True} if you want to keep the\ncurrent text in the element. Other option is to read the current text into a\nvariable, then modify that value as you wish and pass it to the Set Value\nkeyword for a complete text replacement. (without setting the append flag)

\n

The following exceptions may be raised:

\n
\n
    \n
  • ActionNotPossible if the element does not allow the SetValue action\nto be run on it nor having send_keys_fallback=${True}.
  • \n
  • ValueError if the new value to be set can't be set correctly.
  • \n
\n
\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
param locator:String locator or element object.
param value:String value to be set.
param append:False for setting the value, True for appending it. (OFF by\ndefault)
param enter:Set it to True to press the Enter key at the end of the\ninput. (nothing is pressed by default)
param newline:Set it to True to add a new line at the end of the value. (no\nEOL included by default; this won't work with send_keys_fallback enabled)
param send_keys_fallback:
 Tries to set the value by sending it through keys\nif the main way of setting it fails. (enabled by default)
param validator:
 Function receiving two parameters post-setting, the expected\nand the current value, which returns True if the two values match. (by\ndefault, the keyword will raise if the values are different, set this to\nNone to disable validation or pass your custom function instead)
returns:The element object identified through the passed locator.
\n

Example: Robot Framework

\n
\n*** Tasks ***\nSet Values In Notepad\n    Set Value   type:DataItem name:column1   ab c  # Set value to "ab c"\n    # Press ENTER after setting the value.\n    Set Value    type:Edit name:"File name:"    console.txt   enter=${True}\n\n    # Add newline (manually) at the end of the string. (Notepad example)\n    Set Value    name:"Text Editor"  abc\\n\n    # Add newline with parameter.\n    Set Value    name:"Text Editor"  abc   newline=${True}\n\n    # Clear Notepad window and start appending text.\n    Set Anchor  name:"Text Editor"\n    # All the following keyword calls will use the anchor element as a\n    #  starting point, UNLESS they specify a locator explicitly or\n    #  Clear Anchor is used.\n    ${time} =    Get Time\n    # Clears with append=${False}. (default)\n    Set Value    value=The time now is ${time}\n    # Append text and add a newline at the end.\n    Set Value    value= and it's the task run time.   append=${True}\n    ...    newline=${True}\n    # Continue appending and ensure a new line at the end by pressing\n    #  the Enter key this time.\n    Set Value    value=But this will appear on the 2nd line now.\n    ...    append=${True}   enter=${True}   validator=${None}\n
\n

Example: Python

\n
\nfrom RPA.Windows import Windows\n\nlib_win = Windows()\nlocator = "Document - WordPad > Rich Text Window"\nelem = lib_win.set_value(locator, value="My text", send_keys_fallback=True)\ntext = lib_win.get_value(elem)\nprint(text)\n
\n
\n", @@ -82420,7 +82420,7 @@ "name": "RPA.Word.Application", "doc": "
\n

Word.Application is a library for controlling the Word application.

\n

Examples

\n

Robot Framework

\n
\n*** Settings ***\nLibrary                 RPA.Word.Application\nTask Setup              Open Application\nSuite Teardown          Quit Application\n\n*** Tasks ***\nOpen existing file\n    Open File           old.docx\n    Write Text          Extra Line Text\n    Write Text          Another Extra Line of Text\n    Save Document AS    ${CURDIR}${/}new.docx\n    ${texts}=           Get all Texts\n    Close Document\n
\n

Python

\n
\nfrom RPA.Word.Application import Application\n\napp = Application()\napp.open_application()\napp.open_file('old.docx')\napp.write_text('Extra Line Text')\napp.save_document_as('new.docx')\napp.quit_application()\n
\n

Caveats

\n

This library works on a Windows operating system with UI enabled only, and you must\nensure that you open the app first with Open Application before running any\nother relevant keyword which requires to operate on an open app. The application is\nautomatically closed at the end of the task execution, so this can be changed by\nimporting the library with the autoexit=${False} setting.

\n
\n*** Settings ***\nLibrary     RPA.Excel|Outlook|Word.Application    autoexit=${False}\n
\n

If you're running the Process by Control Room through a custom self-hosted Worker\nservice, then please make sure that you enable an RDP session by ticking "Use\nDesktop Connection" under the Step configuration.

\n

If you still encounter issues with opening a document, please ensure that file can\nbe opened first manually and dismiss any alert potentially blocking the process.

\n

Check the documentation below for more info:

\n\n
\n", "version": "", - "generated": "2023-11-10 12:47:53", + "generated": "2023-11-10 12:56:12", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", @@ -82836,8 +82836,8 @@ "specversion": 1, "name": "RPA.version", "doc": "

Documentation for library RPA.version.

", - "version": "27.5.2", - "generated": "2023-11-10 12:47:53", + "version": "27.5.3", + "generated": "2023-11-10 12:56:12", "type": "LIBRARY", "scope": "GLOBAL", "docFormat": "HTML", diff --git a/libdoc/RPA_Archive.html b/libdoc/RPA_Archive.html index 7bcb36c5d2..add578d9e3 100644 --- a/libdoc/RPA_Archive.html +++ b/libdoc/RPA_Archive.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Assistant.html b/libdoc/RPA_Assistant.html index 8abd12785c..940ccc8bcc 100644 --- a/libdoc/RPA_Assistant.html +++ b/libdoc/RPA_Assistant.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Browser.html b/libdoc/RPA_Browser.html index 6c4a8b0dde..abf54e220b 100644 --- a/libdoc/RPA_Browser.html +++ b/libdoc/RPA_Browser.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Browser_Playwright.html b/libdoc/RPA_Browser_Playwright.html index 8d98787780..dbf35237f8 100644 --- a/libdoc/RPA_Browser_Playwright.html +++ b/libdoc/RPA_Browser_Playwright.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Browser_Selenium.html b/libdoc/RPA_Browser_Selenium.html index 22c1351c4e..3558cf83f6 100644 --- a/libdoc/RPA_Browser_Selenium.html +++ b/libdoc/RPA_Browser_Selenium.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Browser_common.html b/libdoc/RPA_Browser_common.html index a9c6c900f5..131601f992 100644 --- a/libdoc/RPA_Browser_common.html +++ b/libdoc/RPA_Browser_common.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Calendar.html b/libdoc/RPA_Calendar.html index e847ad1bf2..916f66b90e 100644 --- a/libdoc/RPA_Calendar.html +++ b/libdoc/RPA_Calendar.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Cloud_AWS.html b/libdoc/RPA_Cloud_AWS.html index daf28d3c13..8be1cea34a 100644 --- a/libdoc/RPA_Cloud_AWS.html +++ b/libdoc/RPA_Cloud_AWS.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Cloud_Azure.html b/libdoc/RPA_Cloud_Azure.html index 1ffbdaa6d1..c151f4b220 100644 --- a/libdoc/RPA_Cloud_Azure.html +++ b/libdoc/RPA_Cloud_Azure.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Cloud_Google.html b/libdoc/RPA_Cloud_Google.html index 48a753c9fe..75f5ad43af 100644 --- a/libdoc/RPA_Cloud_Google.html +++ b/libdoc/RPA_Cloud_Google.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Crypto.html b/libdoc/RPA_Crypto.html index c5fce2efa4..8ab5bc1db2 100644 --- a/libdoc/RPA_Crypto.html +++ b/libdoc/RPA_Crypto.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Database.html b/libdoc/RPA_Database.html index d8b6329675..c921f295a0 100644 --- a/libdoc/RPA_Database.html +++ b/libdoc/RPA_Database.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Desktop.html b/libdoc/RPA_Desktop.html index 0f2293bf65..b432e227e9 100644 --- a/libdoc/RPA_Desktop.html +++ b/libdoc/RPA_Desktop.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Desktop_Clipboard.html b/libdoc/RPA_Desktop_Clipboard.html index 24509189ea..b156f7eeed 100644 --- a/libdoc/RPA_Desktop_Clipboard.html +++ b/libdoc/RPA_Desktop_Clipboard.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Desktop_OperatingSystem.html b/libdoc/RPA_Desktop_OperatingSystem.html index 25f25c324f..30ec6ec1a3 100644 --- a/libdoc/RPA_Desktop_OperatingSystem.html +++ b/libdoc/RPA_Desktop_OperatingSystem.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Desktop_Windows.html b/libdoc/RPA_Desktop_Windows.html index e518bcb6d0..5f263cdec1 100644 --- a/libdoc/RPA_Desktop_Windows.html +++ b/libdoc/RPA_Desktop_Windows.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Dialogs.html b/libdoc/RPA_Dialogs.html index 108217d02f..0fac0cbcfc 100644 --- a/libdoc/RPA_Dialogs.html +++ b/libdoc/RPA_Dialogs.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_DocumentAI.html b/libdoc/RPA_DocumentAI.html index 45724d1764..6986c37980 100644 --- a/libdoc/RPA_DocumentAI.html +++ b/libdoc/RPA_DocumentAI.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_DocumentAI_Base64AI.html b/libdoc/RPA_DocumentAI_Base64AI.html index 778101ceee..9f2efd1df6 100644 --- a/libdoc/RPA_DocumentAI_Base64AI.html +++ b/libdoc/RPA_DocumentAI_Base64AI.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_DocumentAI_DocumentAI.html b/libdoc/RPA_DocumentAI_DocumentAI.html index 982dbf1078..9451cf5eac 100644 --- a/libdoc/RPA_DocumentAI_DocumentAI.html +++ b/libdoc/RPA_DocumentAI_DocumentAI.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_DocumentAI_Nanonets.html b/libdoc/RPA_DocumentAI_Nanonets.html index da7ddf58e1..139be042d0 100644 --- a/libdoc/RPA_DocumentAI_Nanonets.html +++ b/libdoc/RPA_DocumentAI_Nanonets.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Email_Exchange.html b/libdoc/RPA_Email_Exchange.html index 7d414db5a9..b299613d7c 100644 --- a/libdoc/RPA_Email_Exchange.html +++ b/libdoc/RPA_Email_Exchange.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Email_ImapSmtp.html b/libdoc/RPA_Email_ImapSmtp.html index 97b120ab9b..25ae0ef0fa 100644 --- a/libdoc/RPA_Email_ImapSmtp.html +++ b/libdoc/RPA_Email_ImapSmtp.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Excel_Application.html b/libdoc/RPA_Excel_Application.html index 90256e7fd6..f6848011b3 100644 --- a/libdoc/RPA_Excel_Application.html +++ b/libdoc/RPA_Excel_Application.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Excel_Files.html b/libdoc/RPA_Excel_Files.html index 429e1e27a4..d945e5dc62 100644 --- a/libdoc/RPA_Excel_Files.html +++ b/libdoc/RPA_Excel_Files.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_FTP.html b/libdoc/RPA_FTP.html index 39081c2795..179e4de4de 100644 --- a/libdoc/RPA_FTP.html +++ b/libdoc/RPA_FTP.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_FileSystem.html b/libdoc/RPA_FileSystem.html index f15aa9db67..f219cb5fb8 100644 --- a/libdoc/RPA_FileSystem.html +++ b/libdoc/RPA_FileSystem.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_HTTP.html b/libdoc/RPA_HTTP.html index 9ad6c397f7..39a8553d4f 100644 --- a/libdoc/RPA_HTTP.html +++ b/libdoc/RPA_HTTP.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Hubspot.html b/libdoc/RPA_Hubspot.html index 6106f0358e..cc4587a1ee 100644 --- a/libdoc/RPA_Hubspot.html +++ b/libdoc/RPA_Hubspot.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Images.html b/libdoc/RPA_Images.html index be5da21248..8680ad29a8 100644 --- a/libdoc/RPA_Images.html +++ b/libdoc/RPA_Images.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_JSON.html b/libdoc/RPA_JSON.html index f74250e99f..d67ee7274a 100644 --- a/libdoc/RPA_JSON.html +++ b/libdoc/RPA_JSON.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_JavaAccessBridge.html b/libdoc/RPA_JavaAccessBridge.html index eb5c9b2fca..0fb7577f1c 100644 --- a/libdoc/RPA_JavaAccessBridge.html +++ b/libdoc/RPA_JavaAccessBridge.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_MFA.html b/libdoc/RPA_MFA.html index 01fa38fdac..a8b52d8df5 100644 --- a/libdoc/RPA_MFA.html +++ b/libdoc/RPA_MFA.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_MSGraph.html b/libdoc/RPA_MSGraph.html index d2b7de5b8d..db26858b98 100644 --- a/libdoc/RPA_MSGraph.html +++ b/libdoc/RPA_MSGraph.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Netsuite.html b/libdoc/RPA_Netsuite.html index 3ef8afefdb..8249f903d5 100644 --- a/libdoc/RPA_Netsuite.html +++ b/libdoc/RPA_Netsuite.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Notifier.html b/libdoc/RPA_Notifier.html index 9fae1c2af5..0673982ae4 100644 --- a/libdoc/RPA_Notifier.html +++ b/libdoc/RPA_Notifier.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_OpenAI.html b/libdoc/RPA_OpenAI.html index 884a0f27b1..2db2fd7879 100644 --- a/libdoc/RPA_OpenAI.html +++ b/libdoc/RPA_OpenAI.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Outlook_Application.html b/libdoc/RPA_Outlook_Application.html index 68bffee3d5..3e437f8f75 100644 --- a/libdoc/RPA_Outlook_Application.html +++ b/libdoc/RPA_Outlook_Application.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_PDF.html b/libdoc/RPA_PDF.html index 7b57b485b2..ceb1d59e48 100644 --- a/libdoc/RPA_PDF.html +++ b/libdoc/RPA_PDF.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Robocloud_Items.html b/libdoc/RPA_Robocloud_Items.html index 4985eaf279..1cbb5945ee 100644 --- a/libdoc/RPA_Robocloud_Items.html +++ b/libdoc/RPA_Robocloud_Items.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Robocloud_Secrets.html b/libdoc/RPA_Robocloud_Secrets.html index 3bd86c18fe..194c223f89 100644 --- a/libdoc/RPA_Robocloud_Secrets.html +++ b/libdoc/RPA_Robocloud_Secrets.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Robocorp_Process.html b/libdoc/RPA_Robocorp_Process.html index a26677b1ea..ff6b05d5ac 100644 --- a/libdoc/RPA_Robocorp_Process.html +++ b/libdoc/RPA_Robocorp_Process.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Robocorp_Storage.html b/libdoc/RPA_Robocorp_Storage.html index c1c68caa74..5789d78a25 100644 --- a/libdoc/RPA_Robocorp_Storage.html +++ b/libdoc/RPA_Robocorp_Storage.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Robocorp_Vault.html b/libdoc/RPA_Robocorp_Vault.html index 0f13dfcef3..3ccd29caa5 100644 --- a/libdoc/RPA_Robocorp_Vault.html +++ b/libdoc/RPA_Robocorp_Vault.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Robocorp_WorkItems.html b/libdoc/RPA_Robocorp_WorkItems.html index a587b730ae..4ceb550ba0 100644 --- a/libdoc/RPA_Robocorp_WorkItems.html +++ b/libdoc/RPA_Robocorp_WorkItems.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_RobotLogListener.html b/libdoc/RPA_RobotLogListener.html index df992528b9..a57cd99273 100644 --- a/libdoc/RPA_RobotLogListener.html +++ b/libdoc/RPA_RobotLogListener.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_SAP.html b/libdoc/RPA_SAP.html index b7dd8e3a88..5046154456 100644 --- a/libdoc/RPA_SAP.html +++ b/libdoc/RPA_SAP.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Salesforce.html b/libdoc/RPA_Salesforce.html index af0f98726d..2f285a1def 100644 --- a/libdoc/RPA_Salesforce.html +++ b/libdoc/RPA_Salesforce.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Slack.html b/libdoc/RPA_Slack.html index 94585c0273..80a323b923 100644 --- a/libdoc/RPA_Slack.html +++ b/libdoc/RPA_Slack.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Smartsheet.html b/libdoc/RPA_Smartsheet.html index 88965c2c57..6d1faa08e5 100644 --- a/libdoc/RPA_Smartsheet.html +++ b/libdoc/RPA_Smartsheet.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Tables.html b/libdoc/RPA_Tables.html index bafb404f3c..254cb54dc4 100644 --- a/libdoc/RPA_Tables.html +++ b/libdoc/RPA_Tables.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Tasks.html b/libdoc/RPA_Tasks.html index 784073f6ef..3615b2f559 100644 --- a/libdoc/RPA_Tasks.html +++ b/libdoc/RPA_Tasks.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Twitter.html b/libdoc/RPA_Twitter.html index 6494fb9bcc..2aeba19564 100644 --- a/libdoc/RPA_Twitter.html +++ b/libdoc/RPA_Twitter.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Windows.html b/libdoc/RPA_Windows.html index 6ae1f39e93..e91279ca05 100644 --- a/libdoc/RPA_Windows.html +++ b/libdoc/RPA_Windows.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_Word_Application.html b/libdoc/RPA_Word_Application.html index 27d6593a83..a1e2d954ce 100644 --- a/libdoc/RPA_Word_Application.html +++ b/libdoc/RPA_Word_Application.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libdoc/RPA_version.html b/libdoc/RPA_version.html index 538a82a596..2c0cb5729c 100644 --- a/libdoc/RPA_version.html +++ b/libdoc/RPA_version.html @@ -1019,7 +1019,7 @@ //# sourceMappingURL=iframeResizer.contentWindow.map diff --git a/libraries/http/python.html b/libraries/http/python.html index 7b8c26309d..eb43cda9af 100644 --- a/libraries/http/python.html +++ b/libraries/http/python.html @@ -172,7 +172,7 @@

HTTP* On Session.

-DEFAULT_RETRY_METHOD_LIST = ['HEAD', 'TRACE', 'GET', 'DELETE', 'OPTIONS', 'PUT']
+DEFAULT_RETRY_METHOD_LIST = ['DELETE', 'PUT', 'TRACE', 'GET', 'OPTIONS', 'HEAD']
@@ -211,7 +211,7 @@

HTTP
-create_client_cert_session(alias, url, headers={}, cookies={}, client_certs=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['HEAD', 'TRACE', 'GET', 'DELETE', 'OPTIONS', 'PUT'])
+create_client_cert_session(alias, url, headers={}, cookies={}, client_certs=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['DELETE', 'PUT', 'TRACE', 'GET', 'OPTIONS', 'HEAD'])

Create Session: create a HTTP session to a server

url Base url of the server

alias Robot Framework alias to identify the session

@@ -249,7 +249,7 @@

HTTP
-create_custom_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['HEAD', 'TRACE', 'GET', 'DELETE', 'OPTIONS', 'PUT'])
+create_custom_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['DELETE', 'PUT', 'TRACE', 'GET', 'OPTIONS', 'HEAD'])

Create Session: create a HTTP session to a server

url Base url of the server

alias Robot Framework alias to identify the session

@@ -290,7 +290,7 @@

HTTP
-create_digest_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['HEAD', 'TRACE', 'GET', 'DELETE', 'OPTIONS', 'PUT'])
+create_digest_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['DELETE', 'PUT', 'TRACE', 'GET', 'OPTIONS', 'HEAD'])

Create Session: create a HTTP session to a server

url Base url of the server

alias Robot Framework alias to identify the session

@@ -328,7 +328,7 @@

HTTP
-create_ntlm_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['HEAD', 'TRACE', 'GET', 'DELETE', 'OPTIONS', 'PUT'])
+create_ntlm_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['DELETE', 'PUT', 'TRACE', 'GET', 'OPTIONS', 'HEAD'])

Create Session: create a HTTP session to a server

url Base url of the server

alias Robot Framework alias to identify the session

@@ -366,7 +366,7 @@

HTTP
-create_session(alias, url, headers={}, cookies={}, auth=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['HEAD', 'TRACE', 'GET', 'DELETE', 'OPTIONS', 'PUT'])
+create_session(alias, url, headers={}, cookies={}, auth=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['DELETE', 'PUT', 'TRACE', 'GET', 'OPTIONS', 'HEAD'])

Create Session: create a HTTP session to a server

alias Robot Framework alias to identify the session

url Base url of the server

diff --git a/python.json b/python.json index 441e782498..accf5a18d5 100644 --- a/python.json +++ b/python.json @@ -93,7 +93,7 @@ }, { "module": "RPA.HTTP.HTTP", - "markdown": "# Python API\n\n## HTTP\n\n### *class* RPA.HTTP.HTTP(\\*args, \\*\\*kwargs)\n\nThe *RPA.HTTP* library extends functionality of the [RequestsLibrary](https://marketsquare.github.io/robotframework-requests/doc/RequestsLibrary.html).\nSee that documentation for several examples of how to issue `GET`\nrequests and utilize the returned `result` objects.\n\nThis extension provides helper keywords to get an HTTP resource on a\nsession. The `HTTP Get` and `Download` keywords will initiate a\nsession if one does not exist for the provided URL, or use an existing\nsession. When using these keywords, you do not need to manage\nsessions with `Create Session`. Session management is still\nrequired if you use the underlying session keywords, e.g.,\n`* On Session`.\n\n#### DEFAULT_RETRY_METHOD_LIST *= ['OPTIONS', 'PUT', 'HEAD', 'TRACE', 'DELETE', 'GET']*\n\n#### ROBOT_LIBRARY_DOC_FORMAT *= 'reST'*\n\n#### ROBOT_LIBRARY_SCOPE *= 'GLOBAL'*\n\n#### check_vulnerabilities()\n\nCheck for possible vulnerabilities in the installed runtime\nenvironment packages.\n\nCurrently will check only for OpenSSL version and outputs warning message on any\ndiscovered vulnerability.\n\n* **Returns:**\n list of all check results\n\n```robotframework\n*** Tasks ***\nVulnerability Check\n ${results}= Check Vulnerabilities\n FOR ${result} IN @{results}\n Log To Console TYPE: ${result}[type]\n Log To Console VULNERABLE: ${result}[vulnerable]\n Log To Console MESSAGE: ${result}[message]\n END\n```\n\n#### create_client_cert_session(alias, url, headers={}, cookies={}, client_certs=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['OPTIONS', 'PUT', 'HEAD', 'TRACE', 'DELETE', 'GET'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`client_certs` [\u2018client certificate\u2019, \u2018client key\u2019] PEM files containing the client key and certificate\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_custom_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['OPTIONS', 'PUT', 'HEAD', 'TRACE', 'DELETE', 'GET'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` A Custom Authentication object to be passed on to the requests library.\n: [http://docs.python-requests.org/en/master/user/advanced/#custom-authentication](http://docs.python-requests.org/en/master/user/advanced/#custom-authentication)\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_digest_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['OPTIONS', 'PUT', 'HEAD', 'TRACE', 'DELETE', 'GET'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` [\u2018DOMAIN\u2019, \u2018username\u2019, \u2018password\u2019] for NTLM Authentication\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_ntlm_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['OPTIONS', 'PUT', 'HEAD', 'TRACE', 'DELETE', 'GET'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` [\u2018DOMAIN\u2019, \u2018username\u2019, \u2018password\u2019] for NTLM Authentication\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_session(alias, url, headers={}, cookies={}, auth=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['OPTIONS', 'PUT', 'HEAD', 'TRACE', 'DELETE', 'GET'])\n\nCreate Session: create a HTTP session to a server\n\n`alias` Robot Framework alias to identify the session\n\n`url` Base url of the server\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` List of username & password for HTTP Basic Auth\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### delete_all_sessions()\n\nRemoves all the session objects\n\n#### delete_on_session(\\*\\*kwargs)\n\nSends a DELETE request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### delete_request(alias, uri, data=None, json=None, params=None, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use DELETE On Session instead.\n\nSend a DELETE request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the DELETE request to\n\n`json` a value that will be json encoded\n: and sent as request data if data is not specified\n\n`headers` a dictionary of headers to use with the request\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`timeout` connection timeout\n\n#### download(url: str, target_file: Optional[str] = None, verify: Union[bool, str] = True, force_new_session: bool = False, overwrite: bool = False, stream: bool = False, \\*\\*kwargs)\n\nAn alias for the `HTTP Get` keyword.\n\nThe difference in use is that the URL is always downloaded based on\nthe URL path (even without `target_file`). If there is a filename\nin the path, then that is used as `target_file` to save to. By default,\nthe filename will be \u201cdownloaded.html\u201d.\n\n```robotframework\n*** Settings ***\nLibrary RPA.HTTP\n\n*** Variables ***\n${DOWNLOAD_PATH}= ${OUTPUT DIR}${/}downloads\n${WORD_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file-sample_100kB.doc\n${EXCEL_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file_example_XLS_10.xls\n\n*** Tasks ***\nDownload files with reused session with provided file names\n # Starts a new session\n Download ${WORD_EXAMPLE} target_file=${DOWNLOAD_PATH}\n # Uses the previous session\n Download ${EXCEL_EXAMPLE} target_file=${DOWNLOAD_PATH}\n # Above files are downloaded using the same names as they have\n # on the remote server.\n```\n\n* **Parameters:**\n * **url** \u2013 target URL for GET request\n * **target_file** \u2013 filepath to save request content, default `None`\n * **verify** \u2013 if SSL verification should be done, default `True`,\n a CA_BUNDLE path can also be provided\n * **force_new_session** \u2013 if new HTTP session should be created,\n default `False`\n * **overwrite** \u2013 used together with `target_file`, if `True` will overwrite\n the target file, default `False`\n * **stream** \u2013 if `False` the response content will be immediately downloaded\n* **Returns:**\n request response as a dict\n\n#### get_current_session_alias()\n\nGet request session alias that was used with the `HTTP Get` keyword.\n\n* **Returns:**\n name of session alias as a string\n\n#### *static* get_file_for_streaming_upload(path)\n\nOpens and returns a file descriptor of a specified file to be passed as `data` parameter\nto other requests keywords.\n\nThis allows streaming upload of large files without reading them into memory.\n\nFile descriptor is binary mode and read only. Requests keywords will automatically close the file,\nif used outside this library it\u2019s up to the caller to close it.\n\n#### get_on_session(\\*\\*kwargs)\n\nSends a GET request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to retrieve the resource is the `url`, while query\nstring parameters can be passed as string, dictionary (or list of tuples or bytes)\nthrough the `params`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### get_request(alias, uri, headers=None, data=None, json=None, params=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use GET On Session instead.\n\nSend a GET request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the GET request to\n\n`params` url parameters to append to the uri\n\n`headers` a dictionary of headers to use with the request\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as GET data\n or binary data that is sent as the raw body content\n\n`json` a value that will be json encoded\n: and sent as GET data if data is not specified\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`timeout` connection timeout\n\n#### head_on_session(\\*\\*kwargs)\n\nSends a HEAD request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to retrieve the HTTP headers is the `url`.\n\n`allow_redirects` parameter is not provided, it will be set to False (as\nopposed to the default behavior).\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### head_request(alias, uri, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use HEAD On Session instead.\n\nSend a HEAD request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the HEAD request to\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`headers` a dictionary of headers to use with the request\n\n`timeout` connection timeout\n\n#### http_get(url: str, target_file: Optional[str] = None, verify: Union[bool, str] = True, force_new_session: bool = False, overwrite: bool = False, stream: bool = False, \\*\\*kwargs)\n\nA helper method for `Get Request` that will create a session, perform GET\nrequest, and store the target file, if set by the `target_file` parameter.\n\nThe old session will be used if the URL scheme and the host are the same as\npreviously, e.g., \u2018[https://www.google.fi](https://www.google.fi)\u2019 part of the URL.\n\n```robotframework\n*** Settings ***\nLibrary RPA.HTTP\n\n*** Variables ***\n${DOWNLOAD_PATH}= ${OUTPUT DIR}${/}downloads\n${WORD_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file-sample_100kB.doc\n${EXCEL_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file_example_XLS_10.xls\n\n*** Tasks ***\nDownload files with reused session\n # Starts a new session\n HTTP Get ${WORD_EXAMPLE} target_file=${DOWNLOAD_PATH}${/}word-example.doc\n # Uses the previous session\n HTTP Get ${EXCEL_EXAMPLE} target_file=${DOWNLOAD_PATH}${/}excel-example.xls\n```\n\n* **Parameters:**\n * **url** \u2013 target URL for GET request\n * **target_file** \u2013 filepath to save request content, default `None`\n * **verify** \u2013 if SSL verification should be done, default `True`,\n a CA_BUNDLE path can also be provided\n * **force_new_session** \u2013 if new HTTP session should be created,\n default `False`\n * **overwrite** \u2013 used together with `target_file`, if `True` will overwrite\n the target file, default `False`\n * **stream** \u2013 if `False`, the response content will be immediately downloaded\n* **Returns:**\n request response as a dict\n\n#### options_on_session(\\*\\*kwargs)\n\nSends a OPTIONS request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to retrieve the resource is the `url`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### options_request(alias, uri, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use OPTIONS On Session instead.\n\nSend an OPTIONS request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the OPTIONS request to\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`headers` a dictionary of headers to use with the request\n\n`timeout` connection timeout\n\n#### patch_on_session(\\*\\*kwargs)\n\nSends a PATCH request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### patch_request(alias, uri, data=None, json=None, params=None, headers=None, files=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use PATCH On Session instead.\n\nSend a PATCH request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the PATCH request to\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as PATCH data\n or binary data that is sent as the raw body content\n or file descriptor retrieved by Get File For Streaming Upload\n\n`json` a value that will be json encoded\n: and sent as PATCH data if data is not specified\n\n`headers` a dictionary of headers to use with the request\n\n`files` a dictionary of file names containing file data to PATCH to the server\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`params` url parameters to append to the uri\n\n`timeout` connection timeout\n\n#### post_on_session(\\*\\*kwargs)\n\nSends a POST request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### post_request(alias, uri, data=None, json=None, params=None, headers=None, files=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use POST On Session instead.\n\nSend a POST request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the POST request to\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as POST data\n or binary data that is sent as the raw body content\n or passed as such for multipart form data if `files` is also defined\n or file descriptor retrieved by Get File For Streaming Upload\n\n`json` a value that will be json encoded\n: and sent as POST data if files or data is not specified\n\n`params` url parameters to append to the uri\n\n`headers` a dictionary of headers to use with the request\n\n`files` a dictionary of file names containing file data to POST to the server\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`timeout` connection timeout\n\n#### put_on_session(\\*\\*kwargs)\n\nSends a PUT request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### put_request(alias, uri, data=None, json=None, params=None, files=None, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use PUT On Session instead.\n\nSend a PUT request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the PUT request to\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as PUT data\n or binary data that is sent as the raw body content\n or file descriptor retrieved by Get File For Streaming Upload\n\n`json` a value that will be json encoded\n: and sent as PUT data if data is not specified\n\n`headers` a dictionary of headers to use with the request\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`params` url parameters to append to the uri\n\n`timeout` connection timeout\n\n#### request_should_be_successful(response=None)\n\nFails if response status code is a client or server error (4xx, 5xx).\n\n`response` is the output of other requests keywords like GET On Session.\nIf omitted the last response will be used.\n\nIn case of failure an HTTPError will be automatically raised.\n\nFor a more versatile assert keyword see Status Should Be.\n\n#### session_exists(alias)\n\nReturn True if the session has been already created\n\n`alias` that has been used to identify the Session object in the cache\n\n#### session_less_delete(\\*\\*kwargs)\n\nSends a DELETE request.\n\nThe endpoint used to send the request is the `url` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_get(\\*\\*kwargs)\n\nSends a GET request.\n\nThe endpoint used to retrieve the resource is the `url`, while query\nstring parameters can be passed as string, dictionary (or list of tuples or bytes)\nthrough the `params`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs` here is a list:\n\n`data` | Dictionary, list of tuples, bytes, or file-like object to send in the body of the request. |\n
\n`json` | A JSON serializable Python object to send in the body of the request. |\n
\n`headers` | Dictionary of HTTP Headers to send with the request. |\n
\n`cookies` | Dict or CookieJar object to send with the request. |\n
\n`files` | Dictionary of file-like-objects (or `{'name': file-tuple}`) for multipart encoding upload. |\n
\n`file-tuple` | can be a 2-tuple `('filename', fileobj)`, 3-tuple `('filename', fileobj, 'content_type')` or a 4-tuple `('filename', fileobj, 'content_type', custom_headers)`, where `'content-type'` is a string defining the content type of the given file and `custom_headers` a dict-like object containing additional headers to add for the file. |\n
\n`auth` | Auth tuple to enable Basic/Digest/Custom HTTP Auth. |\n
\n`timeout` | How many seconds to wait for the server to send data before giving up, as a float, or a `(connect timeout, read timeout)` tuple. |\n
\n`allow_redirects` | Boolean. Enable/disable GET/OPTIONS/POST/PUT/PATCH/DELETE/HEAD redirection. Defaults to `${True}`. |\n
\n`proxies` | Dictionary mapping protocol or protocol and host to the URL of the proxy (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019}) |\n
\n`verify` | Either a boolean, in which case it controls whether we verify the server\u2019s TLS certificate, or a string, in which case it must be a path to a CA bundle to use. Defaults to `${True}`. Warning: if a session has been created with `verify=${False}` any other requests will not verify the SSL certificate. |\n
\n`stream` | if `${False}`, the response content will be immediately downloaded. |\n
\n`cert` | if String, path to ssl client cert file (.pem). If Tuple, (\u2018cert\u2019, \u2018key\u2019) pair. |\n
\n\nFor more updated and complete information verify the official Requests api documentation:\n[https://requests.readthedocs.io/en/latest/api/](https://requests.readthedocs.io/en/latest/api/)\n\n#### session_less_head(\\*\\*kwargs)\n\nSends a HEAD request.\n\nThe endpoint used to retrieve the HTTP headers is the `url`.\n\n`allow_redirects` parameter is not provided, it will be set to False (as\nopposed to the default behavior).\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_options(\\*\\*kwargs)\n\nSends a OPTIONS request.\n\nThe endpoint used to retrieve the resource is the `url`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_patch(\\*\\*kwargs)\n\nSends a PUT request.\n\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_post(\\*\\*kwargs)\n\nSends a POST request.\n\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_put(\\*\\*kwargs)\n\nSends a PUT request.\n\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### status_should_be(expected_status, response=None, msg=None)\n\nFails if response status code is different than the expected.\n\n`expected_status` could be the code number as an integer or as string.\nBut it could also be a named status code like \u2018ok\u2019, \u2018created\u2019, \u2018accepted\u2019 or\n\u2018bad request\u2019, \u2018not found\u2019 etc.\n\n`response` is the output of other requests keywords like GET or GET On Session.\nIf omitted the last response will be used.\n\nIn case of failure an HTTPError will be automatically raised.\nA custom failure message `msg` can be added like in built-in keywords.\n\nNew requests keywords like GET or GET On Session (starting from 0.8 version) already have an implicit assert\nmechanism that, by default, verifies the response status code.\nStatus Should Be keyword can be useful when you disable implicit assert using `expected_status=anything`.\n\nFor example when you have a nested keyword that is used for both OK and ERROR responses:\n\n**\\* Test Cases \\***\n
\n
\nTest Get Request And Make Sure Is A 404 Response\n
\n${resp}= GET Custom Keyword That Returns OK or ERROR Response case=notfound\n
\nStatus Should Be 404 ${resp}\n
\nShould Be Equal As Strings NOT FOUND ${resp.reason}\n
\n
\nTest Get Request And Make Sure Is OK\n
\n${resp}= GET Custom Keyword That Returns OK or ERROR Response case=pass\n
\nStatus Should Be 200 ${resp}\n
\nShould Be Equal As Strings OK ${resp.reason}\n
\n
\n**\\* Keywords \\***\n
\n
\nGET Custom Keyword That Returns OK or ERROR Response\n
\n[Arguments] $case\n
\n[\u2026]\n
\nIF $case == notfound\n
\n$resp= GET [\u2026] expected_status=Anything\n
\n[Return] $resp\n
\nELSE\n
\n[\u2026]\n
\n\n#### to_json(content, pretty_print=False)\n\n*DEPRECATED* Please use ${resp.json()} instead. Have a look at the improved HTML output as pretty printing replacement.\n\nConvert a string to a JSON object\n\n`content` String content to convert into JSON\n\n`pretty_print` If defined, will output JSON is pretty print format\n\n#### update_session(alias, headers=None, cookies=None)\n\nUpdates HTTP Session Headers and Cookies.\n\nSession will be identified using the `alias` name.\nDictionary of `headers` and `cookies` to be updated and merged into session data.\n" + "markdown": "# Python API\n\n## HTTP\n\n### *class* RPA.HTTP.HTTP(\\*args, \\*\\*kwargs)\n\nThe *RPA.HTTP* library extends functionality of the [RequestsLibrary](https://marketsquare.github.io/robotframework-requests/doc/RequestsLibrary.html).\nSee that documentation for several examples of how to issue `GET`\nrequests and utilize the returned `result` objects.\n\nThis extension provides helper keywords to get an HTTP resource on a\nsession. The `HTTP Get` and `Download` keywords will initiate a\nsession if one does not exist for the provided URL, or use an existing\nsession. When using these keywords, you do not need to manage\nsessions with `Create Session`. Session management is still\nrequired if you use the underlying session keywords, e.g.,\n`* On Session`.\n\n#### DEFAULT_RETRY_METHOD_LIST *= ['PUT', 'GET', 'HEAD', 'OPTIONS', 'DELETE', 'TRACE']*\n\n#### ROBOT_LIBRARY_DOC_FORMAT *= 'reST'*\n\n#### ROBOT_LIBRARY_SCOPE *= 'GLOBAL'*\n\n#### check_vulnerabilities()\n\nCheck for possible vulnerabilities in the installed runtime\nenvironment packages.\n\nCurrently will check only for OpenSSL version and outputs warning message on any\ndiscovered vulnerability.\n\n* **Returns:**\n list of all check results\n\n```robotframework\n*** Tasks ***\nVulnerability Check\n ${results}= Check Vulnerabilities\n FOR ${result} IN @{results}\n Log To Console TYPE: ${result}[type]\n Log To Console VULNERABLE: ${result}[vulnerable]\n Log To Console MESSAGE: ${result}[message]\n END\n```\n\n#### create_client_cert_session(alias, url, headers={}, cookies={}, client_certs=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['PUT', 'GET', 'HEAD', 'OPTIONS', 'DELETE', 'TRACE'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`client_certs` [\u2018client certificate\u2019, \u2018client key\u2019] PEM files containing the client key and certificate\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_custom_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['PUT', 'GET', 'HEAD', 'OPTIONS', 'DELETE', 'TRACE'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` A Custom Authentication object to be passed on to the requests library.\n: [http://docs.python-requests.org/en/master/user/advanced/#custom-authentication](http://docs.python-requests.org/en/master/user/advanced/#custom-authentication)\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_digest_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['PUT', 'GET', 'HEAD', 'OPTIONS', 'DELETE', 'TRACE'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` [\u2018DOMAIN\u2019, \u2018username\u2019, \u2018password\u2019] for NTLM Authentication\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_ntlm_session(alias, url, auth, headers={}, cookies={}, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['PUT', 'GET', 'HEAD', 'OPTIONS', 'DELETE', 'TRACE'])\n\nCreate Session: create a HTTP session to a server\n\n`url` Base url of the server\n\n`alias` Robot Framework alias to identify the session\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` [\u2018DOMAIN\u2019, \u2018username\u2019, \u2018password\u2019] for NTLM Authentication\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n: Defaults to False.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### create_session(alias, url, headers={}, cookies={}, auth=None, timeout=None, proxies=None, verify=False, debug=0, max_retries=3, backoff_factor=0.1, disable_warnings=0, retry_status_list=[], retry_method_list=['PUT', 'GET', 'HEAD', 'OPTIONS', 'DELETE', 'TRACE'])\n\nCreate Session: create a HTTP session to a server\n\n`alias` Robot Framework alias to identify the session\n\n`url` Base url of the server\n\n`headers` Dictionary of default headers\n\n`cookies` Dictionary of cookies\n\n`auth` List of username & password for HTTP Basic Auth\n\n`timeout` Connection timeout\n\n`proxies` Dictionary mapping protocol or protocol and host to the URL of the proxy\n: (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019})\n\n`verify` Whether the SSL cert will be verified. A CA_BUNDLE path can also be provided.\n\n`debug` Enable http verbosity option more information\n: [https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel](https://docs.python.org/2/library/httplib.html#httplib.HTTPConnection.set_debuglevel)\n\n`max_retries` Number of maximum retries each connection should attempt.\n: By default it will retry 3 times in case of connection errors only.\n A 0 value will disable any kind of retries regardless of other retry settings.\n In case the number of retries is reached a retry exception is raised.\n\n`disable_warnings` Disable requests warning useful when you have large number of testcases\n\n`backoff_factor` Introduces a delay time between retries that is longer after each retry.\n: eg. if backoff_factor is set to 0.1\n the sleep between attemps will be: 0.0, 0.2, 0.4\n More info here: [https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html](https://urllib3.readthedocs.io/en/latest/reference/urllib3.util.html)\n\n`retry_method_list` List of uppercased HTTP method verbs where retries are allowed.\n: By default retries are allowed only on HTTP requests methods that are considered to be\n idempotent (multiple requests with the same parameters end with the same state).\n eg. set to [\u2018POST\u2019, \u2018GET\u2019] to retry only those kind of requests.\n\n`retry_status_list` List of integer HTTP status codes that, if returned, a retry is attempted.\n: eg. set to [502, 503] to retry requests if those status are returned.\n Note that max_retries must be greater than 0.\n\n#### delete_all_sessions()\n\nRemoves all the session objects\n\n#### delete_on_session(\\*\\*kwargs)\n\nSends a DELETE request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### delete_request(alias, uri, data=None, json=None, params=None, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use DELETE On Session instead.\n\nSend a DELETE request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the DELETE request to\n\n`json` a value that will be json encoded\n: and sent as request data if data is not specified\n\n`headers` a dictionary of headers to use with the request\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`timeout` connection timeout\n\n#### download(url: str, target_file: Optional[str] = None, verify: Union[bool, str] = True, force_new_session: bool = False, overwrite: bool = False, stream: bool = False, \\*\\*kwargs)\n\nAn alias for the `HTTP Get` keyword.\n\nThe difference in use is that the URL is always downloaded based on\nthe URL path (even without `target_file`). If there is a filename\nin the path, then that is used as `target_file` to save to. By default,\nthe filename will be \u201cdownloaded.html\u201d.\n\n```robotframework\n*** Settings ***\nLibrary RPA.HTTP\n\n*** Variables ***\n${DOWNLOAD_PATH}= ${OUTPUT DIR}${/}downloads\n${WORD_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file-sample_100kB.doc\n${EXCEL_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file_example_XLS_10.xls\n\n*** Tasks ***\nDownload files with reused session with provided file names\n # Starts a new session\n Download ${WORD_EXAMPLE} target_file=${DOWNLOAD_PATH}\n # Uses the previous session\n Download ${EXCEL_EXAMPLE} target_file=${DOWNLOAD_PATH}\n # Above files are downloaded using the same names as they have\n # on the remote server.\n```\n\n* **Parameters:**\n * **url** \u2013 target URL for GET request\n * **target_file** \u2013 filepath to save request content, default `None`\n * **verify** \u2013 if SSL verification should be done, default `True`,\n a CA_BUNDLE path can also be provided\n * **force_new_session** \u2013 if new HTTP session should be created,\n default `False`\n * **overwrite** \u2013 used together with `target_file`, if `True` will overwrite\n the target file, default `False`\n * **stream** \u2013 if `False` the response content will be immediately downloaded\n* **Returns:**\n request response as a dict\n\n#### get_current_session_alias()\n\nGet request session alias that was used with the `HTTP Get` keyword.\n\n* **Returns:**\n name of session alias as a string\n\n#### *static* get_file_for_streaming_upload(path)\n\nOpens and returns a file descriptor of a specified file to be passed as `data` parameter\nto other requests keywords.\n\nThis allows streaming upload of large files without reading them into memory.\n\nFile descriptor is binary mode and read only. Requests keywords will automatically close the file,\nif used outside this library it\u2019s up to the caller to close it.\n\n#### get_on_session(\\*\\*kwargs)\n\nSends a GET request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to retrieve the resource is the `url`, while query\nstring parameters can be passed as string, dictionary (or list of tuples or bytes)\nthrough the `params`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### get_request(alias, uri, headers=None, data=None, json=None, params=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use GET On Session instead.\n\nSend a GET request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the GET request to\n\n`params` url parameters to append to the uri\n\n`headers` a dictionary of headers to use with the request\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as GET data\n or binary data that is sent as the raw body content\n\n`json` a value that will be json encoded\n: and sent as GET data if data is not specified\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`timeout` connection timeout\n\n#### head_on_session(\\*\\*kwargs)\n\nSends a HEAD request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to retrieve the HTTP headers is the `url`.\n\n`allow_redirects` parameter is not provided, it will be set to False (as\nopposed to the default behavior).\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### head_request(alias, uri, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use HEAD On Session instead.\n\nSend a HEAD request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the HEAD request to\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`headers` a dictionary of headers to use with the request\n\n`timeout` connection timeout\n\n#### http_get(url: str, target_file: Optional[str] = None, verify: Union[bool, str] = True, force_new_session: bool = False, overwrite: bool = False, stream: bool = False, \\*\\*kwargs)\n\nA helper method for `Get Request` that will create a session, perform GET\nrequest, and store the target file, if set by the `target_file` parameter.\n\nThe old session will be used if the URL scheme and the host are the same as\npreviously, e.g., \u2018[https://www.google.fi](https://www.google.fi)\u2019 part of the URL.\n\n```robotframework\n*** Settings ***\nLibrary RPA.HTTP\n\n*** Variables ***\n${DOWNLOAD_PATH}= ${OUTPUT DIR}${/}downloads\n${WORD_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file-sample_100kB.doc\n${EXCEL_EXAMPLE}= https://file-examples.com/wp-content/uploads/2017/02/file_example_XLS_10.xls\n\n*** Tasks ***\nDownload files with reused session\n # Starts a new session\n HTTP Get ${WORD_EXAMPLE} target_file=${DOWNLOAD_PATH}${/}word-example.doc\n # Uses the previous session\n HTTP Get ${EXCEL_EXAMPLE} target_file=${DOWNLOAD_PATH}${/}excel-example.xls\n```\n\n* **Parameters:**\n * **url** \u2013 target URL for GET request\n * **target_file** \u2013 filepath to save request content, default `None`\n * **verify** \u2013 if SSL verification should be done, default `True`,\n a CA_BUNDLE path can also be provided\n * **force_new_session** \u2013 if new HTTP session should be created,\n default `False`\n * **overwrite** \u2013 used together with `target_file`, if `True` will overwrite\n the target file, default `False`\n * **stream** \u2013 if `False`, the response content will be immediately downloaded\n* **Returns:**\n request response as a dict\n\n#### options_on_session(\\*\\*kwargs)\n\nSends a OPTIONS request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to retrieve the resource is the `url`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### options_request(alias, uri, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use OPTIONS On Session instead.\n\nSend an OPTIONS request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the OPTIONS request to\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`headers` a dictionary of headers to use with the request\n\n`timeout` connection timeout\n\n#### patch_on_session(\\*\\*kwargs)\n\nSends a PATCH request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### patch_request(alias, uri, data=None, json=None, params=None, headers=None, files=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use PATCH On Session instead.\n\nSend a PATCH request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the PATCH request to\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as PATCH data\n or binary data that is sent as the raw body content\n or file descriptor retrieved by Get File For Streaming Upload\n\n`json` a value that will be json encoded\n: and sent as PATCH data if data is not specified\n\n`headers` a dictionary of headers to use with the request\n\n`files` a dictionary of file names containing file data to PATCH to the server\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`params` url parameters to append to the uri\n\n`timeout` connection timeout\n\n#### post_on_session(\\*\\*kwargs)\n\nSends a POST request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### post_request(alias, uri, data=None, json=None, params=None, headers=None, files=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use POST On Session instead.\n\nSend a POST request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the POST request to\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as POST data\n or binary data that is sent as the raw body content\n or passed as such for multipart form data if `files` is also defined\n or file descriptor retrieved by Get File For Streaming Upload\n\n`json` a value that will be json encoded\n: and sent as POST data if files or data is not specified\n\n`params` url parameters to append to the uri\n\n`headers` a dictionary of headers to use with the request\n\n`files` a dictionary of file names containing file data to POST to the server\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`timeout` connection timeout\n\n#### put_on_session(\\*\\*kwargs)\n\nSends a PUT request on a previously created HTTP Session.\n\nSession will be identified using the `alias` name.\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### put_request(alias, uri, data=None, json=None, params=None, files=None, headers=None, allow_redirects=None, timeout=None)\n\n*DEPRECATED* Please use PUT On Session instead.\n\nSend a PUT request on the session object found using the\ngiven alias\n\n`alias` that will be used to identify the Session object in the cache\n\n`uri` to send the PUT request to\n\n`data` a dictionary of key-value pairs that will be urlencoded\n: and sent as PUT data\n or binary data that is sent as the raw body content\n or file descriptor retrieved by Get File For Streaming Upload\n\n`json` a value that will be json encoded\n: and sent as PUT data if data is not specified\n\n`headers` a dictionary of headers to use with the request\n\n`allow_redirects` Boolean. Set to True if POST/PUT/DELETE redirect following is allowed.\n\n`params` url parameters to append to the uri\n\n`timeout` connection timeout\n\n#### request_should_be_successful(response=None)\n\nFails if response status code is a client or server error (4xx, 5xx).\n\n`response` is the output of other requests keywords like GET On Session.\nIf omitted the last response will be used.\n\nIn case of failure an HTTPError will be automatically raised.\n\nFor a more versatile assert keyword see Status Should Be.\n\n#### session_exists(alias)\n\nReturn True if the session has been already created\n\n`alias` that has been used to identify the Session object in the cache\n\n#### session_less_delete(\\*\\*kwargs)\n\nSends a DELETE request.\n\nThe endpoint used to send the request is the `url` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_get(\\*\\*kwargs)\n\nSends a GET request.\n\nThe endpoint used to retrieve the resource is the `url`, while query\nstring parameters can be passed as string, dictionary (or list of tuples or bytes)\nthrough the `params`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs` here is a list:\n\n`data` | Dictionary, list of tuples, bytes, or file-like object to send in the body of the request. |\n
\n`json` | A JSON serializable Python object to send in the body of the request. |\n
\n`headers` | Dictionary of HTTP Headers to send with the request. |\n
\n`cookies` | Dict or CookieJar object to send with the request. |\n
\n`files` | Dictionary of file-like-objects (or `{'name': file-tuple}`) for multipart encoding upload. |\n
\n`file-tuple` | can be a 2-tuple `('filename', fileobj)`, 3-tuple `('filename', fileobj, 'content_type')` or a 4-tuple `('filename', fileobj, 'content_type', custom_headers)`, where `'content-type'` is a string defining the content type of the given file and `custom_headers` a dict-like object containing additional headers to add for the file. |\n
\n`auth` | Auth tuple to enable Basic/Digest/Custom HTTP Auth. |\n
\n`timeout` | How many seconds to wait for the server to send data before giving up, as a float, or a `(connect timeout, read timeout)` tuple. |\n
\n`allow_redirects` | Boolean. Enable/disable GET/OPTIONS/POST/PUT/PATCH/DELETE/HEAD redirection. Defaults to `${True}`. |\n
\n`proxies` | Dictionary mapping protocol or protocol and host to the URL of the proxy (e.g. {\u2018http\u2019: \u2018foo.bar:3128\u2019, \u2018http://host.name\u2019: \u2018foo.bar:4012\u2019}) |\n
\n`verify` | Either a boolean, in which case it controls whether we verify the server\u2019s TLS certificate, or a string, in which case it must be a path to a CA bundle to use. Defaults to `${True}`. Warning: if a session has been created with `verify=${False}` any other requests will not verify the SSL certificate. |\n
\n`stream` | if `${False}`, the response content will be immediately downloaded. |\n
\n`cert` | if String, path to ssl client cert file (.pem). If Tuple, (\u2018cert\u2019, \u2018key\u2019) pair. |\n
\n\nFor more updated and complete information verify the official Requests api documentation:\n[https://requests.readthedocs.io/en/latest/api/](https://requests.readthedocs.io/en/latest/api/)\n\n#### session_less_head(\\*\\*kwargs)\n\nSends a HEAD request.\n\nThe endpoint used to retrieve the HTTP headers is the `url`.\n\n`allow_redirects` parameter is not provided, it will be set to False (as\nopposed to the default behavior).\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_options(\\*\\*kwargs)\n\nSends a OPTIONS request.\n\nThe endpoint used to retrieve the resource is the `url`.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_patch(\\*\\*kwargs)\n\nSends a PUT request.\n\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_post(\\*\\*kwargs)\n\nSends a POST request.\n\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### session_less_put(\\*\\*kwargs)\n\nSends a PUT request.\n\nThe endpoint used to send the request is the `url` parameter, while its body\ncan be passed using `data` or `json` parameters.\n\n`data` can be a dictionary, list of tuples, bytes, or file-like object.\nIf you want to pass a json body pass a dictionary as `json` parameter.\n\nBy default this keyword fails if a status code with error values is returned in the response,\nthis behavior can be modified using the `expected_status` and `msg` parameters,\nread more about it in Status Should Be keyword documentation.\nIn order to disable this implicit assert mechanism you can pass as `expected_status` the values `any` or\n`anything`.\n\nOther optional requests arguments can be passed using `**kwargs`\nsee the GET keyword for the complete list.\n\n#### status_should_be(expected_status, response=None, msg=None)\n\nFails if response status code is different than the expected.\n\n`expected_status` could be the code number as an integer or as string.\nBut it could also be a named status code like \u2018ok\u2019, \u2018created\u2019, \u2018accepted\u2019 or\n\u2018bad request\u2019, \u2018not found\u2019 etc.\n\n`response` is the output of other requests keywords like GET or GET On Session.\nIf omitted the last response will be used.\n\nIn case of failure an HTTPError will be automatically raised.\nA custom failure message `msg` can be added like in built-in keywords.\n\nNew requests keywords like GET or GET On Session (starting from 0.8 version) already have an implicit assert\nmechanism that, by default, verifies the response status code.\nStatus Should Be keyword can be useful when you disable implicit assert using `expected_status=anything`.\n\nFor example when you have a nested keyword that is used for both OK and ERROR responses:\n\n**\\* Test Cases \\***\n
\n
\nTest Get Request And Make Sure Is A 404 Response\n
\n${resp}= GET Custom Keyword That Returns OK or ERROR Response case=notfound\n
\nStatus Should Be 404 ${resp}\n
\nShould Be Equal As Strings NOT FOUND ${resp.reason}\n
\n
\nTest Get Request And Make Sure Is OK\n
\n${resp}= GET Custom Keyword That Returns OK or ERROR Response case=pass\n
\nStatus Should Be 200 ${resp}\n
\nShould Be Equal As Strings OK ${resp.reason}\n
\n
\n**\\* Keywords \\***\n
\n
\nGET Custom Keyword That Returns OK or ERROR Response\n
\n[Arguments] $case\n
\n[\u2026]\n
\nIF $case == notfound\n
\n$resp= GET [\u2026] expected_status=Anything\n
\n[Return] $resp\n
\nELSE\n
\n[\u2026]\n
\n\n#### to_json(content, pretty_print=False)\n\n*DEPRECATED* Please use ${resp.json()} instead. Have a look at the improved HTML output as pretty printing replacement.\n\nConvert a string to a JSON object\n\n`content` String content to convert into JSON\n\n`pretty_print` If defined, will output JSON is pretty print format\n\n#### update_session(alias, headers=None, cookies=None)\n\nUpdates HTTP Session Headers and Cookies.\n\nSession will be identified using the `alias` name.\nDictionary of `headers` and `cookies` to be updated and merged into session data.\n" }, { "module": "RPA.Hubspot.Hubspot",