ÿØÿà JFIF    ÿÛ „ !.%+&8&+/1555$;@;4?.451 4,$,44444444444414444444444444444444444444444444444444ÿÀ  á á" ÿÄ     ÿÄ ?    !1AQaq"2‘¡±ÁðBRbrÑá#‚’¢²3S CñÿÄ   ÿÄ !    !1QAa‘2ÿÚ   ? 5˜Z¯V¦cø)›t/? z¨±>Õ5€¶‹Á¤·¼z¼Ü¬+ñ®v¤¨_ˆR­BFn©—˜ý®ç̝P8gýt·ÉSTŦˆìät?þé¼íìN/Þa)ì–í6ô… Ï¿øÃj´¿KÇü]ÿ ªô¹-eKànëÕHTx}ýSÜ›ÿ ”7Ø×&µ<¦  ¥ÑO¶[Ù¯ä¨ÞÃÿ PZ-¬;#õ|•oaÿ ©CìÞz3˜öː/¤­ñTûIØ}š^ mÓ%ªxˆ¥ÉŸu=Z+ISe¿45™¼u;ú&WØ÷€æßQ™®{|íx*TC“#ZŠìZ§²‹ 6pv…³¿¡äª*áZÐ%ÒOáˆo"x«OHk w±æ+¬V(kMúŸ5Vö«$ ÁrÏbàb57/luR ¸ÑÛj Òµì`Мq­û žICÀÊ•©4€Âcà¨Ï€O´<èÐ:›ù(Ë^L8þ‘ÍÌ#¸Ð_Ì©ÙK(Öz 4¬û+¸;ü’V’84‘¬ÃŽ:[â‡ÔÌáõp¢~§ªlæ£ö{®G>J¼"°‡7¯ÆÉèßû ‹É‹§ÁòÃýâßî ^ƾÙõ‹×óH#«LP½ïX=xÑÍ$|W?•~• îëÔ©ª‹ {ÝT…Kÿ ”hûâá)J*ö˜–ÔU;iÇ€/ ÆþjóZ\ýwØ=Ìm ºèËL9 ýèÆð/¨’¥öo=nË.%Îì ŽÕ¯È|{Oj²ƒE6e/ßdÄõ²Ìâ1O®ò×TsəԸhOMýíMˆ¿¼H˜l²,7Â¥#MF/Úf°Ö½± ¸–dr‹NýÊ íjqx{œÉ ä-È ¦ øÄër¨q°ð †nцýÑÄÆ’mä…n<0È™;ÁÝá¯ÁZƒ7FÀmì­ É&9ˆîéi¶ùN§Y• ÃZãAâ?•‡©‰ , ó¾IŸŠc1 4â&y­&pŠ­6;M À 0¹qç»p.á …ŸÅáK@%6·y6ƒ‰3?”úºŽ‰éX5ªPT §µ!=Mž«Ú½‹ÅgÂSâÉaþÓoö–¯ÁÔìR>5éÿ üs¶ÆUcÌ kÇR ]ÿ ù¬¼«VŽ;Â|‡~¢¦”ÏŰæ {L™Õ°Óv¹ò¸írޡעCÃ!íVÕ {¶»sŒNPg/ "uÕbkm²“$ďå¿é¹§°½æz¯6 †s¿!s–wÚÝ“™Œ °.ûj>·+™Òa…©Œ&rÝÎtÛë긪Ît’LAVp%c Úý[ÄzJ¾ÇàXXç@˜ó<êL]·T˜¾¥1Ó©V‡g´æ½¦Ý@¹óø!_@´ÞâSÁ —S3™•& ]@JHÚý©ZŽ €×æÔr»Áf!‡yÞ4Mv*èÓã_{‘åóUuљØ«Oïé*®EvÑ Œ÷‡U \"㪒ÍK+À 4“M¡ï:0¥5í!'<@î´”>Ç»&Z–ïCCV˜Ì5Šo&îhè.žû |ÓK©h$s6KìŒëã)¹hI¦GïOåóI;ììü#É$Š0…Ææ¥TØ.5­¾gn´ “ÂÖ\:hœ89G)J@„}œ:’Ò{/Š"¦_Æ×7Æ3VÇŠÊa]ÚŒÙ€Ä–=®uÁßâACZƒ§§£ Qnâ:«,×{tyø¬iÛcœÜÄ€H½ÄÍCk´÷šß .W'b¤Íåh]÷€=,Žv×cÚEÚHXJX¶îo¨FÒtèöŸ>ªª6[J®Fµ£sGÁeqõfe\íjÒÐïÄÐGˆe1Ø‹.Ø”‘Ëuø Y­ˆÜ ŽG|zùªüMpDnQWÄ”%JŠ™)â*p@Örš«ÕT2Ð%ˆG#ª„ ·¤!°ŸOTÂT¸aÚ%4&h™LµšØüÐ.F¿²ÐÞ_Ç‚¾ÅÃaÜ÷09Æ q€öy˜v‡85õN÷]¬äѼóS{°_MެúÔ#°Ç¸0åÞè2ëôPcvÆw9®ií1Ä8F™˜à‰´+‰Ik1òÝ7“Ñ×ÒsÝ\x‚h`ÞÑ`ó"|µEcý£n˜h`}GÞ !±ù²Ápü²ß6 0ïi󜵩SÈÇ7˜-ÕURO˜¦´f$ªž-Í6(œ}<„ éc øs]ŽŽ„*—¾ ìdŽ„)méª\¿êÎIg¾ØÞ~I#C/¼¼´EÁÈŽi8“©õådô·>euä ƒ'Ê×लR1ÉJE1ÐAát`t;ÇР%Ý<‡¥„ÍÆ`×Oyó)õiI€ñQaŸ4Ûù\áàaÃÔ¹HÃu¹*k€¦<„e S‡&õÏ B!ŽhüÞ`yj}mªf×\¿ Ç~æ­9‡û\՞Ǖg²1Žû5V7 !àöšm° c`ܬøÇìµÒ'P"?…´Ö,"§^•õލsÔ)6˜sæéÍR¼ ò|Sl”‹7 nPW Gòú÷½§O¯‡„l¡kSÞŒr½PÊ@æ¢pŽ-mÿ #Ÿ˜Àº¶Áä¦;ïÔæ$1££`“Õ>„—·ž)ßð³ñ#Ï Ô$¶œ‰ÊE‹À;÷º ¯«P:Ñ”8–IÊtpÞ3ª“>ê“þës4ò2OÏÕ­±zô†Õ§‰.÷ä¸;¿˜“'œ›žª}«Œ{ª±Ì 9ÔóÞÕ‡0 $íWV3Üì¬ —@kÝ4@¿r¼±½¬™›?øØæ´'Áé®CË3-g$˜ö‡×auÚi´Žp/êÛ æF›Ú2v‹ã¿¿,nB1̨ƃqÞa5͝@&Æû“él÷ \C²½UÍc ¯k×¢U ÖéQå™—-r wô ÞÏ<Ò=&=ÿ Ôê Òêˈt,i—;LîÜ á¸*ÚÃ1$êL•LÍ <É)ýÐà’ ;F™{ƒ™˜€&'}‚ãÄK`¡ÞT@I;®žZóè‚s’7®°›+§O­Åq©é»²9<Ô J ¼9O’HL»Ùïì¸rk¼Ž_ý‘TŸu[²ßÚŒ·ü÷B%¯E ŸÔX5êO´ Ç•€’I0 ÉJX` ñ¹õ%;µŸD‘«´€àwÒ™U ûئžÖö\×®×´8 ½‡ºÐÆÓ§?Àkmœ=;d5*@-ì0F Rªýš[Ü6âö̃ڸr*KA9· u*µæ£?U¸Âêí†8@¦X4 e-ò„0s{ HâUpU?¼mñRa°®a%Ð'tÉ×’\¾ÊÉ]t›h>·(Ë@R¼¡Ãt h}’O÷au<+nT…Ö…MӐ??Óe95 q>í/;&JSû °¯ÊéÞ øƒ*Ã2½Ài&:nôUl=¾¿5eˆ3”ñc|Ú2V”>„»&eE;«ÚäC p¢Û úy 9š[ŒÌx¼擼A&DåÒ¯ˆ¤ÀÌ;"˜ ÏQä¸åhÊ}Ûq«Û0WžÒ|»€ø®öCm5•\ÇÀ§Pe3£]0ÃàLDÉ‰1øªxjgwT‚÷¿LΨK‹›ùs—xˆÜ±µ kæ¸f‰‰ÜGk/LÛØ6d9ò¶ùA{ƒA3š/¬D¬khÓk‰`˜"㯒r¿±Óã jx‡°e}<Ñø\3y:'À•/h½Í€Ç4~g ?Û(¼]v‘ªlKÎâ~?O‚W%{Ì:“'©úNq¾›úo(X’¥¯ˆ nFê{Ç€ü?º'ë ø‹ì Þ09ŒÌç9Æ —ËC`j@ÓÄ(+a‹un¸#ÂꟋ{K`‘ÑÍÍ'à´»/Û,KW;Þ4²þð ï Nm|~fGÏ(…³Ã)«1ö­Õ ¥‡¨©ƒÃ™ü-s=à=U66Ï«Ýc蓦W¹íž®›nÔ%êÇìŒ<#Ü×84ån®Ð ÒåOC` ñânÑs‡¢ç 1õ%Îhì½Ã½® e:ݼUZo™`  ÅZŸŒÊ«ê1ÏÄo$q¹Þ€©ˆhÐÉä¯ñ[!…Ú˜àJ:x2$Íß&PåT£6ç— ‡Í*4Ýšçjÿ ‰É nófÐ ó(L5C•åÆ\rMÒ@ò }y-W}™üýVù—ú¢=Ù”c®‘< M ž ´Phr ¦©TD ‘ù.$´÷O‡‘V2Æò.=IUŒ=ž‡â¬i™aþÓåÙ?òUø'ØÖ•.~* šTŒ!•-×áºTâ®ä#õü'´ eýlYÅÓeÕKÂrT"CÚ@u!Óxƒ{š3€}1¿(r}%«nËamjÑ%ÑNEò v ˜à  σöK³,*º.àzù¨™Ó ÚçâU¦*¿ 9{%Ö¹ njûdaXöb) kÛÆ±ûÓ\°M7ˆÂ=û›ç¿Ã‚­V»Cg–8ÙêE- j)k$º`Ã-ùEýeBÆÇ]c¡°ñty&Òd0nõ'¡W+ƒ*|–øµFa\GQªEAÔp5\Ǽ·¼Ç8·õ -â§Ú[ ‡ uZeÖ 3}×d'+¹:ð+K†Û®s!Ï$úe€<Û”x)1»a­¡LC]¸µík…ÚàA»AYº{†ªS[¦5HÒ7ù --,ísòDØ€èk ÞÀîÜ ò@â( ËNˆë›4ô½•/¦o‡€Û7 ê•ÆêòðÜy'Án½µ á˜ݦ ndeo…[ì¶Ê,¥R³Ä=À±—–ß;£™´ñSâ*g§”ïaið‘Jå~™ÓÞ ß³Õ¢»8x埒²52>AÊb&-÷\7´éÄù€T˜,w;3{ï˜k…à¹ÄqÀ«œ{€\ ˆ¾[´¨јr &Úé„Ívˆ±8†¿]|¬ņ4I×pÞS1ÈÖz‰#Ìv‡G!YNògñ:màTz¢Ý1ô©^O=~ë|5Bã™ç•¼µõ•bÆ@úÕS¬ÈŒ#¬zünrŸ û” Z²•èðV"ÁHÚý©wÝ €7¼Ìu1hÑa3Éä û f$o¿É ™Ú›ÝçnpÒ3äÌ3†Í§,Äï]$‰/pê †«À¼¸e9­Æê_C]žƒ·ý·frÁN«, E=›Çq -‰öŒ:aÏ¿±í&£Í:-} 84‘ÿ eƒQÑeëSsuiA ³g㟥ú£?ÿ ʼn*”“÷aühe:ÊWa@ÒÞk±eØ] F Ô—r.åä˜ @ö¥ªZoÐýYL·¥S²G/‡ñ <~*ZÆ´è>JlòàÛÆ½ÿ 窘ìGN¢:I®KšJp/`íIÁÀõ#Ä-€ö­šµŒoF4|ÆQØÆ@Ì|£Ô…¢À{9˜è½Üó›€ôYÒÎYsið;ís¤€à²ˆ‚4qÉVŒI$ ‰"° æµ8cXGjœˏ¡Aâý•ËÜ¢ûï e·çLx']á"oÅÎê3¯Ç—¹”ó0nå‚âg{Œñ> S´˜îè°g238‚ãköÝfÚd´6Ò€;ò÷±¢™¼›º ¢Æ'¥Ðx'e¬ç ]bÈÆV¢ó‹kýBO ðÊâ$Ÿ!×T 3Mýמ žìٍàÌü‘8÷€àæØ8æ©6‰©L´«…oãpð„~Çk‰!ñ;‹”ÛžÍ àž±z Ÿôû øŸÝužÏ;ÿ #|u6™Þ¬ÚˆÐõA4¶â|ôl|Ê2ŽÇ¤ÝÅÇY.<#Aí.k§hóF‚”Y; M½Ö4hŸ4&›­¿tès´%FìL¥£Ãk‰ÇT¤haÁ¤ÚxfÉ`ÑìË›>i 3t‚:,–+^÷´–{Û–Nxi"x‘Ûg î¨>¥Õ܁ùZH,2Û“:8xÊ¢Çí9.É-Ìâã-=çjwµS˜dütžçwýGòú®®ûº_ˆýx$–¡ãøO EÚÛÏ÷R„×w+3£Á£öUMyR²¹âŒ°š›¸Ñãò9§Ó_Dl+Ùßc›úšGÅÌc†Ž!Ko=¶.‘Îÿ c²(2®V mª.ÿ ¹B›¹å ù„öŸSV>™ü¯$y:G¢Z×àøúdî¹û­·ýÇ´:•c LÍõi_‹ö+ÎæGÊè>OŠ•äž´§Þ{X}¨1ÚTc›»Qþ•êô°t¿OP?eæ~É{5]•ÙR£r5†nZ\ã@ &îJõ ¾àC°þV>fé¥/ü5ñÊIº_é5 ;e­h<@ Ä&æÃëE%;X,ÒãÆÞ`Oò¦kŸm#˜!ÀyÄ¢| óLšò¥Ä` ¶R=|ÈCâh5ò3DˆïF†ðÒ#ÅìÛœ?¸yhBãœí ZxßÎÄhºRK„`Þödvײ™ÀÈÑÒgŒuY w³%†ƒÓzõ ÖÏp‚dH®¦A´ù§»ÓÇMæ~)ˆð‡û:ù&Ä •vGD´À n ݇¼Ö8Fö óáà£~Ë¥x`oK|Ä?fxiØü%pìR>éò+Û±éÎ>núlFŤ'tq8LZÏvÃ?„¡ß±È⽆¯³íü@x|PöUäèØã¡ð‚ŒAìÏ"vÍwóŸÍ{ ý0.z È•Ö{,N¡£¡ŸKÕÙž>Ýœþ ÍÀ°<×EA!Å‚D™IúOÍ¡>ôG}Â` ÍßkÜL™Ž Þð™ {IøF²¹òQ3&!ÃÂÞz.d&Ï-sH¸,Ôõ˜ŽP€ 77ˆÝ¼ÊëÜw =cÕ Ú,ØÐ5ÎYÐ)ì´öœgŒ[¤ßv㙑8心>h]§µháYš£²ºÑ.{Ï7Sð•?´~×SÃKýJÛ˜ ™Íäiúu<µX¶1õ^kâçIÑ£sZ4h>j*ÔšD:4­¿_ ÷¸ Õxæÿ ¸?Mù _•­ÊÐ ä ÷ý ÑwL œ­ïnTkÛUÍN©ë:¦fV ¶ÜÔÜMªÅâA½–¿R×TXš-%iTÊT•‡Ù‚JôϐZxWÑè‰f‰òG º ×Õû2aZ7OU3[“×AT–ÞŒ…-‘¤”Ì ì&(ˆ¿­•ƒkï’:ðY¦W‘ Å)“†‘˜³Åtcø˜ñTÂwÚÇ4|üLÇªí–v- qˆèU qPE.†â‘˜µ Æ,ÐÅs]8¾„oúÑ i>ÜxxÈó)ƒ ´æÁâØ$À‰vžŸf$Ž |ãw;ÀÁIJ»b` {¦Ó¤Ú$©YÀ‘n@Óïž«9J¼êG m¤ ܯ¹ÌW4€ÐÒÅÛ‡#褕Ÿn-?í|с¥÷Ú¹¬'´ÞÜ9ÓK `hê£SÄSà?7—Wí_´…óB›»:=Ãïq`<8ñÓŒÑlú2d¬ê³£hÖ[l|$vÝro~'R®‰§°ñmY ͧäP |PUª¹·:3Œ[Û{Xÿ ºâ@‚W–Äé u‚ ¯´*=íή.pûÒdt @G‰¬ s¸ ëÉücr ÞæÑ¨Ê@>¤¢Ö±. Þ'¯°ÌME[YéïĵÂCå½ Ué©Áû'Ê9%eÔðNU”ë‘ÌsD3/®+UI˜9h.WC”빓$#:pz:YÓ ¿xž* ³$Í +$kñAŠ‹†¢ Uê>¸)_š¬÷©ßAÂÔb9ÇU ¯¾á•9¯ÏÏ÷O÷¼¼Fähal1‰3Ì[Ïr•´UCksNÐ] R‘¸¥H+§Šé†c©vÖÞ0iÓ76s†î!§=ß ¼~Ô'°Ãmäoäš³ªøi1úÉ)³yV8 CLÄØÁ‘WYïi€H6ÖÑiámø^ÈY´°Ñ7¥Û*—Ñ©L«Qƒï—Ùrÿ ›£Ð*š¸ˆL©ˆ$ˆ ÷¾D§9È®«qbqC)–ˆïv´çñsÑVT­Ø, <àïºÀO«Jý·õ àfPìð .wFšir´þ’2_Y *Æ€x\« ì€9š@ Ž|F⇥ˆkZ@hÖÄ0t¿-<“‹qµ¾*ZL¤Ú)&BJpÓF5=$„at*Zš$’ÑtdûÝRI1 2މ$€$I$#‰SÞ’Hë¬ï;Á$¡t$’`<(ñÇt)$‡Ð.Êf¢X’Kt=Éé$‚ˆªè¢oÝëòI%Rgcª÷ŠyI%¡‰ÿ !ñ)´õ $¤ Ô’IIGÿÙimport logging import agent_util import jpype from jpype import java, javax logger = logging.getLogger(__name__) class TomcatJMXPlugin(agent_util.Plugin): """Tomcat Plugin for the FortiMonitor Agent using JMX to collect data.""" textkey = "tomcat_jmx" label = "Tomcat (JMX)" JMX_MAPPING = { # Memory "memory.heap": ( "Heap Memory Usage Used", "java.lang", "Memory", None, "HeapMemoryUsage", "used", "bytes", ), "memory.heap.committed": ( "Heap Memory Usage Committed", "java.lang", "Memory", None, "HeapMemoryUsage", "committed", "bytes", ), "memory.heap.init": ( "Heap Memory Usage Init", "java.lang", "Memory", None, "HeapMemoryUsage", "init", "bytes", ), "memory.heap.max": ( "Heap Memory Usage Max", "java.lang", "Memory", None, "HeapMemoryUsage", "max", "bytes", ), "memory.non_heap": ( "Non-Heap Memory Usage Used", "java.lang", "Memory", None, "NonHeapMemoryUsage", "used", "bytes", ), "memory.non_heap.committed": ( "Non-Heap Memory Usage Committed", "java.lang", "Memory", None, "NonHeapMemoryUsage", "committed", "bytes", ), "memory.non_heap.init": ( "Non-Heap Memory Usage Init", "java.lang", "Memory", None, "NonHeapMemoryUsage", "init", "bytes", ), "memory.non_heap.max": ( "Non-Heap Memory Usage Max", "java.lang", "Memory", None, "NonHeapMemoryUsage", "max", "bytes", ), # Threading "threading.count": ( "Thread Count", "java.lang", "Threading", None, "ThreadCount", None, "count", ), # OS "os.cpu_load.process": ( "OS Process CPU Load", "java.lang", "OperatingSystem", None, "ProcessCpuLoad", None, "percent", ), "os.cpu_load.system": ( "OS System CPU Load", "java.lang", "OperatingSystem", None, "SystemCpuLoad", None, "percent", ), "os.open_file_descriptors": ( "OS Open File Descriptor Count", "java.lang", "OperatingSystem", None, "OpenFileDescriptorCount", None, "count", ), # Class loading "class_loading.loaded_classes": ( "Loaded Class Count", "java.lang", "ClassLoading", None, "LoadedClassCount", None, "count", ), # MemoryPool "memory_pool.eden": ( "Eden Space", "java.lang", "MemoryPool", "Eden Space", "Usage", "used", "bytes", ), "memory_pool.eden.ps": ( "PS Eden Space", "java.lang", "MemoryPool", "PS Eden Space", "Usage", "used", "bytes", ), "memory_pool.eden.par": ( "Par Eden Space", "java.lang", "MemoryPool", "Par Eden Space", "Usage", "used", "bytes", ), "memory_pool.eden.g1": ( "G1 Eden Space", "java.lang", "MemoryPool", "G1 Eden Space", "Usage", "used", "bytes", ), "memory_pool.survivor": ( "Survivor Space", "java.lang", "MemoryPool", "Survivor Space", "Usage", "used", "bytes", ), "memory_pool.survivor.ps": ( "PS Survivor Space", "java.lang", "MemoryPool", "PS Survivor Space", "Usage", "used", "bytes", ), "memory_pool.survivor.par": ( "Par Survivor Space", "java.lang", "MemoryPool", "Par Survivor Space", "Usage", "used", "bytes", ), "memory_pool.survivor.g1": ( "G1 Survivor Space", "java.lang", "MemoryPool", "G1 Survivor Space", "Usage", "used", "bytes", ), "memory_pool.old.ps": ( "PS Old Gen", "java.lang", "MemoryPool", "PS Old Gen", "Usage", "used", "bytes", ), "memory_pool.old.cms": ( "CMS Old Gen", "java.lang", "MemoryPool", "CMS Old Gen", "Usage", "used", "bytes", ), "memory_pool.old.g1": ( "G1 Old Gen", "java.lang", "MemoryPool", "G1 Old Gen", "Usage", "used", "bytes", ), # Garbage Collector "gc.young.copy": ( "Copy", "java.lang", "GarbageCollector", "Copy", "CollectionCount", None, "count", ), "gc.young.ps_scavenge": ( "PS Scavenge", "java.lang", "MemoryPool", "PS Scavenge", "CollectionCount", None, "count", ), "gc.young.par_new": ( "ParNew", "java.lang", "GarbageCollector", "ParNew", "CollectionCount", None, "count", ), "gc.young.g1_generation": ( "G1 Young Generation", "java.lang", "GarbageCollector", "G1 Young Generation", "CollectionCount", None, "count", ), "gc.mixed.g1_generation": ( "G1 Mixed Generation", "java.lang", "GarbageCollector", "G1 Mixed Generation", "CollectionCount", None, "count", ), "gc.old.mark_sweep_compact": ( "MarkSweepCompact", "java.lang", "GarbageCollector", "MarkSweepCompact", "CollectionCount", None, "count", ), "gc.old.ps_mark_sweep": ( "PS MarkSweep", "java.lang", "GarbageCollector", "PS MarkSweep", "CollectionCount", None, "count", ), "gc.old.concurrent_mark_sweep": ( "ConcurrentMarkSweep", "java.lang", "GarbageCollector", "ConcurrentMarkSweep", "CollectionCount", None, "count", ), "gc.old.g1_generation": ( "G1 Old Generation", "java.lang", "GarbageCollector", "G1 Old Generation", "CollectionCount", None, "count", ), } @staticmethod def __get_object_name_from_tuple(tuple_): """returns a constructed ObjectName. :type tuple_: tuple (label, domain, type, bean_name, attribute_name, composite_data_key, unit) :param tuple_: A tuple with all the information for an ObjectName. A string that represents the label, a string that represents the domain, and so on and so forth. :rtype: javax.management.ObjectName :return: An ObjectName object that can be used to lookup a MBean. """ domain, type_, bean_name = tuple_[1], tuple_[2], tuple_[3] canonical_name = "%s:" % domain if bean_name: canonical_name += "name=%s," % bean_name if type_: canonical_name += "type=%s" % type_ return javax.management.ObjectName(canonical_name) @classmethod def get_connections_from_config(cls, config): """ Parse the config object to build a structure of connections parameters based on the number of entries that are in each key. The main parameter we base on to split off is host. :type config: dict (host, port, username, password, jvm_path) :param config: Dictionary with the information stored in the config file. :rtype: Dict :return: Dictionary with connection information split up in multiple if needed. """ keys = ["host", "port", "username", "password", "jvm_path"] data = {} for key in keys: key_value = config.get(key) if not key_value and key not in ("username", "password"): raise ValueError("Missing %s information from config" % key) elif not key_value and key in ("username", "password"): # Username and password are not required continue else: values = [value.strip(" ") for value in key_value.split(",")] data[key] = values connections = {} hosts = data["host"] for index, host in enumerate(hosts): connections[index] = { "host": host, } for key in ["port", "username", "password", "jvm_path"]: if len(data.get(key, [])) > 1: # Multiple entries in this config, use the index to apply. value = data[key][index] elif len(data.get(key, [])) == 1: value = data[key][0] elif key not in ("username", "password"): raise ValueError("Missing %s information from config" % (key)) else: # Username and password can be skipped continue connections[index][key] = value return connections @classmethod def __get_connection(cls, config): """ returns a list of connections from the jpype library - a python interface to the Java Native Interface. Wheter there are 1 or many connections depends on the number of entries in the host, port and optionally username/password/jvm entries. :type config: dict :param config: Mapping of information under the application block for this plugin. :rtype: tuple (status, connection, error_message) :return: A tuple containing a numeric value corresponding to the agent_util status'. A MBeanServerConnection object. And, a string with an error message if any. """ status = agent_util.SUPPORTED msg = None # Check that we have an agent config if not config: msg = "No JMX configuration found" cls.log.info(msg) status = agent_util.MISCONFIGURED # Make sure at least host and port are in the config if "host" not in config or "port" not in config: msg = ( "Missing value in the [%s] block of the agent config file" " (e.g host, port)." ) % cls.textkey cls.log.info(msg) status = agent_util.MISCONFIGURED # Try and get the jvm path from the config jvm_path = config.get("jvm_path") # If we can't find it then try and use the default if not jvm_path: try: jvm_path = jpype.getDefaultJVMPath() if not jvm_path: status = agent_util.MISCONFIGURED msg = ( "Unable to find JVM, please specify 'jvm_path' in" " the [%s] block of the agent config file." ) % cls.textkey cls.log.info(msg) except: status = agent_util.MISCONFIGURED msg = ( "Unable to find JVM, please specify 'jvm_path' in the" " [%s] block of the agent config file." ) % cls.textkey cls.log.info(msg) try: # If the JVM has not been started try and start it if status is agent_util.SUPPORTED and not jpype.isJVMStarted(): jpype.startJVM(jvm_path) except: status = agent_util.MISCONFIGURED msg = "Unable to access JMX metrics because JVM cannot be started." cls.log.info(msg) if status is agent_util.SUPPORTED: try: # Start the JVM if its not started # XXX: Redundant logic - is this necessary? if not jpype.isJVMStarted(): jpype.startJVM(jvm_path) j_hash = java.util.HashMap() # If we have a username and password use it as our credentials if config.get("username") and config.get("password"): j_array = jpype.JArray(java.lang.String)( [config.get("username"), config.get("password")] ) j_hash.put( javax.management.remote.JMXConnector.CREDENTIALS, j_array ) url = "service:jmx:rmi:///jndi/rmi://%s:%s/jmxrmi" % ( config.get("host"), int(config.get("port")), ) jmx_url = javax.management.remote.JMXServiceURL(url) jmx_soc = javax.management.remote.JMXConnectorFactory.connect( jmx_url, j_hash ) connection = jmx_soc.getMBeanServerConnection() return status, connection, None except Exception: msg = ( "Unable to access JMX metrics, JMX is not running or not installed." ) cls.log.exception(msg) return status, None, msg @classmethod def get_metadata(cls, config): """returns a json object who's textkeys correspond to a given metric available on the JVM. :type config: dict :param config: Mapping of information under the application block for this plugin. :return: JSON Object for all metrics """ result = {} configs = cls.get_connections_from_config(config) connections = {} errors = [] for entry in configs.values(): status, connection, msg = cls.__get_connection(entry) connection_key = "%s:%s" % (entry["host"], entry["port"]) if msg: errors.append("%s %s" % (connection_key, msg)) else: connections[connection_key] = connection if not connections.keys(): cls.log.info("Unable to connect to any connection") for msg in errors: cls.log.error(msg) return result else: status = agent_util.SUPPORTED msg = "" for error in errors: cls.log.warning(error) for key, tuple_ in cls.JMX_MAPPING.items(): object_name = cls.__get_object_name_from_tuple(tuple_) # Check to see if the object exists, if it doesnt we will throw # an error which will be handled silently by continuing through # the for loop. options = [] for connection_key, connection in connections.items(): try: connection.getObjectInstance(object_name) options.append(connection_key) except Exception: cls.log.exception( "Tomcat (JMX) plugin - %s bean not found at %s." % (object_name, connection_key) ) continue if len(connections.keys()) >= 1 and not options: # No connection was able to get this value. Set it to unsupported. options = None status = agent_util.UNSUPPORTED msg = "Unreachable %s at any connection" % key else: # We found options. Is supported. msg = "" status = agent_util.SUPPORTED label, unit = tuple_[0], tuple_[6] result[key] = { "label": label, "options": options, "status": status, "error_message": msg, "unit": unit, } return result @classmethod def check(cls, textkey, data, config): """returns a value for the metric. :type textkey: string :param textkey: Canonical name for a metric. :type data: string :param data: Specific option to check for. :type config: dict :param config: Mapping of information under the application block for this plugin. :rtype: double :return: Value for a specific metric """ entries = cls.get_connections_from_config(config) if data: for entry in entries.values(): possible_match = "%s:%s" % (entry["host"], entry["port"]) if possible_match == data: config = entry else: # Default to the first configuration config = entries[0] status, connection, msg = cls.__get_connection(config) if msg: cls.log.info("Failed to get a connection: %s" % msg) return None tuple_ = cls.JMX_MAPPING.get(textkey) attribute_name, composite_data_key = tuple_[4], tuple_[5] # Create an ObjectName object to lookup object_name = cls.__get_object_name_from_tuple(tuple_) try: object_instance = connection.getObjectInstance(object_name) attribute_value = connection.getAttribute( object_instance.getObjectName(), attribute_name ) attribute_class_name = attribute_value.__class__.__name__ # If the object returned is just a numeric value if "CompositeDataSupport" not in attribute_class_name: return attribute_value.floatValue() # If the attribute object does not have the composite data key # return none if not attribute_value.containsKey(composite_data_key): return None # If the object returned is of type CompositeDataSupport return the # correct value from that object check_result = attribute_value.get(composite_data_key) return check_result.floatValue() except: cls.log.info("Tomcat (JMX) plugin - %s bean not found." % object_name) return 0