jlog.cpp 91 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800
  1. /*##############################################################################
  2. HPCC SYSTEMS software Copyright (C) 2012 HPCC Systems®.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. ############################################################################## */
  13. #include "platform.h"
  14. #include "build-config.h"
  15. #include <algorithm>
  16. #include "stdio.h"
  17. #include "jlog.hpp"
  18. #include "jlog.ipp"
  19. #include "jmutex.hpp"
  20. #include "jarray.hpp"
  21. #include "jsocket.hpp"
  22. #include "jmisc.hpp"
  23. #include "jprop.hpp"
  24. #define MSGCOMP_NUMBER 1000
  25. #define FILE_LOG_ENABLES_QUEUEUING
  26. #ifndef _WIN32
  27. #define AUDIT_DATA_LOG_TEMPLATE "/var/log/seisint/log_data_XXXXXX"
  28. #endif
  29. // Time, in nanoseconds, after which the clock field loops --- 3600000000000ns = 1hr
  30. #define CLOCK_LOOP_NANOSECONDS I64C(3600000000000)
  31. // LogMsgSysInfo
  32. static FILE *getNullHandle()
  33. {
  34. #ifdef _WIN32
  35. return fopen("nul","w");
  36. #else
  37. return fopen("/dev/null","w");
  38. #endif
  39. }
  40. LogMsgSysInfo::LogMsgSysInfo(LogMsgId _id, unsigned port, LogMsgSessionId session)
  41. {
  42. id = _id;
  43. #ifdef _WIN32
  44. // Hack for the fact that Windows doesn't handle gettimeofday
  45. // Subsecond timing granularities in log files will not be available
  46. time(&timeStarted);
  47. #else
  48. gettimeofday(&timeStarted, NULL);
  49. #endif
  50. processID = GetCurrentProcessId();
  51. threadID = threadLogID();
  52. sessionID = session;
  53. node.setLocalHost(port);
  54. }
  55. void LogMsgSysInfo::serialize(MemoryBuffer & out) const
  56. {
  57. out.append(id).append((unsigned) queryTime()).append(processID).append(threadID).append(sessionID); node.serialize(out);
  58. }
  59. void LogMsgSysInfo::deserialize(MemoryBuffer & in)
  60. {
  61. unsigned t;
  62. in.read(id).read(t).read(processID).read(threadID).read(sessionID); node.deserialize(in);
  63. #ifdef _WIN32
  64. timeStarted = t;
  65. #else
  66. timeStarted.tv_sec = t;
  67. timeStarted.tv_usec = 0; // For back-compatibility reasons, the subsecond timings are not serialized
  68. #endif
  69. }
  70. // LogMsg
  71. StringBuffer & LogMsg::toStringPlain(StringBuffer & out, unsigned fields) const
  72. {
  73. out.ensureCapacity(LOG_MSG_FORMAT_BUFFER_LENGTH);
  74. if(fields & MSGFIELD_audience)
  75. out.append("aud=").append(LogMsgAudienceToVarString(category.queryAudience())).append(' ');
  76. if(fields & MSGFIELD_class)
  77. out.append("cls=").append(LogMsgClassToVarString(category.queryClass())).append(' ');
  78. if(fields & MSGFIELD_detail)
  79. out.appendf("det=%d ", category.queryDetail());
  80. if(fields & MSGFIELD_msgID)
  81. out.appendf("id=%X ", sysInfo.queryMsgID());
  82. if(fields & MSGFIELD_timeDate)
  83. {
  84. time_t timeNum = sysInfo.queryTime();
  85. char timeString[12];
  86. struct tm timeStruct;
  87. localtime_r(&timeNum, &timeStruct);
  88. if(fields & MSGFIELD_date)
  89. {
  90. strftime(timeString, 12, "%Y-%m-%d ", &timeStruct);
  91. out.append(timeString);
  92. }
  93. if(fields & MSGFIELD_microTime)
  94. {
  95. out.appendf("%02d:%02d:%02d.%06d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs());
  96. }
  97. else if(fields & MSGFIELD_milliTime)
  98. {
  99. out.appendf("%02d:%02d:%02d.%03d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs()/1000);
  100. }
  101. else if(fields & MSGFIELD_time)
  102. {
  103. strftime(timeString, 12, "%H:%M:%S ", &timeStruct);
  104. out.append(timeString);
  105. }
  106. }
  107. if(fields & MSGFIELD_process)
  108. out.appendf("pid=%d ",sysInfo.queryProcessID());
  109. if(fields & MSGFIELD_thread)
  110. out.appendf("tid=%d ",sysInfo.queryThreadID());
  111. if(fields & MSGFIELD_session)
  112. {
  113. if(sysInfo.querySessionID() == UnknownSession)
  114. out.append("sid=unknown ");
  115. else
  116. out.appendf("sid=%" I64F "u ", sysInfo.querySessionID());
  117. }
  118. if(fields & MSGFIELD_node)
  119. {
  120. sysInfo.queryNode()->getUrlStr(out);
  121. out.append(" ");
  122. }
  123. if(fields & MSGFIELD_job)
  124. {
  125. if(jobInfo.queryJobID() == UnknownJob)
  126. out.append("job=unknown ");
  127. else
  128. out.appendf("job=%" I64F "u ", jobInfo.queryJobID());
  129. }
  130. if(fields & MSGFIELD_user)
  131. {
  132. if(jobInfo.queryUserID() == UnknownUser)
  133. out.append("usr=unknown ");
  134. else
  135. out.appendf("usr=%" I64F "u ", jobInfo.queryUserID());
  136. }
  137. if(fields & MSGFIELD_component)
  138. out.appendf("cmp=%u ", component);
  139. if (fields & MSGFIELD_quote)
  140. out.append('"');
  141. if (fields & MSGFIELD_prefix)
  142. out.append(msgPrefix(category.queryClass()));
  143. if((fields & MSGFIELD_code) && (msgCode != NoLogMsgCode))
  144. out.append(msgCode).append(": ").append(text.str());
  145. else
  146. out.append(text.str());
  147. if (fields & MSGFIELD_quote)
  148. out.append('"');
  149. return out;
  150. }
  151. StringBuffer & LogMsg::toStringXML(StringBuffer & out, unsigned fields) const
  152. {
  153. out.ensureCapacity(LOG_MSG_FORMAT_BUFFER_LENGTH);
  154. out.append("<msg ");
  155. if(fields & MSGFIELD_audience)
  156. out.append("Audience=\"").append(LogMsgAudienceToVarString(category.queryAudience())).append("\" ");
  157. if(fields & MSGFIELD_class)
  158. out.append("Class=\"").append(LogMsgClassToVarString(category.queryClass())).append("\" ");
  159. if(fields & MSGFIELD_detail)
  160. out.append("Detail=\"").append(category.queryDetail()).append("\" ");
  161. #ifdef LOG_MSG_NEWLINE
  162. if(fields & MSGFIELD_allCategory) out.append("\n ");
  163. #endif
  164. if(fields & MSGFIELD_msgID)
  165. out.append("MessageID=\"").append(sysInfo.queryMsgID()).append("\" ");
  166. if(fields & MSGFIELD_timeDate)
  167. {
  168. time_t timeNum = sysInfo.queryTime();
  169. char timeString[20];
  170. struct tm timeStruct;
  171. localtime_r(&timeNum, &timeStruct);
  172. if(fields & MSGFIELD_date)
  173. {
  174. strftime(timeString, 20, "date=\"%Y-%m-%d\" ", &timeStruct);
  175. out.append(timeString);
  176. }
  177. if(fields & MSGFIELD_microTime)
  178. {
  179. out.appendf("time=\"%02d:%02d:%02d.%06d\" ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs());
  180. }
  181. else if(fields & MSGFIELD_milliTime)
  182. {
  183. out.appendf("time=\"%02d:%02d:%02d.%03d\" ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs()/1000);
  184. }
  185. else if(fields & MSGFIELD_time)
  186. {
  187. strftime(timeString, 20, "time=\"%H:%M:%S\" ", &timeStruct);
  188. out.append(timeString);
  189. }
  190. }
  191. if(fields & MSGFIELD_process)
  192. out.append("PID=\"").append(sysInfo.queryProcessID()).append("\" ");
  193. if(fields & MSGFIELD_thread)
  194. out.append("TID=\"").append(sysInfo.queryThreadID()).append("\" ");
  195. if(fields & MSGFIELD_session)
  196. {
  197. if(sysInfo.querySessionID() == UnknownSession)
  198. out.append("SessionID=\"unknown\" ");
  199. else
  200. out.append("SessionID=\"").append(sysInfo.querySessionID()).append("\" ");
  201. }
  202. if(fields & MSGFIELD_node)
  203. {
  204. out.append("Node=\"");
  205. sysInfo.queryNode()->getUrlStr(out);
  206. out.append("\" ");
  207. }
  208. #ifdef LOG_MSG_NEWLINE
  209. if(fields & MSGFIELD_allSysInfo) out.append("\n ");
  210. #endif
  211. if(fields & MSGFIELD_job)
  212. {
  213. if(jobInfo.queryJobID() == UnknownJob)
  214. out.append("JobID=\"unknown\" ");
  215. else
  216. out.append("JobID=\"").append(jobInfo.queryJobID()).append("\" ");
  217. }
  218. if(fields & MSGFIELD_user)
  219. {
  220. if(jobInfo.queryUserID() == UnknownUser)
  221. out.append("UserID=\"unknown\" ");
  222. else
  223. out.append("UserID=\"").append(jobInfo.queryUserID()).append("\" ");
  224. }
  225. #ifdef LOG_MSG_NEWLINE
  226. if(fields & MSGFIELD_allJobInfo) out.append("\n ");
  227. #endif
  228. if(fields & MSGFIELD_component) out.append("Component=\"").append(component).append("\" ");
  229. if((fields & MSGFIELD_code) && (msgCode != NoLogMsgCode))
  230. out.append("code=\"").append(msgCode).append("\" ");
  231. out.append("text=\"").append(text.str()).append("\" />\n");
  232. return out;
  233. }
  234. StringBuffer & LogMsg::toStringTable(StringBuffer & out, unsigned fields) const
  235. {
  236. out.ensureCapacity(LOG_MSG_FORMAT_BUFFER_LENGTH);
  237. if(fields & MSGFIELD_audience)
  238. out.append(LogMsgAudienceToFixString(category.queryAudience()));
  239. if(fields & MSGFIELD_class)
  240. out.append(LogMsgClassToFixString(category.queryClass()));
  241. if(fields & MSGFIELD_detail)
  242. out.appendf("%10d ", category.queryDetail());
  243. if(fields & MSGFIELD_msgID)
  244. out.appendf("%8X ", sysInfo.queryMsgID());
  245. if(fields & MSGFIELD_timeDate)
  246. {
  247. time_t timeNum = sysInfo.queryTime();
  248. char timeString[12];
  249. struct tm timeStruct;
  250. localtime_r(&timeNum, &timeStruct);
  251. if(fields & MSGFIELD_date)
  252. {
  253. strftime(timeString, 12, "%Y-%m-%d ", &timeStruct);
  254. out.append(timeString);
  255. }
  256. if(fields & MSGFIELD_microTime)
  257. {
  258. out.appendf("%02d:%02d:%02d.%06d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs());
  259. }
  260. else if(fields & MSGFIELD_milliTime)
  261. {
  262. out.appendf("%02d:%02d:%02d.%03d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs()/1000);
  263. }
  264. else if(fields & MSGFIELD_time)
  265. {
  266. strftime(timeString, 12, "%H:%M:%S ", &timeStruct);
  267. out.append(timeString);
  268. }
  269. }
  270. if(fields & MSGFIELD_process)
  271. out.appendf("%5d ",sysInfo.queryProcessID());
  272. if(fields & MSGFIELD_thread)
  273. out.appendf("%5d ",sysInfo.queryThreadID());
  274. if(fields & MSGFIELD_session)
  275. {
  276. if(sysInfo.querySessionID() == UnknownSession)
  277. out.append(" unknown ");
  278. else
  279. out.appendf("%20" I64F "u ", sysInfo.querySessionID());
  280. }
  281. if(fields & MSGFIELD_node)
  282. {
  283. size32_t len = out.length();
  284. sysInfo.queryNode()->getUrlStr(out);
  285. out.appendN(20 + len - out.length(), ' ');
  286. }
  287. if(fields & MSGFIELD_job)
  288. {
  289. if(jobInfo.queryJobID() == UnknownJob)
  290. out.append("unknown ");
  291. else
  292. out.appendf("%7" I64F "u ", jobInfo.queryJobID());
  293. }
  294. if(fields & MSGFIELD_user)
  295. {
  296. if(jobInfo.queryUserID() == UnknownUser)
  297. out.append("unknown ");
  298. else
  299. out.appendf("%7" I64F "u ", jobInfo.queryUserID());
  300. }
  301. if(fields & MSGFIELD_component)
  302. out.appendf("%6u ", component);
  303. if (fields & MSGFIELD_quote)
  304. out.append('"');
  305. if (fields & MSGFIELD_prefix)
  306. out.append(msgPrefix(category.queryClass()));
  307. if((fields & MSGFIELD_code) && (msgCode != NoLogMsgCode))
  308. out.append(msgCode).append(": ").append(text.str());
  309. else
  310. out.append(text.str());
  311. if (fields & MSGFIELD_quote)
  312. out.append('"');
  313. out.append('\n');
  314. return out;
  315. }
  316. StringBuffer & LogMsg::toStringTableHead(StringBuffer & out, unsigned fields)
  317. {
  318. if(fields & MSGFIELD_audience)
  319. out.append("Audience ");
  320. if(fields & MSGFIELD_class)
  321. out.append("Class ");
  322. if(fields & MSGFIELD_detail)
  323. out.append(" Detail ");
  324. if(fields & MSGFIELD_msgID)
  325. out.append(" MsgID ");
  326. if(fields & MSGFIELD_date)
  327. out.append(" Date ");
  328. if(fields & (MSGFIELD_microTime | MSGFIELD_milliTime | MSGFIELD_time))
  329. out.append(" Time ");
  330. if(fields & MSGFIELD_process)
  331. out.append(" PID ");
  332. if(fields & MSGFIELD_thread)
  333. out.append(" TID ");
  334. if(fields & MSGFIELD_session)
  335. out.append(" SessionID ");
  336. if(fields & MSGFIELD_node)
  337. out.append(" Node ");
  338. if(fields & MSGFIELD_job)
  339. out.append(" JobID ");
  340. if(fields & MSGFIELD_user)
  341. out.append(" UserID ");
  342. if(fields & MSGFIELD_component)
  343. out.append(" Compo ");
  344. out.append("\n\n");
  345. return out;
  346. }
  347. void LogMsg::fprintPlain(FILE * handle, unsigned fields) const
  348. {
  349. if(fields & MSGFIELD_audience)
  350. fprintf(handle, "aud=%s", LogMsgAudienceToVarString(category.queryAudience()));
  351. if(fields & MSGFIELD_class)
  352. fprintf(handle, "cls=%s", LogMsgClassToVarString(category.queryClass()));
  353. if(fields & MSGFIELD_detail)
  354. fprintf(handle, "det=%d ", category.queryDetail());
  355. if(fields & MSGFIELD_msgID)
  356. fprintf(handle, "id=%X ", sysInfo.queryMsgID());
  357. if(fields & MSGFIELD_timeDate)
  358. {
  359. time_t timeNum = sysInfo.queryTime();
  360. char timeString[12];
  361. struct tm timeStruct;
  362. localtime_r(&timeNum, &timeStruct);
  363. if(fields & MSGFIELD_date)
  364. {
  365. strftime(timeString, 12, "%Y-%m-%d ", &timeStruct);
  366. fputs(timeString, handle);
  367. }
  368. if(fields & MSGFIELD_microTime)
  369. {
  370. fprintf(handle, "%02d:%02d:%02d.%06d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs());
  371. }
  372. else if(fields & MSGFIELD_milliTime)
  373. {
  374. fprintf(handle, "%02d:%02d:%02d.%03d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs()/1000);
  375. }
  376. else if(fields & MSGFIELD_time)
  377. {
  378. strftime(timeString, 12, "%H:%M:%S ", &timeStruct);
  379. fputs(timeString, handle);
  380. }
  381. }
  382. if(fields & MSGFIELD_process)
  383. fprintf(handle, "pid=%d ",sysInfo.queryProcessID());
  384. if(fields & MSGFIELD_thread)
  385. fprintf(handle, "tid=%d ",sysInfo.queryThreadID());
  386. if(fields & MSGFIELD_session)
  387. {
  388. if(sysInfo.querySessionID() == UnknownSession)
  389. fprintf(handle, "sid=unknown ");
  390. else
  391. fprintf(handle, "sid=%" I64F "u ", sysInfo.querySessionID());
  392. }
  393. if(fields & MSGFIELD_node)
  394. {
  395. StringBuffer buff;
  396. sysInfo.queryNode()->getUrlStr(buff);
  397. fprintf(handle, "%s ", buff.str());
  398. }
  399. if(fields & MSGFIELD_job)
  400. {
  401. if(jobInfo.queryJobID() == UnknownJob)
  402. fprintf(handle, "job=unknown ");
  403. else
  404. fprintf(handle, "job=%" I64F "u ", jobInfo.queryJobID());
  405. }
  406. if(fields & MSGFIELD_user)
  407. {
  408. if(jobInfo.queryUserID() == UnknownUser)
  409. fprintf(handle, "usr=unknown ");
  410. else
  411. fprintf(handle, "usr=%" I64F "u ", jobInfo.queryUserID());
  412. }
  413. if(fields & MSGFIELD_component)
  414. fprintf(handle, "cmp=%u ", component);
  415. const char * quote = (fields & MSGFIELD_quote) ? "\"" : "";
  416. const char * prefix = (fields & MSGFIELD_prefix) ? msgPrefix(category.queryClass()) : "";
  417. if((fields & MSGFIELD_code) && (msgCode != NoLogMsgCode))
  418. fprintf(handle, "%s%s%d: %s%s", quote, prefix, msgCode, text.str(), quote);
  419. else
  420. fprintf(handle, "%s%s%s%s", quote, prefix, text.str(), quote);
  421. }
  422. void LogMsg::fprintXML(FILE * handle, unsigned fields) const
  423. {
  424. fprintf(handle, "<msg ");
  425. if(fields & MSGFIELD_audience)
  426. fprintf(handle, "Audience=\"%s\" ", LogMsgAudienceToVarString(category.queryAudience()));
  427. if(fields & MSGFIELD_class)
  428. fprintf(handle, "Class=\"%s\" ", LogMsgClassToVarString(category.queryClass()));
  429. if(fields & MSGFIELD_detail)
  430. fprintf(handle, "Detail=\"%d\" ", category.queryDetail());
  431. #ifdef LOG_MSG_NEWLINE
  432. if(fields & MSGFIELD_allCategory) fprintf(handle, "\n ");
  433. #endif
  434. if(fields & MSGFIELD_msgID)
  435. fprintf(handle, "MessageID=\"%d\" ",sysInfo.queryMsgID());
  436. if(fields & MSGFIELD_timeDate)
  437. {
  438. time_t timeNum = sysInfo.queryTime();
  439. char timeString[20];
  440. struct tm timeStruct;
  441. localtime_r(&timeNum, &timeStruct);
  442. if(fields & MSGFIELD_date)
  443. {
  444. strftime(timeString, 20, "date=\"%Y-%m-%d\" ", &timeStruct);
  445. fputs(timeString, handle);
  446. }
  447. if(fields & MSGFIELD_microTime)
  448. {
  449. fprintf(handle, "time=\"%02d:%02d:%02d.%06d\" ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs());
  450. }
  451. else if(fields & MSGFIELD_milliTime)
  452. {
  453. fprintf(handle, "time=\"%02d:%02d:%02d.%03d\" ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs()/1000);
  454. }
  455. else if(fields & MSGFIELD_time)
  456. {
  457. strftime(timeString, 20, "time=\"%H:%M:%S\" ", &timeStruct);
  458. fputs(timeString, handle);
  459. }
  460. }
  461. if(fields & MSGFIELD_process)
  462. fprintf(handle, "PID=\"%d\" ", sysInfo.queryProcessID());
  463. if(fields & MSGFIELD_thread)
  464. fprintf(handle, "TID=\"%d\" ", sysInfo.queryThreadID());
  465. if(fields & MSGFIELD_session)
  466. {
  467. if(sysInfo.querySessionID() == UnknownSession)
  468. fprintf(handle, "SessionID=\"unknown\" ");
  469. else
  470. fprintf(handle, "SessionID=\"%" I64F "u\" ", sysInfo.querySessionID());
  471. }
  472. if(fields & MSGFIELD_node)
  473. {
  474. StringBuffer buff;
  475. sysInfo.queryNode()->getUrlStr(buff);
  476. fprintf(handle, "Node=\"%s\" ", buff.str());
  477. }
  478. #ifdef LOG_MSG_NEWLINE
  479. if(fields & MSGFIELD_allSysInfo) fprintf(handle, "\n ");
  480. #endif
  481. if(fields & MSGFIELD_job)
  482. {
  483. if(jobInfo.queryJobID() == UnknownJob)
  484. fprintf(handle, "JobID=\"unknown\" ");
  485. else
  486. fprintf(handle, "JobID=\"%" I64F "u\" ", jobInfo.queryJobID());
  487. }
  488. if(fields & MSGFIELD_user)
  489. {
  490. if(jobInfo.queryUserID() == UnknownUser)
  491. fprintf(handle, "UserID=\"unknown\" ");
  492. else
  493. fprintf(handle, "UserID=\"%" I64F "u\" ", jobInfo.queryUserID());
  494. }
  495. if(fields & MSGFIELD_component)
  496. fprintf(handle, "Component=\"%6u\" ", component);
  497. #ifdef LOG_MSG_NEWLINE
  498. if(fields & MSGFIELD_allJobInfo) fprintf(handle, "\n ");
  499. #endif
  500. if((fields & MSGFIELD_code) && (msgCode != NoLogMsgCode))
  501. fprintf(handle, "code=\"%d\" ", msgCode);
  502. fprintf(handle, "text=\"%s\" />\n", text.str());
  503. }
  504. void LogMsg::fprintTable(FILE * handle, unsigned fields) const
  505. {
  506. if(fields & MSGFIELD_audience)
  507. fputs(LogMsgAudienceToFixString(category.queryAudience()), handle);
  508. if(fields & MSGFIELD_class)
  509. fputs(LogMsgClassToFixString(category.queryClass()), handle);
  510. if(fields & MSGFIELD_detail)
  511. fprintf(handle, "%10d ", category.queryDetail());
  512. if(fields & MSGFIELD_msgID)
  513. fprintf(handle, "%08X ", sysInfo.queryMsgID());
  514. if(fields & MSGFIELD_timeDate)
  515. {
  516. time_t timeNum = sysInfo.queryTime();
  517. char timeString[12];
  518. struct tm timeStruct;
  519. localtime_r(&timeNum, &timeStruct);
  520. if(fields & MSGFIELD_date)
  521. {
  522. strftime(timeString, 12, "%Y-%m-%d ", &timeStruct);
  523. fputs(timeString, handle);
  524. }
  525. if(fields & MSGFIELD_microTime)
  526. {
  527. fprintf(handle, "%02d:%02d:%02d.%06d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs());
  528. }
  529. else if(fields & MSGFIELD_milliTime)
  530. {
  531. fprintf(handle, "%02d:%02d:%02d.%03d ", timeStruct.tm_hour, timeStruct.tm_min, timeStruct.tm_sec, sysInfo.queryUSecs()/1000);
  532. }
  533. else if(fields & MSGFIELD_time)
  534. {
  535. strftime(timeString, 12, "%H:%M:%S ", &timeStruct);
  536. fputs(timeString, handle);
  537. }
  538. }
  539. if(fields & MSGFIELD_process)
  540. fprintf(handle, "%5d ",sysInfo.queryProcessID());
  541. if(fields & MSGFIELD_thread)
  542. fprintf(handle, "%5d ",sysInfo.queryThreadID());
  543. if(fields & MSGFIELD_session)
  544. {
  545. if(sysInfo.querySessionID() == UnknownSession)
  546. fprintf(handle, " unknown ");
  547. else
  548. fprintf(handle, "%20" I64F "u ", sysInfo.querySessionID());
  549. }
  550. if(fields & MSGFIELD_node)
  551. {
  552. StringBuffer buff;
  553. static const char * twenty_spaces = " ";
  554. sysInfo.queryNode()->getUrlStr(buff);
  555. fprintf(handle, "%s%s", buff.str(), (buff.length()<=20) ? twenty_spaces+buff.length() : "");
  556. }
  557. if(fields & MSGFIELD_job)
  558. {
  559. if(jobInfo.queryJobID() == UnknownJob)
  560. fprintf(handle, "unknown ");
  561. else
  562. fprintf(handle, "%7" I64F "u ", jobInfo.queryJobID());
  563. }
  564. if(fields & MSGFIELD_user)
  565. {
  566. if(jobInfo.queryUserID() == UnknownUser)
  567. fprintf(handle, "unknown ");
  568. else
  569. fprintf(handle, "%7" I64F "u ", jobInfo.queryUserID());
  570. }
  571. if(fields & MSGFIELD_component)
  572. fprintf(handle, "%6u ", component);
  573. const char * quote = (fields & MSGFIELD_quote) ? "\"" : "";
  574. const char * prefix = (fields & MSGFIELD_prefix) ? msgPrefix(category.queryClass()) : "";
  575. if((fields & MSGFIELD_code) && (msgCode != NoLogMsgCode))
  576. fprintf(handle, "%s%s%d: %s%s\n", quote, prefix, msgCode, text.str(), quote);
  577. else
  578. fprintf(handle, "%s%s%s%s\n", quote, prefix, text.str(), quote);
  579. }
  580. void LogMsg::fprintTableHead(FILE * handle, unsigned fields)
  581. {
  582. if(fields & MSGFIELD_audience)
  583. fprintf(handle, "Audience ");
  584. if(fields & MSGFIELD_class)
  585. fprintf(handle, "Class ");
  586. if(fields & MSGFIELD_detail)
  587. fprintf(handle, " Detail ");
  588. if(fields & MSGFIELD_msgID)
  589. fprintf(handle, " MsgID ");
  590. if(fields & MSGFIELD_date)
  591. fprintf(handle, " Date ");
  592. if(fields & MSGFIELD_time)
  593. fprintf(handle, " Time ");
  594. if(fields & MSGFIELD_process)
  595. fprintf(handle, " PID ");
  596. if(fields & MSGFIELD_thread)
  597. fprintf(handle, " TID ");
  598. if(fields & MSGFIELD_session)
  599. fprintf(handle, " SessionID ");
  600. if(fields & MSGFIELD_node)
  601. fprintf(handle, " Node ");
  602. if(fields & MSGFIELD_job)
  603. fprintf(handle, " JobID ");
  604. if(fields & MSGFIELD_user)
  605. fprintf(handle, " UserID ");
  606. if(fields & MSGFIELD_component)
  607. fprintf(handle, " Compo ");
  608. fprintf(handle, "\n\n");
  609. }
  610. // Implementations of ILogMsgFilter
  611. void PassAllLogMsgFilter::addToPTree(IPropertyTree * tree) const
  612. {
  613. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  614. filterTree->setProp("@type", "all");
  615. tree->addPropTree("filter", filterTree);
  616. }
  617. void PassLocalLogMsgFilter::addToPTree(IPropertyTree * tree) const
  618. {
  619. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  620. filterTree->setProp("@type", "local");
  621. tree->addPropTree("filter", filterTree);
  622. }
  623. void PassNoneLogMsgFilter::addToPTree(IPropertyTree * tree) const
  624. {
  625. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  626. filterTree->setProp("@type", "none");
  627. tree->addPropTree("filter", filterTree);
  628. }
  629. void CategoryLogMsgFilter::addToPTree(IPropertyTree * tree) const
  630. {
  631. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  632. filterTree->setProp("@type", "category");
  633. filterTree->setPropInt("@audience", audienceMask);
  634. filterTree->setPropInt("@class", classMask);
  635. filterTree->setPropInt("@detail", maxDetail);
  636. if(localFlag) filterTree->setPropInt("@local", 1);
  637. tree->addPropTree("filter", filterTree);
  638. }
  639. void PIDLogMsgFilter::addToPTree(IPropertyTree * tree) const
  640. {
  641. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  642. filterTree->setProp("@type", "pid");
  643. filterTree->setPropInt("@pid", pid);
  644. if(localFlag) filterTree->setPropInt("@local", 1);
  645. tree->addPropTree("filter", filterTree);
  646. }
  647. void TIDLogMsgFilter::addToPTree(IPropertyTree * tree) const
  648. {
  649. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  650. filterTree->setProp("@type", "tid");
  651. filterTree->setPropInt("@tid", tid);
  652. if(localFlag) filterTree->setPropInt("@local", 1);
  653. tree->addPropTree("filter", filterTree);
  654. }
  655. void NodeLogMsgFilter::addToPTree(IPropertyTree * tree) const
  656. {
  657. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  658. filterTree->setProp("@type", "node");
  659. StringBuffer buff;
  660. node.getIpText(buff);
  661. filterTree->setProp("@ip", buff.str());
  662. filterTree->setPropInt("@port", node.port);
  663. if(localFlag) filterTree->setPropInt("@local", 1);
  664. tree->addPropTree("filter", filterTree);
  665. }
  666. void IpLogMsgFilter::addToPTree(IPropertyTree * tree) const
  667. {
  668. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  669. filterTree->setProp("@type", "ip");
  670. StringBuffer buff;
  671. ip.getIpText(buff);
  672. filterTree->setProp("@ip", buff.str());
  673. if(localFlag) filterTree->setPropInt("@local", 1);
  674. tree->addPropTree("filter", filterTree);
  675. }
  676. void JobLogMsgFilter::addToPTree(IPropertyTree * tree) const
  677. {
  678. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  679. filterTree->setProp("@type", "job");
  680. filterTree->setPropInt("@job", (int)job);
  681. if(localFlag) filterTree->setPropInt("@local", 1);
  682. tree->addPropTree("filter", filterTree);
  683. }
  684. void UserLogMsgFilter::addToPTree(IPropertyTree * tree) const
  685. {
  686. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  687. filterTree->setProp("@type", "user");
  688. filterTree->setPropInt("@user", (int)user);
  689. if(localFlag) filterTree->setPropInt("@local", 1);
  690. tree->addPropTree("filter", filterTree);
  691. }
  692. void SessionLogMsgFilter::addToPTree(IPropertyTree * tree) const
  693. {
  694. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  695. filterTree->setProp("@type", "session");
  696. filterTree->setPropInt("@session", (int)session);
  697. if(localFlag) filterTree->setPropInt("@local", 1);
  698. tree->addPropTree("filter", filterTree);
  699. }
  700. void ComponentLogMsgFilter::addToPTree(IPropertyTree * tree) const
  701. {
  702. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  703. filterTree->setProp("@type", "component");
  704. filterTree->setPropInt("@component", component);
  705. if(localFlag) filterTree->setPropInt("@local", 1);
  706. tree->addPropTree("filter", filterTree);
  707. }
  708. bool RegexLogMsgFilter::includeMessage(const LogMsg & msg) const
  709. {
  710. if(localFlag && msg.queryRemoteFlag()) return false;
  711. SpinBlock b(lock);
  712. return const_cast<RegExpr &>(regex).find(msg.queryText()) != NULL;
  713. }
  714. void RegexLogMsgFilter::addToPTree(IPropertyTree * tree) const
  715. {
  716. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  717. filterTree->setProp("@type", "regex");
  718. filterTree->setProp("@regex", regexText);
  719. if(localFlag) filterTree->setPropInt("@local", 1);
  720. tree->addPropTree("filter", filterTree);
  721. }
  722. void NotLogMsgFilter::addToPTree(IPropertyTree * tree) const
  723. {
  724. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  725. filterTree->setProp("@type", "not");
  726. arg->addToPTree(filterTree);
  727. tree->addPropTree("filter", filterTree);
  728. }
  729. void AndLogMsgFilter::addToPTree(IPropertyTree * tree) const
  730. {
  731. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  732. filterTree->setProp("@type", "and");
  733. arg1->addToPTree(filterTree);
  734. arg2->addToPTree(filterTree);
  735. tree->addPropTree("filter", filterTree);
  736. }
  737. void OrLogMsgFilter::addToPTree(IPropertyTree * tree) const
  738. {
  739. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  740. filterTree->setProp("@type", "or");
  741. arg1->addToPTree(filterTree);
  742. arg2->addToPTree(filterTree);
  743. tree->addPropTree("filter", filterTree);
  744. }
  745. void SwitchLogMsgFilter::addToPTree(IPropertyTree * tree) const
  746. {
  747. IPropertyTree * filterTree = createPTree(ipt_caseInsensitive);
  748. filterTree->setProp("@type", "switch");
  749. cond->addToPTree(filterTree);
  750. yes->addToPTree(filterTree);
  751. no->addToPTree(filterTree);
  752. tree->addPropTree("filter", filterTree);
  753. }
  754. void CategoryLogMsgFilter::orWithFilter(const ILogMsgFilter * filter)
  755. {
  756. audienceMask |= filter->queryAudienceMask();
  757. classMask |= filter->queryClassMask();
  758. maxDetail = std::max(maxDetail, filter->queryMaxDetail());
  759. }
  760. void CategoryLogMsgFilter::reset()
  761. {
  762. audienceMask = 0;
  763. classMask = 0;
  764. maxDetail = 0;
  765. }
  766. // HandleLogMsgHandler
  767. void HandleLogMsgHandlerTable::addToPTree(IPropertyTree * tree) const
  768. {
  769. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  770. if(handle==stderr)
  771. handlerTree->setProp("@type", "stderr");
  772. else
  773. handlerTree->setProp("@type", "mischandle");
  774. handlerTree->setPropInt("@fields", messageFields);
  775. tree->addPropTree("handler", handlerTree);
  776. }
  777. void HandleLogMsgHandlerXML::addToPTree(IPropertyTree * tree) const
  778. {
  779. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  780. if(handle==stderr)
  781. handlerTree->setProp("@type", "stderr");
  782. else
  783. handlerTree->setProp("@type", "mischandle");
  784. handlerTree->setPropInt("@fields", messageFields);
  785. handlerTree->setProp("@writeXML", "true");
  786. tree->addPropTree("handler", handlerTree);
  787. }
  788. // FileLogMsgHandler
  789. FileLogMsgHandler::FileLogMsgHandler(const char * _filename, const char * _headerText, unsigned _fields, bool _append, bool _flushes)
  790. : messageFields(_fields), filename(_filename), headerText(_headerText), append(_append), flushes(_flushes)
  791. {
  792. recursiveCreateDirectoryForFile(filename);
  793. if(append)
  794. handle = fopen(filename, "a");
  795. else
  796. handle = fopen(filename, "w");
  797. if(!handle) {
  798. handle = getNullHandle();
  799. StringBuffer err;
  800. err.appendf("LOGGING: could not open file '%s' for output",filename.get());
  801. ERRLOG("%s",err.str()); // make sure doesn't get lost!
  802. throw MakeStringException(3000,"%s",err.str()); // 3000: internal error
  803. }
  804. if(headerText) fprintf(handle, "--- %s ---\n", (const char *)headerText);
  805. }
  806. static void closeAndDeleteEmpty(const char * filename, FILE *handle)
  807. {
  808. if (handle) {
  809. fpos_t pos;
  810. bool del = (fgetpos(handle, &pos)==0)&&
  811. #if defined( _WIN32) || defined(__FreeBSD__) || defined(__APPLE__)
  812. (pos==0);
  813. #else
  814. (pos.__pos==0);
  815. #endif
  816. fclose(handle);
  817. if (del)
  818. remove(filename);
  819. }
  820. }
  821. FileLogMsgHandler::~FileLogMsgHandler()
  822. {
  823. closeAndDeleteEmpty(filename,handle);
  824. }
  825. char const * FileLogMsgHandler::disable()
  826. {
  827. crit.enter();
  828. fclose(handle);
  829. handle = NULL;
  830. return filename;
  831. }
  832. void FileLogMsgHandler::enable()
  833. {
  834. recursiveCreateDirectoryForFile(filename);
  835. handle = fopen(filename, "a");
  836. if(!handle) {
  837. handle = getNullHandle();
  838. assertex(!"FileLogMsgHandler::enable : could not open file for output");
  839. }
  840. crit.leave();
  841. }
  842. void FileLogMsgHandlerTable::addToPTree(IPropertyTree * tree) const
  843. {
  844. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  845. handlerTree->setProp("@type", "file");
  846. handlerTree->setProp("@filename", filename.get());
  847. if(headerText) handlerTree->setProp("@headertext", headerText.get());
  848. handlerTree->setPropInt("@fields", messageFields);
  849. handlerTree->setProp("@writeTable", "true");
  850. if(append) handlerTree->setProp("@append", "true");
  851. if(flushes) handlerTree->setProp("@flushes", "true");
  852. tree->addPropTree("handler", handlerTree);
  853. }
  854. void FileLogMsgHandlerXML::addToPTree(IPropertyTree * tree) const
  855. {
  856. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  857. handlerTree->setProp("@type", "file");
  858. handlerTree->setProp("@filename", filename.get());
  859. if(headerText) handlerTree->setProp("@headertext", headerText.get());
  860. handlerTree->setPropInt("@fields", messageFields);
  861. if(append) handlerTree->setProp("@append", "true");
  862. if(flushes) handlerTree->setProp("@flushes", "true");
  863. tree->addPropTree("handler", handlerTree);
  864. }
  865. // RollingFileLogMsgHandler
  866. RollingFileLogMsgHandler::RollingFileLogMsgHandler(const char * _filebase, const char * _fileextn, unsigned _fields, bool _append, bool _flushes, const char *initialName, const char *_alias, bool daily)
  867. : handle(0), messageFields(_fields), alias(_alias), filebase(_filebase), fileextn(_fileextn), append(_append), flushes(_flushes)
  868. {
  869. time_t tNow;
  870. time(&tNow);
  871. localtime_r(&tNow, &startTime);
  872. doRollover(daily, initialName);
  873. }
  874. RollingFileLogMsgHandler::~RollingFileLogMsgHandler()
  875. {
  876. closeAndDeleteEmpty(filename,handle);
  877. }
  878. char const * RollingFileLogMsgHandler::disable()
  879. {
  880. crit.enter();
  881. fclose(handle);
  882. return filename;
  883. }
  884. void RollingFileLogMsgHandler::enable()
  885. {
  886. recursiveCreateDirectoryForFile(filename);
  887. handle = fopen(filename, "a");
  888. if(!handle) {
  889. handle = getNullHandle();
  890. assertex(!"RollingFileLogMsgHandler::enable : could not open file for output");
  891. }
  892. crit.leave();
  893. }
  894. void RollingFileLogMsgHandler::addToPTree(IPropertyTree * tree) const
  895. {
  896. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  897. handlerTree->setProp("@type", "rollingfile");
  898. handlerTree->setProp("@filebase", filebase.get());
  899. handlerTree->setProp("@fileextn", fileextn.get());
  900. handlerTree->setPropInt("@fields", messageFields);
  901. if(append) handlerTree->setProp("@append", "true");
  902. if(flushes) handlerTree->setProp("@flushes", "true");
  903. tree->addPropTree("handler", handlerTree);
  904. }
  905. #define ROLLOVER_PERIOD 86400
  906. void RollingFileLogMsgHandler::checkRollover() const
  907. {
  908. time_t tNow;
  909. time(&tNow);
  910. struct tm ltNow;
  911. localtime_r(&tNow, &ltNow);
  912. if(ltNow.tm_year != startTime.tm_year || ltNow.tm_yday != startTime.tm_yday)
  913. {
  914. localtime_r(&tNow, &startTime); // reset the start time for next rollover check
  915. doRollover(true);
  916. }
  917. }
  918. void RollingFileLogMsgHandler::doRollover(bool daily, const char *forceName) const
  919. {
  920. CriticalBlock block(crit);
  921. closeAndDeleteEmpty(filename,handle);
  922. handle = 0;
  923. filename.clear();
  924. if (forceName)
  925. filename.append(forceName);
  926. else
  927. {
  928. filename.clear().append(filebase.get());
  929. addFileTimestamp(filename, daily);
  930. filename.append(fileextn.get());
  931. }
  932. recursiveCreateDirectoryForFile(filename.str());
  933. handle = fopen(filename.str(), append ? "a" : "w");
  934. if (handle && alias && alias.length())
  935. {
  936. fclose(handle);
  937. handle = 0;
  938. remove(alias);
  939. try
  940. {
  941. createHardLink(alias, filename.str());
  942. }
  943. catch (IException *E)
  944. {
  945. recursiveCreateDirectoryForFile(filename.str());
  946. handle = fopen(filename.str(), append ? "a" : "w");
  947. EXCLOG(E); // Log the fact that we could not create the alias - probably it is locked (tail a bit unfortunate on windows).
  948. E->Release();
  949. }
  950. if (!handle)
  951. {
  952. recursiveCreateDirectoryForFile(filename.str());
  953. handle = fopen(filename.str(), append ? "a" : "w");
  954. }
  955. }
  956. if(!handle)
  957. {
  958. handle = getNullHandle();
  959. DBGLOG("RollingFileLogMsgHandler::doRollover : could not open log file %s for output", filename.str());
  960. // actually this is pretty fatal
  961. }
  962. }
  963. // BinLogMsgHandler
  964. BinLogMsgHandler::BinLogMsgHandler(const char * _filename, bool _append) : filename(_filename), append(_append)
  965. {
  966. file.setown(createIFile(filename.get()));
  967. if(!file) assertex(!"BinLogMsgHandler::BinLogMsgHandler : Could not create IFile");
  968. if(append)
  969. fio.setown(file->open(IFOwrite));
  970. else
  971. fio.setown(file->open(IFOcreate));
  972. if(!fio) assertex(!"BinLogMsgHandler::BinLogMsgHandler : Could not create IFileIO");
  973. fstr.setown(createIOStream(fio));
  974. if(!fstr) assertex(!"BinLogMsgHandler::BinLogMsgHandler : Could not create IFileIOStream");
  975. if(append)
  976. fstr->seek(0, IFSend);
  977. }
  978. BinLogMsgHandler::~BinLogMsgHandler()
  979. {
  980. fstr.clear();
  981. fio.clear();
  982. file.clear();
  983. }
  984. void BinLogMsgHandler::handleMessage(const LogMsg & msg) const
  985. {
  986. CriticalBlock block(crit);
  987. mbuff.clear();
  988. msg.serialize(mbuff);
  989. msglen = mbuff.length();
  990. fstr->write(sizeof(msglen), &msglen);
  991. fstr->write(msglen, mbuff.toByteArray());
  992. }
  993. void BinLogMsgHandler::addToPTree(IPropertyTree * tree) const
  994. {
  995. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  996. handlerTree->setProp("@type", "binary");
  997. handlerTree->setProp("@filename", filename.get());
  998. if(append) handlerTree->setProp("@append", "true");
  999. tree->addPropTree("handler", handlerTree);
  1000. }
  1001. char const * BinLogMsgHandler::disable()
  1002. {
  1003. crit.enter();
  1004. fstr.clear();
  1005. fio.clear();
  1006. return filename.get();
  1007. }
  1008. void BinLogMsgHandler::enable()
  1009. {
  1010. fio.setown(file->open(IFOwrite));
  1011. if(!fio) assertex(!"BinLogMsgHandler::enable : Could not create IFileIO");
  1012. fstr.setown(createIOStream(fio));
  1013. if(!fstr) assertex(!"BinLogMsgHandler::enable : Could not create IFileIOStream");
  1014. fstr->seek(0, IFSend);
  1015. crit.leave();
  1016. }
  1017. // LogMsgComponentReporter
  1018. void LogMsgComponentReporter::report(const LogMsgCategory & cat, const char * format, ...)
  1019. {
  1020. va_list args;
  1021. va_start(args, format);
  1022. queryLogMsgManager()->report_va(component, cat, unknownJob, format, args);
  1023. va_end(args);
  1024. }
  1025. void LogMsgComponentReporter::report_va(const LogMsgCategory & cat, const char * format, va_list args)
  1026. {
  1027. queryLogMsgManager()->report_va(component, cat, unknownJob, format, args);
  1028. }
  1029. void LogMsgComponentReporter::report(const LogMsgCategory & cat, LogMsgCode code, const char * format, ...)
  1030. {
  1031. va_list args;
  1032. va_start(args, format);
  1033. queryLogMsgManager()->report_va(component, cat, unknownJob, code, format, args);
  1034. va_end(args);
  1035. }
  1036. void LogMsgComponentReporter::report_va(const LogMsgCategory & cat, LogMsgCode code, const char * format, va_list args)
  1037. {
  1038. queryLogMsgManager()->report_va(component, cat, unknownJob, code, format, args);
  1039. }
  1040. void LogMsgComponentReporter::report(const LogMsgCategory & cat, const IException * exception, const char * prefix)
  1041. {
  1042. StringBuffer buff;
  1043. if(prefix) buff.append(prefix).append(" : ");
  1044. exception->errorMessage(buff);
  1045. queryLogMsgManager()->report(component, cat, unknownJob, exception->errorCode(), "%s", buff.str());
  1046. }
  1047. void LogMsgComponentReporter::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, ...)
  1048. {
  1049. va_list args;
  1050. va_start(args, format);
  1051. queryLogMsgManager()->report_va(component, cat, job, format, args);
  1052. va_end(args);
  1053. }
  1054. void LogMsgComponentReporter::report_va(const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, va_list args)
  1055. {
  1056. queryLogMsgManager()->report_va(component, cat, job, format, args);
  1057. }
  1058. void LogMsgComponentReporter::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, ...)
  1059. {
  1060. va_list args;
  1061. va_start(args, format);
  1062. queryLogMsgManager()->report_va(component, cat, job, code, format, args);
  1063. va_end(args);
  1064. }
  1065. void LogMsgComponentReporter::report_va(const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, va_list args)
  1066. {
  1067. queryLogMsgManager()->report_va(component, cat, job, code, format, args);
  1068. }
  1069. void LogMsgComponentReporter::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, const IException * exception, const char * prefix)
  1070. {
  1071. StringBuffer buff;
  1072. if(prefix) buff.append(prefix).append(" : ");
  1073. exception->errorMessage(buff);
  1074. queryLogMsgManager()->report(component, cat, job, exception->errorCode(), "%s", buff.str());
  1075. }
  1076. void LogMsgComponentReporter::report(const LogMsg & msg)
  1077. {
  1078. queryLogMsgManager()->report(msg);
  1079. }
  1080. // LogMsgPrepender
  1081. void LogMsgPrepender::report(const LogMsgCategory & cat, const char * format, ...)
  1082. {
  1083. StringBuffer buff;
  1084. buff.append(file).append("(").append(line).append(") : ").append(format);
  1085. va_list args;
  1086. va_start(args, format);
  1087. if(reporter)
  1088. reporter->report_va(cat, unknownJob, buff.str(), args);
  1089. else
  1090. queryLogMsgManager()->report_va(cat, unknownJob, buff.str(), args);
  1091. va_end(args);
  1092. }
  1093. void LogMsgPrepender::report_va(const LogMsgCategory & cat, const char * format, va_list args)
  1094. {
  1095. StringBuffer buff;
  1096. buff.append(file).append("(").append(line).append(") : ").append(format);
  1097. if(reporter)
  1098. reporter->report_va(cat, unknownJob, buff.str(), args);
  1099. else
  1100. queryLogMsgManager()->report_va(cat, unknownJob, buff.str(), args);
  1101. }
  1102. void LogMsgPrepender::report(const LogMsgCategory & cat, LogMsgCode code, const char * format, ...)
  1103. {
  1104. StringBuffer buff;
  1105. buff.append(file).append("(").append(line).append(") : ").append(format);
  1106. va_list args;
  1107. va_start(args, format);
  1108. if(reporter)
  1109. reporter->report_va(cat, unknownJob, buff.str(), args);
  1110. else
  1111. queryLogMsgManager()->report_va(cat, unknownJob, buff.str(), args);
  1112. va_end(args);
  1113. }
  1114. void LogMsgPrepender::report_va(const LogMsgCategory & cat, LogMsgCode code, const char * format, va_list args)
  1115. {
  1116. StringBuffer buff;
  1117. buff.append(file).append("(").append(line).append(") : ").append(format);
  1118. if(reporter)
  1119. reporter->report_va(cat, unknownJob, buff.str(), args);
  1120. else
  1121. queryLogMsgManager()->report_va(cat, unknownJob, buff.str(), args);
  1122. }
  1123. void LogMsgPrepender::report(const LogMsgCategory & cat, const IException * exception, const char * prefix)
  1124. {
  1125. StringBuffer buff;
  1126. buff.append(file).append("(").append(line).append(") : ");
  1127. if(prefix) buff.append(prefix).append(" : ");
  1128. exception->errorMessage(buff);
  1129. if(reporter)
  1130. reporter->report(cat, unknownJob, exception->errorCode(), "%s", buff.str());
  1131. else
  1132. queryLogMsgManager()->report(cat, unknownJob, exception->errorCode(), "%s", buff.str());
  1133. }
  1134. void LogMsgPrepender::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, ...)
  1135. {
  1136. StringBuffer buff;
  1137. buff.append(file).append("(").append(line).append(") : ").append(format);
  1138. va_list args;
  1139. va_start(args, format);
  1140. if(reporter)
  1141. reporter->report_va(cat, job, buff.str(), args);
  1142. else
  1143. queryLogMsgManager()->report_va(cat, job, buff.str(), args);
  1144. va_end(args);
  1145. }
  1146. void LogMsgPrepender::report_va(const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, va_list args)
  1147. {
  1148. StringBuffer buff;
  1149. buff.append(file).append("(").append(line).append(") : ").append(format);
  1150. if(reporter)
  1151. reporter->report_va(cat, job, buff.str(), args);
  1152. else
  1153. queryLogMsgManager()->report_va(cat, job, buff.str(), args);
  1154. }
  1155. void LogMsgPrepender::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, ...)
  1156. {
  1157. StringBuffer buff;
  1158. buff.append(file).append("(").append(line).append(") : ").append(format);
  1159. va_list args;
  1160. va_start(args, format);
  1161. if(reporter)
  1162. reporter->report_va(cat, job, buff.str(), args);
  1163. else
  1164. queryLogMsgManager()->report_va(cat, job, buff.str(), args);
  1165. va_end(args);
  1166. }
  1167. void LogMsgPrepender::report_va(const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, va_list args)
  1168. {
  1169. StringBuffer buff;
  1170. buff.append(file).append("(").append(line).append(") : ").append(format);
  1171. if(reporter)
  1172. reporter->report_va(cat, job, buff.str(), args);
  1173. else
  1174. queryLogMsgManager()->report_va(cat, job, buff.str(), args);
  1175. }
  1176. void LogMsgPrepender::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, const IException * exception, const char * prefix)
  1177. {
  1178. StringBuffer txt;
  1179. if (prefix)
  1180. txt.append(prefix).append(" : ");
  1181. exception->errorMessage(txt);
  1182. if (reporter)
  1183. reporter->report(cat, job, exception->errorCode(), "%s(%d) : %s", file, line, txt.str());
  1184. else
  1185. queryLogMsgManager()->report(cat, job, exception->errorCode(), "%s(%d) : %s", file, line, txt.str());
  1186. }
  1187. IException * LogMsgPrepender::report(IException * e, const char * prefix, LogMsgClass cls)
  1188. {
  1189. report(MCexception(e, cls), unknownJob, e, prefix);
  1190. return e;
  1191. }
  1192. // LogMsgMonitor
  1193. void LogMsgMonitor::addToPTree(IPropertyTree * tree) const
  1194. {
  1195. IPropertyTree * monitorTree = createPTree(ipt_caseInsensitive);
  1196. handler->addToPTree(monitorTree);
  1197. filter->addToPTree(monitorTree);
  1198. tree->addPropTree("monitor", monitorTree);
  1199. }
  1200. // CLogMsgManager
  1201. void CLogMsgManager::MsgProcessor::push(LogMsg * msg)
  1202. {
  1203. //assertex(more); an assertex will just recurse here
  1204. if (!more) // we are effective stopped so don't bother even dropping (and leak parameter) as drop will involve
  1205. // interaction with the base class which is stopped and could easily crash (as this condition
  1206. // is expected not to occur - typically occurs if the user has incorrectly called exit on one thread
  1207. // while still in the process of logging on another)
  1208. // cf Bug #53695 for more discussion of the issue
  1209. return;
  1210. else if(droppingLimit && (q.ordinality() >= droppingLimit))
  1211. drop();
  1212. q.enqueue(msg);
  1213. }
  1214. int CLogMsgManager::MsgProcessor::run()
  1215. {
  1216. Owned<LogMsg> msg;
  1217. while(more)
  1218. {
  1219. msg.setown(q.dequeueAndNotify(this)); // notify locks mutex on non-null return
  1220. if(!msg)
  1221. break;
  1222. owner->doReport(*msg);
  1223. pullCycleMutex.unlock();
  1224. }
  1225. while(true)
  1226. {
  1227. msg.setown(q.dequeueNowAndNotify(this)); // notify locks mutex on non-null return
  1228. if(!msg)
  1229. break;
  1230. owner->doReport(*msg);
  1231. pullCycleMutex.unlock();
  1232. }
  1233. return 0;
  1234. }
  1235. void CLogMsgManager::MsgProcessor::notify(LogMsg *)
  1236. {
  1237. pullCycleMutex.lock();
  1238. }
  1239. void CLogMsgManager::MsgProcessor::setBlockingLimit(unsigned lim)
  1240. {
  1241. q.setLimit(lim);
  1242. droppingLimit = 0;
  1243. }
  1244. void CLogMsgManager::MsgProcessor::setDroppingLimit(unsigned lim, unsigned num)
  1245. {
  1246. numToDrop = num;
  1247. droppingLimit = lim;
  1248. q.setLimit(0);
  1249. }
  1250. void CLogMsgManager::MsgProcessor::resetLimit()
  1251. {
  1252. droppingLimit = 0;
  1253. q.setLimit(0);
  1254. }
  1255. void CLogMsgManager::MsgProcessor::stop()
  1256. {
  1257. more = false;
  1258. q.stop();
  1259. }
  1260. void CLogMsgManager::MsgProcessor::drop()
  1261. {
  1262. Owned<LogMsg> msg, lastMsg;
  1263. unsigned count;
  1264. unsigned prev = 0;
  1265. for(count = 0; count < numToDrop; count++)
  1266. {
  1267. msg.setown(q.dequeueTail(0));
  1268. if(!msg) break;
  1269. DropLogMsg * dmsg = dynamic_cast<DropLogMsg *>(msg.get());
  1270. if(dmsg) prev += dmsg->queryCount()-1;
  1271. lastMsg.setown(msg.getClear());
  1272. }
  1273. if(lastMsg)
  1274. q.enqueue(new DropLogMsg(owner, lastMsg->querySysInfo().queryMsgID(), count+prev));
  1275. }
  1276. bool CLogMsgManager::MsgProcessor::flush(unsigned timeout)
  1277. {
  1278. unsigned start = msTick();
  1279. if(!q.waitMaxOrdinality(0, timeout))
  1280. return false;
  1281. unsigned now = msTick();
  1282. if(now >= (start+timeout))
  1283. return false;
  1284. try
  1285. {
  1286. synchronized block(pullCycleMutex, timeout+start-now);
  1287. }
  1288. catch(IException * e)
  1289. {
  1290. e->Release();
  1291. return false;
  1292. }
  1293. return true;
  1294. }
  1295. CLogMsgManager::~CLogMsgManager()
  1296. {
  1297. CriticalBlock crit(modeLock);
  1298. if(processor)
  1299. {
  1300. processor->stop();
  1301. processor->join();
  1302. }
  1303. }
  1304. void CLogMsgManager::enterQueueingMode()
  1305. {
  1306. CriticalBlock crit(modeLock);
  1307. if(processor) return;
  1308. processor.setown(new MsgProcessor(this));
  1309. processor->setBlockingLimit(defaultMsgQueueLimit);
  1310. processor->start();
  1311. }
  1312. void CLogMsgManager::setQueueBlockingLimit(unsigned lim)
  1313. {
  1314. CriticalBlock crit(modeLock);
  1315. if(processor)
  1316. processor->setBlockingLimit(lim);
  1317. }
  1318. void CLogMsgManager::setQueueDroppingLimit(unsigned lim, unsigned numToDrop)
  1319. {
  1320. CriticalBlock crit(modeLock);
  1321. if(processor)
  1322. processor->setDroppingLimit(lim, numToDrop);
  1323. }
  1324. void CLogMsgManager::resetQueueLimit()
  1325. {
  1326. CriticalBlock crit(modeLock);
  1327. if(processor)
  1328. processor->resetLimit();
  1329. }
  1330. void CLogMsgManager::report(const LogMsgCategory & cat, const char * format, ...)
  1331. {
  1332. if(rejectsCategory(cat)) return;
  1333. va_list args;
  1334. va_start(args, format);
  1335. pushMsg(new LogMsg(cat, getNextID(), unknownJob, NoLogMsgCode, format, args, 0, port, session));
  1336. va_end(args);
  1337. }
  1338. void CLogMsgManager::report_va(const LogMsgCategory & cat, const char * format, va_list args)
  1339. {
  1340. if(rejectsCategory(cat)) return;
  1341. pushMsg(new LogMsg(cat, getNextID(), unknownJob, NoLogMsgCode, format, args, 0, port, session));
  1342. }
  1343. void CLogMsgManager::report(const LogMsgCategory & cat, LogMsgCode code, const char * format, ...)
  1344. {
  1345. if(rejectsCategory(cat)) return;
  1346. va_list args;
  1347. va_start(args, format);
  1348. pushMsg(new LogMsg(cat, getNextID(), unknownJob, code, format, args, 0, port, session));
  1349. va_end(args);
  1350. }
  1351. void CLogMsgManager::report_va(const LogMsgCategory & cat, LogMsgCode code, const char * format, va_list args)
  1352. {
  1353. if(rejectsCategory(cat)) return;
  1354. pushMsg(new LogMsg(cat, getNextID(), unknownJob, code, format, args, 0, port, session));
  1355. }
  1356. void CLogMsgManager::report(const LogMsgCategory & cat, const IException * exception, const char * prefix)
  1357. {
  1358. if(rejectsCategory(cat)) return;
  1359. StringBuffer buff;
  1360. if(prefix) buff.append(prefix).append(" : ");
  1361. exception->errorMessage(buff);
  1362. pushMsg(new LogMsg(cat, getNextID(), unknownJob, exception->errorCode(), buff.str(), 0, port, session));
  1363. }
  1364. void CLogMsgManager::report(unsigned compo, const LogMsgCategory & cat, const char * format, ...)
  1365. {
  1366. if(rejectsCategory(cat)) return;
  1367. va_list args;
  1368. va_start(args, format);
  1369. pushMsg(new LogMsg(cat, getNextID(), unknownJob, NoLogMsgCode, format, args, compo, port, session));
  1370. va_end(args);
  1371. }
  1372. void CLogMsgManager::report_va(unsigned compo, const LogMsgCategory & cat, const char * format, va_list args)
  1373. {
  1374. if(rejectsCategory(cat)) return;
  1375. pushMsg(new LogMsg(cat, getNextID(), unknownJob, NoLogMsgCode, format, args, compo, port, session));
  1376. }
  1377. void CLogMsgManager::report(unsigned compo, const LogMsgCategory & cat, LogMsgCode code, const char * format, ...)
  1378. {
  1379. if(rejectsCategory(cat)) return;
  1380. va_list args;
  1381. va_start(args, format);
  1382. pushMsg(new LogMsg(cat, getNextID(), unknownJob, code, format, args, compo, port, session));
  1383. va_end(args);
  1384. }
  1385. void CLogMsgManager::report_va(unsigned compo, const LogMsgCategory & cat, LogMsgCode code, const char * format, va_list args)
  1386. {
  1387. if(rejectsCategory(cat)) return;
  1388. pushMsg(new LogMsg(cat, getNextID(), unknownJob, code, format, args, compo, port, session));
  1389. }
  1390. void CLogMsgManager::report(unsigned compo, const LogMsgCategory & cat, const IException * exception, const char * prefix)
  1391. {
  1392. if(rejectsCategory(cat)) return;
  1393. StringBuffer buff;
  1394. if(prefix) buff.append(prefix).append(" : ");
  1395. exception->errorMessage(buff);
  1396. pushMsg(new LogMsg(cat, getNextID(), unknownJob, exception->errorCode(), buff.str(), compo, port, session));
  1397. }
  1398. void CLogMsgManager::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, ...)
  1399. {
  1400. if(rejectsCategory(cat)) return;
  1401. va_list args;
  1402. va_start(args, format);
  1403. pushMsg(new LogMsg(cat, getNextID(), job, NoLogMsgCode, format, args, 0, port, session));
  1404. va_end(args);
  1405. }
  1406. void CLogMsgManager::report_va(const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, va_list args)
  1407. {
  1408. if(rejectsCategory(cat)) return;
  1409. pushMsg(new LogMsg(cat, getNextID(), job, NoLogMsgCode, format, args, 0, port, session));
  1410. }
  1411. void CLogMsgManager::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, ...)
  1412. {
  1413. if(rejectsCategory(cat)) return;
  1414. va_list args;
  1415. va_start(args, format);
  1416. pushMsg(new LogMsg(cat, getNextID(), job, code, format, args, 0, port, session));
  1417. va_end(args);
  1418. }
  1419. void CLogMsgManager::report_va(const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, va_list args)
  1420. {
  1421. if(rejectsCategory(cat)) return;
  1422. pushMsg(new LogMsg(cat, getNextID(), job, code, format, args, 0, port, session));
  1423. }
  1424. void CLogMsgManager::report(const LogMsgCategory & cat, const LogMsgJobInfo & job, const IException * exception, const char * prefix)
  1425. {
  1426. if(rejectsCategory(cat)) return;
  1427. StringBuffer buff;
  1428. if(prefix) buff.append(prefix).append(" : ");
  1429. exception->errorMessage(buff);
  1430. pushMsg(new LogMsg(cat, getNextID(), job, exception->errorCode(), buff.str(), 0, port, session));
  1431. }
  1432. void CLogMsgManager::report(unsigned compo, const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, ...)
  1433. {
  1434. if(rejectsCategory(cat)) return;
  1435. va_list args;
  1436. va_start(args, format);
  1437. pushMsg(new LogMsg(cat, getNextID(), job, NoLogMsgCode, format, args, compo, port, session));
  1438. va_end(args);
  1439. }
  1440. void CLogMsgManager::report_va(unsigned compo, const LogMsgCategory & cat, const LogMsgJobInfo & job, const char * format, va_list args)
  1441. {
  1442. if(rejectsCategory(cat)) return;
  1443. pushMsg(new LogMsg(cat, getNextID(), job, NoLogMsgCode, format, args, compo, port, session));
  1444. }
  1445. void CLogMsgManager::report(unsigned compo, const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, ...)
  1446. {
  1447. if(rejectsCategory(cat)) return;
  1448. va_list args;
  1449. va_start(args, format);
  1450. pushMsg(new LogMsg(cat, getNextID(), job, code, format, args, compo, port, session));
  1451. va_end(args);
  1452. }
  1453. void CLogMsgManager::report_va(unsigned compo, const LogMsgCategory & cat, const LogMsgJobInfo & job, LogMsgCode code, const char * format, va_list args)
  1454. {
  1455. if(rejectsCategory(cat)) return;
  1456. pushMsg(new LogMsg(cat, getNextID(), job, code, format, args, compo, port, session));
  1457. }
  1458. void CLogMsgManager::report(unsigned compo, const LogMsgCategory & cat, const LogMsgJobInfo & job, const IException * exception, const char * prefix)
  1459. {
  1460. if(rejectsCategory(cat)) return;
  1461. StringBuffer buff;
  1462. if(prefix) buff.append(prefix).append(" : ");
  1463. exception->errorMessage(buff);
  1464. pushMsg(new LogMsg(cat, getNextID(), job, exception->errorCode(), buff.str(), compo, port, session));
  1465. }
  1466. void CLogMsgManager::pushMsg(LogMsg * _msg)
  1467. {
  1468. Owned<LogMsg> msg(_msg);
  1469. if(processor)
  1470. processor->push(msg.getLink());
  1471. else
  1472. doReport(*msg);
  1473. }
  1474. void CLogMsgManager::doReport(const LogMsg & msg) const
  1475. {
  1476. try
  1477. {
  1478. ReadLockBlock block(monitorLock);
  1479. ForEachItemIn(i, monitors)
  1480. monitors.item(i).processMessage(msg);
  1481. }
  1482. catch(IException * e)
  1483. {
  1484. StringBuffer err("exception reporting log message: ");
  1485. err.append(e->errorCode());
  1486. e->errorMessage(err);
  1487. panic(err.str());
  1488. e->Release();
  1489. }
  1490. catch(...)
  1491. {
  1492. panic("unknown exception reporting log message");
  1493. }
  1494. }
  1495. void CLogMsgManager::panic(char const * reason) const
  1496. {
  1497. fprintf(stderr, "%s", reason); // not sure there's anything more useful we can do here
  1498. }
  1499. offset_t CLogMsgManager::getLogPosition(StringBuffer &logFileName, const ILogMsgHandler * handler) const
  1500. {
  1501. if (processor)
  1502. processor->flush(10*1000);
  1503. WriteLockBlock block(monitorLock); // Prevents any incoming messages as we are doing this
  1504. return handler->getLogPosition(logFileName);
  1505. }
  1506. aindex_t CLogMsgManager::find(const ILogMsgHandler * handler) const
  1507. {
  1508. // N.B. Should be used inside critical block
  1509. ForEachItemIn(i, monitors)
  1510. if(monitors.item(i).queryHandler()==handler) return i;
  1511. return NotFound;
  1512. }
  1513. bool CLogMsgManager::addMonitor(ILogMsgHandler * handler, ILogMsgFilter * filter)
  1514. {
  1515. flushQueue(10*1000);
  1516. WriteLockBlock block(monitorLock);
  1517. if(find(handler) != NotFound) return false;
  1518. monitors.append(*(new LogMsgMonitor(filter, handler)));
  1519. prefilter.orWithFilter(filter);
  1520. sendFilterToChildren(true);
  1521. return true;
  1522. }
  1523. bool CLogMsgManager::addMonitorOwn(ILogMsgHandler * handler, ILogMsgFilter * filter)
  1524. {
  1525. bool ret = addMonitor(handler, filter);
  1526. filter->Release();
  1527. handler->Release();
  1528. return ret;
  1529. }
  1530. void CLogMsgManager::buildPrefilter()
  1531. {
  1532. // N.B. Should be used inside critical block
  1533. prefilter.reset();
  1534. ForEachItemIn(i, monitors)
  1535. prefilter.orWithFilter(monitors.item(i).queryFilter());
  1536. }
  1537. bool CLogMsgManager::removeMonitor(ILogMsgHandler * handler)
  1538. {
  1539. Linked<LogMsgMonitor> todelete;
  1540. {
  1541. WriteLockBlock block(monitorLock);
  1542. aindex_t pos = find(handler);
  1543. if(pos == NotFound) return false;
  1544. todelete.set(&monitors.item(pos));
  1545. monitors.remove(pos);
  1546. buildPrefilter();
  1547. sendFilterToChildren(true);
  1548. return true;
  1549. }
  1550. }
  1551. unsigned CLogMsgManager::removeMonitorsMatching(HandlerTest & test)
  1552. {
  1553. CIArrayOf<LogMsgMonitor> todelete; // delete outside monitorLock
  1554. unsigned count = 0;
  1555. {
  1556. WriteLockBlock block(monitorLock);
  1557. ForEachItemInRev(i, monitors)
  1558. if(test(monitors.item(i).queryHandler()))
  1559. {
  1560. LogMsgMonitor &it = monitors.item(i);
  1561. it.Link();
  1562. todelete.append(it);
  1563. monitors.remove(i);
  1564. ++count;
  1565. }
  1566. buildPrefilter();
  1567. sendFilterToChildren(true);
  1568. }
  1569. return count;
  1570. }
  1571. void CLogMsgManager::removeAllMonitors()
  1572. {
  1573. CIArrayOf<LogMsgMonitor> todelete; // delete outside monitorLock
  1574. {
  1575. WriteLockBlock block(monitorLock);
  1576. ForEachItemInRev(i, monitors) {
  1577. LogMsgMonitor &it = monitors.item(i);
  1578. it.Link();
  1579. todelete.append(it);
  1580. monitors.remove(i);
  1581. }
  1582. prefilter.reset();
  1583. sendFilterToChildren(true);
  1584. }
  1585. }
  1586. void CLogMsgManager::resetMonitors()
  1587. {
  1588. suspendChildren();
  1589. removeAllMonitors();
  1590. Owned<ILogMsgFilter> defaultFilter = getDefaultLogMsgFilter();
  1591. addMonitor(theStderrHandler, defaultFilter);
  1592. unsuspendChildren();
  1593. }
  1594. ILogMsgFilter * CLogMsgManager::queryMonitorFilter(const ILogMsgHandler * handler) const
  1595. {
  1596. ReadLockBlock block(monitorLock);
  1597. aindex_t pos = find(handler);
  1598. if(pos == NotFound) return 0;
  1599. return monitors.item(pos).queryFilter();
  1600. }
  1601. bool CLogMsgManager::changeMonitorFilter(const ILogMsgHandler * handler, ILogMsgFilter * newFilter)
  1602. {
  1603. WriteLockBlock block(monitorLock);
  1604. aindex_t pos = find(handler);
  1605. if(pos == NotFound) return 0;
  1606. monitors.item(pos).setFilter(newFilter);
  1607. buildPrefilter();
  1608. sendFilterToChildren(true);
  1609. return true;
  1610. }
  1611. void CLogMsgManager::prepAllHandlers() const
  1612. {
  1613. ReadLockBlock block(monitorLock);
  1614. ForEachItemIn(i, monitors)
  1615. if(monitors.item(i).queryHandler()->needsPrep()) monitors.item(i).queryHandler()->prep();
  1616. }
  1617. aindex_t CLogMsgManager::findChild(ILogMsgLinkToChild * child) const
  1618. {
  1619. ForEachItemIn(i, children)
  1620. if(&(children.item(i)) == child ) return i;
  1621. return NotFound;
  1622. }
  1623. ILogMsgFilter * CLogMsgManager::getCompoundFilter(bool locked) const
  1624. {
  1625. if(!locked) monitorLock.lockRead();
  1626. Owned<CategoryLogMsgFilter> categoryFilter = new CategoryLogMsgFilter(0, 0, 0, false);
  1627. Owned<ILogMsgFilter> otherFilters;
  1628. ILogMsgFilter * ifilter;
  1629. bool hadCat = false;
  1630. ForEachItemIn(i, monitors)
  1631. {
  1632. ifilter = monitors.item(i).queryFilter();
  1633. if(ifilter->queryLocalFlag()) continue;
  1634. if(ifilter->isCategoryFilter())
  1635. {
  1636. categoryFilter->orWithFilter(ifilter);
  1637. hadCat = true;
  1638. }
  1639. else
  1640. {
  1641. if(otherFilters)
  1642. otherFilters.setown(getOrLogMsgFilter(otherFilters, ifilter));
  1643. else
  1644. otherFilters.set(ifilter);
  1645. }
  1646. }
  1647. if(hadCat)
  1648. {
  1649. if(otherFilters)
  1650. otherFilters.setown(getOrLogMsgFilter(otherFilters, categoryFilter));
  1651. else
  1652. otherFilters.set(categoryFilter);
  1653. }
  1654. if(!locked) monitorLock.unlock();
  1655. if(!otherFilters)
  1656. return getPassNoneLogMsgFilter();
  1657. return otherFilters.getLink();
  1658. }
  1659. void CLogMsgManager::sendFilterToChildren(bool locked) const
  1660. {
  1661. if(suspendedChildren) return;
  1662. ReadLockBlock block(childLock);
  1663. if(children.length()==0) return;
  1664. ILogMsgFilter * filter = getCompoundFilter(locked);
  1665. ForEachItemIn(i, children)
  1666. children.item(i).sendFilter(filter);
  1667. filter->Release();
  1668. }
  1669. bool CLogMsgManager::addMonitorToPTree(const ILogMsgHandler * handler, IPropertyTree * tree) const
  1670. {
  1671. ReadLockBlock block(monitorLock);
  1672. aindex_t pos = find(handler);
  1673. if(pos == NotFound) return false;
  1674. monitors.item(pos).addToPTree(tree);
  1675. return true;
  1676. }
  1677. void CLogMsgManager::addAllMonitorsToPTree(IPropertyTree * tree) const
  1678. {
  1679. ReadLockBlock block(monitorLock);
  1680. ForEachItemIn(i, monitors)
  1681. monitors.item(i).addToPTree(tree);
  1682. }
  1683. bool CLogMsgManager::rejectsCategory(const LogMsgCategory & cat) const
  1684. {
  1685. if (!prefilter.includeCategory(cat))
  1686. return true;
  1687. ReadLockBlock block(monitorLock);
  1688. ForEachItemIn(i, monitors)
  1689. {
  1690. if (monitors.item(i).queryFilter()->mayIncludeCategory(cat))
  1691. return false;
  1692. }
  1693. return true;
  1694. }
  1695. // Helper functions
  1696. ILogMsgFilter * getDeserializedLogMsgFilter(MemoryBuffer & in)
  1697. {
  1698. unsigned type;
  1699. in.read(type);
  1700. switch(type)
  1701. {
  1702. case MSGFILTER_passall : return LINK(thePassAllFilter);
  1703. case MSGFILTER_passlocal : return LINK(thePassLocalFilter);
  1704. case MSGFILTER_passnone : return LINK(thePassNoneFilter);
  1705. case MSGFILTER_category : return new CategoryLogMsgFilter(in);
  1706. case MSGFILTER_pid : return new PIDLogMsgFilter(in);
  1707. case MSGFILTER_tid : return new TIDLogMsgFilter(in);
  1708. case MSGFILTER_node : return new NodeLogMsgFilter(in);
  1709. case MSGFILTER_ip : return new IpLogMsgFilter(in);
  1710. case MSGFILTER_job : return new JobLogMsgFilter(in);
  1711. case MSGFILTER_user : return new UserLogMsgFilter(in);
  1712. case MSGFILTER_session : return new SessionLogMsgFilter(in);
  1713. case MSGFILTER_component : return new ComponentLogMsgFilter(in);
  1714. case MSGFILTER_regex : return new RegexLogMsgFilter(in);
  1715. case MSGFILTER_not : return new NotLogMsgFilter(in);
  1716. case MSGFILTER_and : return new AndLogMsgFilter(in);
  1717. case MSGFILTER_or : return new OrLogMsgFilter(in);
  1718. case MSGFILTER_switch : return new SwitchLogMsgFilter(in);
  1719. default: assertex(!"getDeserializedLogMsgFilter: unrecognized LogMsgFilterType");
  1720. }
  1721. return 0;
  1722. }
  1723. ILogMsgFilter * getLogMsgFilterFromPTree(IPropertyTree * xml)
  1724. {
  1725. /* Note that several of these constructors use GetPropInt and GetPropInt64 to get unsigneds. I think this is OK? (all int64 internally)*/
  1726. StringBuffer type;
  1727. xml->getProp("@type", type);
  1728. if(strcmp(type.str(), "all")==0) return LINK(thePassAllFilter);
  1729. else if(strcmp(type.str(), "local")==0) return LINK(thePassLocalFilter);
  1730. else if(strcmp(type.str(), "none")==0) return LINK(thePassNoneFilter);
  1731. else if(strcmp(type.str(), "category")==0) return new CategoryLogMsgFilter(xml);
  1732. else if(strcmp(type.str(), "pid")==0) return new PIDLogMsgFilter(xml);
  1733. else if(strcmp(type.str(), "tid")==0) return new TIDLogMsgFilter(xml);
  1734. else if(strcmp(type.str(), "node")==0) return new NodeLogMsgFilter(xml);
  1735. else if(strcmp(type.str(), "ip")==0) return new IpLogMsgFilter(xml);
  1736. else if(strcmp(type.str(), "job")==0) return new JobLogMsgFilter(xml);
  1737. else if(strcmp(type.str(), "user")==0) return new UserLogMsgFilter(xml);
  1738. else if(strcmp(type.str(), "session")==0) return new SessionLogMsgFilter(xml);
  1739. else if(strcmp(type.str(), "component")==0) return new ComponentLogMsgFilter(xml);
  1740. else if(strcmp(type.str(), "regex")==0) return new RegexLogMsgFilter(xml);
  1741. else if(strcmp(type.str(), "not")==0) return new NotLogMsgFilter(xml);
  1742. else if(strcmp(type.str(), "and")==0) return new AndLogMsgFilter(xml);
  1743. else if(strcmp(type.str(), "or")==0) return new OrLogMsgFilter(xml);
  1744. else if(strcmp(type.str(), "filter")==0) return new SwitchLogMsgFilter(xml);
  1745. else assertex(!"getLogMsgFilterFromPTree : unrecognized LogMsgFilter type");
  1746. return getPassAllLogMsgFilter();
  1747. }
  1748. ILogMsgFilter * getDefaultLogMsgFilter()
  1749. {
  1750. return new CategoryLogMsgFilter(MSGAUD_all, MSGCLS_all, DefaultDetail, true);
  1751. }
  1752. ILogMsgFilter * getPassAllLogMsgFilter()
  1753. {
  1754. return LINK(thePassAllFilter);
  1755. }
  1756. ILogMsgFilter * getLocalLogMsgFilter()
  1757. {
  1758. return LINK(thePassLocalFilter);
  1759. }
  1760. ILogMsgFilter * getPassNoneLogMsgFilter()
  1761. {
  1762. return LINK(thePassNoneFilter);
  1763. }
  1764. ILogMsgFilter * queryPassAllLogMsgFilter()
  1765. {
  1766. return thePassAllFilter;
  1767. }
  1768. ILogMsgFilter * queryLocalLogMsgFilter()
  1769. {
  1770. return thePassLocalFilter;
  1771. }
  1772. ILogMsgFilter * queryPassNoneLogMsgFilter()
  1773. {
  1774. return thePassNoneFilter;
  1775. }
  1776. ILogMsgFilter * getCategoryLogMsgFilter(unsigned audiences, unsigned classes, LogMsgDetail maxDetail, bool local)
  1777. {
  1778. if((audiences==MSGAUD_all) && (classes==MSGCLS_all) && (maxDetail==TopDetail))
  1779. {
  1780. if(local)
  1781. return LINK(thePassLocalFilter);
  1782. else
  1783. return LINK(thePassAllFilter);
  1784. }
  1785. return new CategoryLogMsgFilter(audiences, classes, maxDetail, local);
  1786. }
  1787. ILogMsgFilter * getPIDLogMsgFilter(unsigned pid, bool local)
  1788. {
  1789. return new PIDLogMsgFilter(pid, local);
  1790. }
  1791. ILogMsgFilter * getTIDLogMsgFilter(unsigned tid, bool local)
  1792. {
  1793. return new TIDLogMsgFilter(tid, local);
  1794. }
  1795. ILogMsgFilter * getNodeLogMsgFilter(const char * name, unsigned port, bool local)
  1796. {
  1797. return new NodeLogMsgFilter(name, port, local);
  1798. }
  1799. ILogMsgFilter * getNodeLogMsgFilter(const IpAddress & ip, unsigned port, bool local)
  1800. {
  1801. return new NodeLogMsgFilter(ip, port, local);
  1802. }
  1803. ILogMsgFilter * getNodeLogMsgFilter(unsigned port, bool local)
  1804. {
  1805. return new NodeLogMsgFilter(port, local);
  1806. }
  1807. ILogMsgFilter * getIpLogMsgFilter(const char * name, bool local)
  1808. {
  1809. return new IpLogMsgFilter(name, local);
  1810. }
  1811. ILogMsgFilter * getIpLogMsgFilter(const IpAddress & ip, bool local)
  1812. {
  1813. return new IpLogMsgFilter(ip, local);
  1814. }
  1815. ILogMsgFilter * getIpLogMsgFilter(bool local)
  1816. {
  1817. return new IpLogMsgFilter(local);
  1818. }
  1819. ILogMsgFilter * getJobLogMsgFilter(LogMsgJobId job, bool local)
  1820. {
  1821. return new JobLogMsgFilter(job, local);
  1822. }
  1823. ILogMsgFilter * getUserLogMsgFilter(LogMsgUserId user, bool local)
  1824. {
  1825. return new UserLogMsgFilter(user, local);
  1826. }
  1827. ILogMsgFilter * getSessionLogMsgFilter(LogMsgSessionId session, bool local)
  1828. {
  1829. return new SessionLogMsgFilter(session, local);
  1830. }
  1831. ILogMsgFilter * getComponentLogMsgFilter(unsigned component, bool local)
  1832. {
  1833. return new ComponentLogMsgFilter(component, local);
  1834. }
  1835. ILogMsgFilter * getRegexLogMsgFilter(const char *regex, bool local)
  1836. {
  1837. return new RegexLogMsgFilter(regex, local);
  1838. }
  1839. ILogMsgFilter * getNotLogMsgFilter(ILogMsgFilter * arg)
  1840. {
  1841. return new NotLogMsgFilter(arg);
  1842. }
  1843. ILogMsgFilter * getNotLogMsgFilterOwn(ILogMsgFilter * arg)
  1844. {
  1845. ILogMsgFilter * ret = new NotLogMsgFilter(arg);
  1846. arg->Release();
  1847. return ret;
  1848. }
  1849. ILogMsgFilter * getAndLogMsgFilter(ILogMsgFilter * arg1, ILogMsgFilter * arg2)
  1850. {
  1851. return new AndLogMsgFilter(arg1, arg2);
  1852. }
  1853. ILogMsgFilter * getAndLogMsgFilterOwn(ILogMsgFilter * arg1, ILogMsgFilter * arg2)
  1854. {
  1855. ILogMsgFilter * ret = new AndLogMsgFilter(arg1, arg2);
  1856. arg1->Release();
  1857. arg2->Release();
  1858. return ret;
  1859. }
  1860. ILogMsgFilter * getOrLogMsgFilter(ILogMsgFilter * arg1, ILogMsgFilter * arg2)
  1861. {
  1862. return new OrLogMsgFilter(arg1, arg2);
  1863. }
  1864. ILogMsgFilter * getOrLogMsgFilterOwn(ILogMsgFilter * arg1, ILogMsgFilter * arg2)
  1865. {
  1866. ILogMsgFilter * ret = new OrLogMsgFilter(arg1, arg2);
  1867. arg1->Release();
  1868. arg2->Release();
  1869. return ret;
  1870. }
  1871. ILogMsgFilter * getSwitchLogMsgFilterOwn(ILogMsgFilter * switchFilter, ILogMsgFilter * yesFilter, ILogMsgFilter * noFilter)
  1872. {
  1873. ILogMsgFilter * ret = new SwitchLogMsgFilter(switchFilter, yesFilter, noFilter);
  1874. switchFilter->Release();
  1875. yesFilter->Release();
  1876. noFilter->Release();
  1877. return ret;
  1878. }
  1879. ILogMsgHandler * getHandleLogMsgHandler(FILE * handle, unsigned fields, bool writeXML)
  1880. {
  1881. if(writeXML)
  1882. return new HandleLogMsgHandlerXML(handle, fields);
  1883. return new HandleLogMsgHandlerTable(handle, fields);
  1884. }
  1885. ILogMsgHandler * getFileLogMsgHandler(const char * filename, const char * headertext, unsigned fields, bool writeXML, bool append, bool flushes)
  1886. {
  1887. if(writeXML)
  1888. return new FileLogMsgHandlerXML(filename, headertext, fields, append, flushes);
  1889. return new FileLogMsgHandlerTable(filename, headertext, fields, append, flushes);
  1890. }
  1891. ILogMsgHandler * getRollingFileLogMsgHandler(const char * filebase, const char * fileextn, unsigned fields, bool append, bool flushes, const char *initialName, const char *alias, bool daily)
  1892. {
  1893. return new RollingFileLogMsgHandler(filebase, fileextn, fields, append, flushes, initialName, alias, daily);
  1894. }
  1895. ILogMsgHandler * getBinLogMsgHandler(const char * filename, bool append)
  1896. {
  1897. return new BinLogMsgHandler(filename, append);
  1898. }
  1899. void installLogMsgFilterSwitch(ILogMsgHandler * handler, ILogMsgFilter * switchFilter, ILogMsgFilter * newFilter)
  1900. {
  1901. queryLogMsgManager()->changeMonitorFilterOwn(handler, getSwitchLogMsgFilterOwn(switchFilter, newFilter, queryLogMsgManager()->getMonitorFilter(handler)));
  1902. }
  1903. ILogMsgHandler * getLogMsgHandlerFromPTree(IPropertyTree * tree)
  1904. {
  1905. StringBuffer type;
  1906. tree->getProp("@type", type);
  1907. unsigned fields = MSGFIELD_all;
  1908. char const * fstr = tree->queryProp("@fields");
  1909. if(fstr)
  1910. {
  1911. if(isdigit(fstr[0]))
  1912. fields = atoi(fstr);
  1913. else
  1914. fields = LogMsgFieldsFromAbbrevs(fstr);
  1915. }
  1916. if(strcmp(type.str(), "stderr")==0)
  1917. return getHandleLogMsgHandler(stderr, fields, tree->hasProp("@writeXML"));
  1918. else if(strcmp(type.str(), "file")==0)
  1919. {
  1920. StringBuffer filename;
  1921. tree->getProp("@filename", filename);
  1922. if(tree->hasProp("@headertext"))
  1923. {
  1924. StringBuffer headertext;
  1925. tree->getProp("@headertext", headertext);
  1926. return getFileLogMsgHandler(filename.str(), headertext.str(), fields, !(tree->hasProp("@writeTable")), tree->hasProp("@append"), tree->hasProp("@flushes"));
  1927. }
  1928. else
  1929. return getFileLogMsgHandler(filename.str(), 0, fields, !(tree->hasProp("@writeTable")), tree->hasProp("@append"), tree->hasProp("@flushes"));
  1930. }
  1931. else if(strcmp(type.str(), "binary")==0)
  1932. {
  1933. StringBuffer filename;
  1934. tree->getProp("@filename", filename);
  1935. return getBinLogMsgHandler(filename.str(), tree->hasProp("@append"));
  1936. }
  1937. else assertex(!"getLogMsgFilterFromPTree : unrecognized LogMsgHandler type");
  1938. return LINK(theStderrHandler);
  1939. }
  1940. ILogMsgHandler * attachStandardFileLogMsgMonitor(const char * filename, const char * headertext, unsigned fields, unsigned audiences, unsigned classes, LogMsgDetail detail, bool writeXML, bool append, bool flushes, bool local)
  1941. {
  1942. #ifdef FILE_LOG_ENABLES_QUEUEUING
  1943. queryLogMsgManager()->enterQueueingMode();
  1944. #endif
  1945. ILogMsgFilter * filter = getCategoryLogMsgFilter(audiences, classes, detail, local);
  1946. ILogMsgHandler * handler = getFileLogMsgHandler(filename, headertext, fields, writeXML, append, flushes);
  1947. queryLogMsgManager()->addMonitorOwn(handler, filter);
  1948. return handler;
  1949. }
  1950. ILogMsgHandler * attachStandardBinLogMsgMonitor(const char * filename, unsigned audiences, unsigned classes, LogMsgDetail detail, bool append, bool local)
  1951. {
  1952. #ifdef FILE_LOG_ENABLES_QUEUEUING
  1953. queryLogMsgManager()->enterQueueingMode();
  1954. #endif
  1955. ILogMsgFilter * filter = getCategoryLogMsgFilter(audiences, classes, detail, local);
  1956. ILogMsgHandler * handler = getBinLogMsgHandler(filename, append);
  1957. queryLogMsgManager()->addMonitorOwn(handler, filter);
  1958. return handler;
  1959. }
  1960. ILogMsgHandler * attachStandardHandleLogMsgMonitor(FILE * handle, unsigned fields, unsigned audiences, unsigned classes, LogMsgDetail detail, bool writeXML, bool local)
  1961. {
  1962. ILogMsgFilter * filter = getCategoryLogMsgFilter(audiences, classes, detail, local);
  1963. ILogMsgHandler * handler = getHandleLogMsgHandler(handle, fields, writeXML);
  1964. queryLogMsgManager()->addMonitorOwn(handler, filter);
  1965. return handler;
  1966. }
  1967. ILogMsgHandler * attachLogMsgMonitorFromPTree(IPropertyTree * tree)
  1968. {
  1969. Owned<IPropertyTree> handlertree = tree->getPropTree("handler");
  1970. Owned<IPropertyTree> filtertree = tree->getPropTree("filter");
  1971. ILogMsgHandler * handler = getLogMsgHandlerFromPTree(handlertree);
  1972. ILogMsgFilter * filter = getLogMsgFilterFromPTree(filtertree);
  1973. queryLogMsgManager()->addMonitorOwn(handler, filter);
  1974. return handler;
  1975. }
  1976. void attachManyLogMsgMonitorsFromPTree(IPropertyTree * tree)
  1977. {
  1978. Owned<IPropertyTreeIterator> iter = tree->getElements("monitor");
  1979. ForEach(*iter)
  1980. attachLogMsgMonitorFromPTree(&(iter->query()));
  1981. }
  1982. // Standard categories and unknown jobInfo
  1983. const LogMsgCategory MCdisaster(MSGAUD_all, MSGCLS_disaster);
  1984. const LogMsgCategory MCuserError(MSGAUD_user, MSGCLS_error);
  1985. const LogMsgCategory MCoperatorError(MSGAUD_operator, MSGCLS_error);
  1986. const LogMsgCategory MCinternalError((LogMsgAudience)(MSGAUD_internal & MSGAUD_programmer), MSGCLS_error, 1);
  1987. const LogMsgCategory MCuserWarning(MSGAUD_user, MSGCLS_warning);
  1988. const LogMsgCategory MCoperatorWarning(MSGAUD_operator, MSGCLS_warning);
  1989. const LogMsgCategory MCinternalWarning((LogMsgAudience)(MSGAUD_internal & MSGAUD_programmer), MSGCLS_warning, 1);
  1990. const LogMsgCategory MCuserProgress(MSGAUD_user, MSGCLS_progress);
  1991. const LogMsgCategory MCoperatorProgress(MSGAUD_operator, MSGCLS_progress);
  1992. const LogMsgCategory MCdebugProgress(MSGAUD_programmer, MSGCLS_progress);
  1993. const LogMsgCategory MCdebugInfo(MSGAUD_programmer, MSGCLS_information);
  1994. const LogMsgCategory MCstats(MSGAUD_performance, MSGCLS_information);
  1995. const LogMsgCategory MCevent(MSGAUD_monitor, MSGCLS_event);
  1996. const LogMsgCategory MClegacy(MSGAUD_legacy, MSGCLS_legacy, DefaultDetail);
  1997. const LogMsgJobInfo unknownJob(UnknownJob, UnknownUser);
  1998. // Calls to make, remove, and return the manager, standard handler, pass all/none filters, reporter array
  1999. PassAllLogMsgFilter * thePassAllFilter;
  2000. PassLocalLogMsgFilter * thePassLocalFilter;
  2001. PassNoneLogMsgFilter * thePassNoneFilter;
  2002. HandleLogMsgHandlerTable * theStderrHandler;
  2003. CLogMsgManager * theManager;
  2004. CSysLogEventLogger * theSysLogEventLogger;
  2005. LogMsgComponentReporter * theReporters[MSGCOMP_NUMBER];
  2006. MODULE_INIT(INIT_PRIORITY_JLOG)
  2007. {
  2008. thePassAllFilter = new PassAllLogMsgFilter();
  2009. thePassLocalFilter = new PassLocalLogMsgFilter();
  2010. thePassNoneFilter = new PassNoneLogMsgFilter();
  2011. theStderrHandler = new HandleLogMsgHandlerTable(stderr, MSGFIELD_STANDARD);
  2012. theSysLogEventLogger = new CSysLogEventLogger;
  2013. theManager = new CLogMsgManager();
  2014. theManager->resetMonitors();
  2015. for(unsigned compo = 0; compo<MSGCOMP_NUMBER; compo++)
  2016. theReporters[compo] = new LogMsgComponentReporter(compo);
  2017. return true;
  2018. }
  2019. MODULE_EXIT()
  2020. {
  2021. for(unsigned compo = 0; compo<MSGCOMP_NUMBER; compo++)
  2022. {
  2023. delete theReporters[compo];
  2024. theReporters[compo] = NULL;
  2025. }
  2026. delete theManager;
  2027. delete theSysLogEventLogger;
  2028. delete theStderrHandler;
  2029. delete thePassNoneFilter;
  2030. delete thePassLocalFilter;
  2031. delete thePassAllFilter;
  2032. theManager = NULL;
  2033. theSysLogEventLogger = NULL;
  2034. theStderrHandler = NULL;
  2035. thePassNoneFilter = NULL;
  2036. thePassLocalFilter = NULL;
  2037. thePassAllFilter = NULL;
  2038. }
  2039. ILogMsgManager * queryLogMsgManager()
  2040. {
  2041. return theManager;
  2042. }
  2043. ILogMsgHandler * queryStderrLogMsgHandler()
  2044. {
  2045. return theStderrHandler;
  2046. }
  2047. LogMsgComponentReporter * queryLogMsgComponentReporter(unsigned compo)
  2048. {
  2049. return theReporters[compo];
  2050. }
  2051. ILogMsgManager * createLogMsgManager() // use with care! (needed by mplog listener facility)
  2052. {
  2053. return new CLogMsgManager();
  2054. }
  2055. // Event Logging
  2056. ISysLogEventLogger * querySysLogEventLogger()
  2057. {
  2058. return theSysLogEventLogger;
  2059. }
  2060. ILogMsgHandler * getSysLogMsgHandler(unsigned fields)
  2061. {
  2062. return new SysLogMsgHandler(theSysLogEventLogger, fields);
  2063. }
  2064. #ifdef _WIN32
  2065. #include <WINNT.H>
  2066. #include "jelog.h"
  2067. struct AuditTypeWin32Data
  2068. {
  2069. public:
  2070. unsigned eventtype;
  2071. unsigned categoryid;
  2072. unsigned eventid;
  2073. };
  2074. #define CATEGORY_AUDIT_FUNCTION_REQUIRED
  2075. #define AUDIT_TYPES_BEGIN AuditTypeWin32Data auditTypeDataMap[NUM_AUDIT_TYPES+1] = {
  2076. #define MAKE_AUDIT_TYPE(name, type, categoryid, eventid, level) {type, categoryid, eventid},
  2077. #define AUDIT_TYPES_END {0, 0, 0} };
  2078. #include "jelogtype.hpp"
  2079. #undef CATEGORY_AUDIT_FUNCTION_REQUIRED
  2080. #undef AUDIT_TYPES_BEGIN
  2081. #undef MAKE_AUDIT_TYPE
  2082. #undef AUDIT_TYPES_END
  2083. CSysLogEventLogger::CSysLogEventLogger() : hEventLog(0)
  2084. {
  2085. }
  2086. bool CSysLogEventLogger::log(AuditType auditType, char const * msg, size32_t datasize, void const * data)
  2087. {
  2088. assertex(auditType < NUM_AUDIT_TYPES);
  2089. AuditTypeWin32Data const & typeData = auditTypeDataMap[auditType];
  2090. return win32Report(typeData.eventtype, typeData.categoryid, typeData.eventid, msg, datasize, data);
  2091. }
  2092. bool CSysLogEventLogger::win32Report(unsigned eventtype, unsigned category, unsigned eventid, const char * msg, size32_t datasize, const void * data)
  2093. {
  2094. if (hEventLog==0) {
  2095. // MORE - this doesn't work on Vista/Win7 as can't copy to system32...
  2096. // Perhaps we should just kill this code
  2097. char path[_MAX_PATH+1];
  2098. GetEnvironmentVariable("SystemRoot",path,sizeof(path));
  2099. strcat(path,"\\System32\\JELOG.dll");
  2100. Owned<IFile> file = createIFile(path);
  2101. try {
  2102. if (!file->exists()) {
  2103. char src[_MAX_PATH+1];
  2104. LPTSTR tail;
  2105. DWORD res = SearchPath(NULL,"JELOG.DLL",NULL,sizeof(src),src,&tail);
  2106. if (res>0)
  2107. copyFile(path,src);
  2108. else
  2109. throw makeOsException(GetLastError());
  2110. }
  2111. }
  2112. catch (IException *e)
  2113. {
  2114. EXCLOG(e, "reportEventLog: Could not install JELOG.DLL");
  2115. hEventLog=(HANDLE)-1;
  2116. e->Release();
  2117. return false;
  2118. }
  2119. HKEY hk;
  2120. if (RegCreateKeyEx(HKEY_LOCAL_MACHINE,"SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\Seisint",
  2121. NULL, NULL, 0, KEY_ALL_ACCESS, NULL, &hk, NULL)==0) {
  2122. DWORD sizedata = 0;
  2123. DWORD type = REG_EXPAND_SZ;
  2124. if ((RegQueryValueEx(hk,"EventMessageFile",NULL, &type, NULL, &sizedata)!=0)||!sizedata) {
  2125. StringAttr str("%SystemRoot%\\System32\\JELOG.dll");
  2126. RegSetValueEx(hk,"EventMessageFile", 0, REG_EXPAND_SZ, (LPBYTE) str.get(), (DWORD)str.length() + 1);
  2127. RegSetValueEx(hk,"CategoryMessageFile", 0, REG_EXPAND_SZ, (LPBYTE) str.get(), (DWORD)str.length() + 1);
  2128. DWORD dwData = EVENTLOG_ERROR_TYPE | EVENTLOG_WARNING_TYPE | EVENTLOG_INFORMATION_TYPE | EVENTLOG_AUDIT_SUCCESS | EVENTLOG_AUDIT_FAILURE;
  2129. RegSetValueEx(hk, "TypesSupported", 0, REG_DWORD, (LPBYTE) &dwData, sizeof(DWORD));
  2130. dwData = 16;
  2131. RegSetValueEx(hk, "CategoryCount", 0, REG_DWORD, (LPBYTE) &dwData, sizeof(DWORD));
  2132. }
  2133. RegCloseKey(hk);
  2134. }
  2135. hEventLog = RegisterEventSource(NULL,"Seisint");
  2136. if (!hEventLog) {
  2137. ERRLOG("reportEventLog: Could not register Seisint event source");
  2138. hEventLog=(HANDLE)-1;
  2139. return false;
  2140. }
  2141. }
  2142. if (hEventLog==(HANDLE)-1)
  2143. return false;
  2144. assertex((unsigned)eventtype<=16);
  2145. if (!data)
  2146. datasize = 0;
  2147. else if (!datasize)
  2148. data = NULL;
  2149. #if 1 //useful for debugging...
  2150. ReportEvent(hEventLog, eventtype, category, eventid, NULL, 1, datasize, &msg, (LPVOID)data);
  2151. #else
  2152. if(datasize)
  2153. {
  2154. char * buff = (char *)malloc(datasize*3+1);
  2155. unsigned char const * cdata = (unsigned char *)data;
  2156. unsigned i;
  2157. for(i=0; i<datasize; i++)
  2158. sprintf(buff+i*3, "%02X ", cdata[i]);
  2159. buff[datasize*3-1] = 0;
  2160. DBGLOG("ReportEvent: type=%X categoryid=%X eventid=%X msg='%s' data=[%s]", eventtype, category, eventid, msg, buff);
  2161. free(buff);
  2162. }
  2163. else
  2164. DBGLOG("ReportEvent: type=%X categoryid=%X eventid=%X msg='%s'", eventtype, category, eventid, msg);
  2165. #endif
  2166. return true;
  2167. }
  2168. CSysLogEventLogger::~CSysLogEventLogger()
  2169. {
  2170. if (hEventLog!=0)
  2171. DeregisterEventSource(hEventLog);
  2172. }
  2173. #else
  2174. #include <syslog.h>
  2175. #define CATEGORY_AUDIT_FUNCTION_REQUIRED
  2176. #define AUDIT_TYPES_BEGIN int auditTypeDataMap[NUM_AUDIT_TYPES+1] = {
  2177. #define MAKE_AUDIT_TYPE(name, type, categoryid, eventid, level) level,
  2178. #define AUDIT_TYPES_END 0 };
  2179. #include "jelogtype.hpp"
  2180. #undef CATEGORY_AUDIT_FUNCTION_REQUIRED
  2181. #undef AUDIT_TYPES_BEGIN
  2182. #undef MAKE_AUDIT_TYPE
  2183. #undef AUDIT_TYPES_END
  2184. CSysLogEventLogger::CSysLogEventLogger() : dataLogUsed(false), dataLogName(0), dataLogFile(-1)
  2185. {
  2186. StringBuffer folder;
  2187. const char * processName = splitDirTail(queryCurrentProcessPath(), folder);
  2188. if (!processName||!*processName)
  2189. processName = "hpcc";
  2190. openlog(processName, LOG_PID, LOG_USER);
  2191. }
  2192. CSysLogEventLogger::~CSysLogEventLogger()
  2193. {
  2194. if(dataLogFile != -1)
  2195. close(dataLogFile);
  2196. if(dataLogName)
  2197. delete [] dataLogName;
  2198. closelog();
  2199. }
  2200. bool CSysLogEventLogger::log(AuditType auditType, const char *msg, size32_t datasize, const void * data)
  2201. {
  2202. assertex(auditType < NUM_AUDIT_TYPES);
  2203. int level = auditTypeDataMap[auditType];
  2204. return linuxReport(level, msg, datasize, data);
  2205. }
  2206. bool CSysLogEventLogger::linuxReport(int level, const char * msg, size32_t datasize, const void * data)
  2207. {
  2208. if (!data)
  2209. datasize = 0;
  2210. else if (!datasize)
  2211. data = NULL;
  2212. bool ret = true;
  2213. #if 1 //useful for debugging...
  2214. if(data)
  2215. {
  2216. if(!dataLogUsed)
  2217. openDataLog();
  2218. if(dataLogFile != -1)
  2219. {
  2220. int fpos = writeDataLog(datasize, (byte const *)data);
  2221. if(fpos != -1)
  2222. syslog(level, "%s [0x%X bytes of data at %s byte 0x%X]", msg, datasize, dataLogName, fpos);
  2223. else
  2224. syslog(level, "%s [could not write 0x%X bytes of data to %s]", msg, datasize, dataLogName);
  2225. }
  2226. else
  2227. {
  2228. ret = false;
  2229. syslog(level, "%s [could not open file of form %s to write data]", msg, AUDIT_DATA_LOG_TEMPLATE);
  2230. }
  2231. }
  2232. else
  2233. {
  2234. syslog(level, "%s", msg);
  2235. }
  2236. #else
  2237. if(datasize)
  2238. {
  2239. char * buff = (char *)malloc(datasize*3+1);
  2240. unsigned char const * cdata = (unsigned char *)data;
  2241. unsigned i;
  2242. for(i=0; i<datasize; i++)
  2243. sprintf(buff+i*3, "%02X ", cdata[i]);
  2244. buff[datasize*3-1] = 0;
  2245. DBGLOG("syslog: priority=%X msg='%s' data=[%s]", level, msg, buff);
  2246. free(buff);
  2247. }
  2248. else
  2249. DBGLOG("syslog: priority=%X msg='%s'", level, msg);
  2250. #endif
  2251. return ret;
  2252. }
  2253. void CSysLogEventLogger::openDataLog()
  2254. {
  2255. CriticalBlock block(dataLogLock);
  2256. dataLogUsed = true;
  2257. unsigned len = strlen(AUDIT_DATA_LOG_TEMPLATE);
  2258. dataLogName = new char[len+1];
  2259. strcpy(dataLogName, AUDIT_DATA_LOG_TEMPLATE);
  2260. dataLogFile = mkstemp(dataLogName);
  2261. }
  2262. int CSysLogEventLogger::writeDataLog(size32_t datasize, byte const * data)
  2263. {
  2264. CriticalBlock block(dataLogLock);
  2265. off_t fpos = lseek(dataLogFile, 0, SEEK_CUR);
  2266. while(datasize > 0)
  2267. {
  2268. ssize_t written = write(dataLogFile, data, datasize);
  2269. if (written == -1)
  2270. return -1;
  2271. data += written;
  2272. datasize -= written;
  2273. }
  2274. #ifndef _WIN32
  2275. #ifdef F_FULLFSYNC
  2276. fcntl(dataLogFile, F_FULLFSYNC);
  2277. #else
  2278. fdatasync(dataLogFile);
  2279. #endif
  2280. #ifdef POSIX_FADV_DONTNEED
  2281. posix_fadvise(dataLogFile, 0, 0, POSIX_FADV_DONTNEED);
  2282. #endif
  2283. #endif
  2284. return fpos;
  2285. }
  2286. #endif
  2287. void SysLogMsgHandler::handleMessage(const LogMsg & msg) const
  2288. {
  2289. AuditType type = categoryToAuditType(msg.queryCategory());
  2290. StringBuffer text;
  2291. msg.toStringPlain(text, fields);
  2292. logger->log(type, text.str());
  2293. }
  2294. void SysLogMsgHandler::addToPTree(IPropertyTree * tree) const
  2295. {
  2296. IPropertyTree * handlerTree = createPTree(ipt_caseInsensitive);
  2297. handlerTree->setProp("@type", "audit");
  2298. tree->addPropTree("handler", handlerTree);
  2299. }
  2300. // Default implementations of the functions in IContextLogger interface
  2301. void IContextLogger::CTXLOG(const char *format, ...) const
  2302. {
  2303. va_list args;
  2304. va_start(args, format);
  2305. CTXLOGva(format, args);
  2306. va_end(args);
  2307. }
  2308. void IContextLogger::logOperatorException(IException *E, const char *file, unsigned line, const char *format, ...) const
  2309. {
  2310. va_list args;
  2311. va_start(args, format);
  2312. logOperatorExceptionVA(E, file, line, format, args);
  2313. va_end(args);
  2314. }
  2315. class DummyLogCtx : implements IContextLogger
  2316. {
  2317. public:
  2318. // It's a static object - we don't want to actually link-count it...
  2319. virtual void Link() const {}
  2320. virtual bool Release() const { return false; }
  2321. virtual void CTXLOGva(const char *format, va_list args) const __attribute__((format(printf,2,0)))
  2322. {
  2323. StringBuffer ss;
  2324. ss.valist_appendf(format, args);
  2325. DBGLOG("%s", ss.str());
  2326. }
  2327. virtual void logOperatorExceptionVA(IException *E, const char *file, unsigned line, const char *format, va_list args) const __attribute__((format(printf,5,0)))
  2328. {
  2329. StringBuffer ss;
  2330. ss.append("ERROR");
  2331. if (E)
  2332. ss.append(": ").append(E->errorCode());
  2333. if (file)
  2334. ss.appendf(": %s(%d) ", sanitizeSourceFile(file), line);
  2335. if (E)
  2336. E->errorMessage(ss.append(": "));
  2337. if (format)
  2338. ss.append(": ").valist_appendf(format, args);
  2339. LOG(MCoperatorProgress, unknownJob, "%s", ss.str());
  2340. }
  2341. virtual void noteStatistic(StatisticKind kind, unsigned __int64 value) const
  2342. {
  2343. }
  2344. virtual void mergeStats(const CRuntimeStatisticCollection &from) const
  2345. {
  2346. }
  2347. virtual unsigned queryTraceLevel() const
  2348. {
  2349. return 0;
  2350. }
  2351. } dummyContextLogger;
  2352. extern jlib_decl const IContextLogger &queryDummyContextLogger()
  2353. {
  2354. return dummyContextLogger;
  2355. }
  2356. extern jlib_decl void UseSysLogForOperatorMessages(bool use)
  2357. {
  2358. static ILogMsgHandler *msgHandler=NULL;
  2359. if (use==(msgHandler!=NULL))
  2360. return;
  2361. if (use) {
  2362. msgHandler = getSysLogMsgHandler();
  2363. ILogMsgFilter * operatorFilter = getCategoryLogMsgFilter(MSGAUD_operator|MSGAUD_monitor, MSGCLS_all, DefaultDetail, true);
  2364. queryLogMsgManager()->addMonitorOwn(msgHandler, operatorFilter);
  2365. }
  2366. else {
  2367. queryLogMsgManager()->removeMonitor(msgHandler);
  2368. msgHandler = NULL;
  2369. }
  2370. }
  2371. extern jlib_decl void AuditSystemAccess(const char *userid, bool success, char const * msg,...)
  2372. {
  2373. va_list args;
  2374. va_start(args, msg);
  2375. VStringBuffer s("User %s: ", userid);
  2376. SYSLOG((success) ? AUDIT_TYPE_ACCESS_SUCCESS : AUDIT_TYPE_ACCESS_FAILURE, s.valist_appendf(msg, args).str());
  2377. va_end(args);
  2378. }
  2379. //--------------------------------------------------------------
  2380. class jlib_decl CComponentLogFileCreator : implements IComponentLogFileCreator, public CInterface
  2381. {
  2382. private:
  2383. StringBuffer component;
  2384. //filename parts
  2385. StringBuffer prefix;
  2386. StringBuffer name;
  2387. StringBuffer postfix;
  2388. StringBuffer extension;
  2389. StringBuffer fullFileSpec;
  2390. bool createAlias;
  2391. StringBuffer aliasName;
  2392. StringBuffer logDirSubdir;
  2393. bool rolling;
  2394. //ILogMsgHandler fields
  2395. bool append;
  2396. bool flushes;
  2397. unsigned msgFields;
  2398. //ILogMsgFilter fields
  2399. unsigned msgAudiences;
  2400. unsigned msgClasses;
  2401. LogMsgDetail maxDetail;
  2402. bool local;
  2403. //available after logging started
  2404. StringBuffer logDir; //access via queryLogDir()
  2405. StringBuffer aliasFileSpec; //access via queryAliasFileSpec()
  2406. StringBuffer expandedLogSpec;//access via queryLogFileSpec()
  2407. private:
  2408. void setDefaults()
  2409. {
  2410. rolling = true;
  2411. append = true;
  2412. flushes = true;
  2413. const char *logFields = queryEnvironmentConf().queryProp("logfields");
  2414. if (!isEmptyString(logFields))
  2415. msgFields = LogMsgFieldsFromAbbrevs(logFields);
  2416. else
  2417. msgFields = MSGFIELD_STANDARD;
  2418. msgAudiences = MSGAUD_all;
  2419. msgClasses = MSGCLS_all;
  2420. maxDetail = DefaultDetail;
  2421. name.set(component); //logfile defaults to component name. Change via setName(), setPrefix() and setPostfix()
  2422. extension.set(".log");
  2423. local = false;
  2424. createAlias = true;
  2425. }
  2426. public:
  2427. IMPLEMENT_IINTERFACE;
  2428. CComponentLogFileCreator(IPropertyTree * _properties, const char *_component) : component(_component)
  2429. {
  2430. setDefaults();
  2431. if (_properties && !getConfigurationDirectory(_properties->queryPropTree("Directories"), "log", _component, _properties->queryProp("@name"), logDir))
  2432. _properties->getProp("@logDir", logDir);
  2433. }
  2434. CComponentLogFileCreator(const char *_logDir, const char *_component) : component(_component), logDir(_logDir)
  2435. {
  2436. setDefaults();
  2437. }
  2438. CComponentLogFileCreator(const char *_component) : component(_component)
  2439. {
  2440. setDefaults();
  2441. if (!getConfigurationDirectory(NULL, "log", _component, _component, logDir))
  2442. {
  2443. appendCurrentDirectory(logDir,false);
  2444. }
  2445. }
  2446. //set methods
  2447. void setExtension(const char * _ext) { extension.set(_ext); }
  2448. void setPrefix(const char * _prefix) { prefix.set(_prefix); }
  2449. void setName(const char * _name) { name.set(_name); }
  2450. void setCompleteFilespec(const char * _fs){fullFileSpec.set(_fs); setExtension(NULL); setRolling(false);}
  2451. void setPostfix(const char * _postfix) { postfix.set(_postfix); }
  2452. void setCreateAliasFile(bool _create) { createAlias = _create; }
  2453. void setAliasName(const char * _aliasName) { aliasName.set(_aliasName); }
  2454. void setLogDirSubdir(const char * _subdir) { logDirSubdir.set(_subdir); }
  2455. void setRolling(const bool _rolls) { rolling = _rolls; }
  2456. //ILogMsgHandler fields
  2457. void setAppend(const bool _append) { append = _append; }
  2458. void setFlushes(const bool _flushes) { flushes = _flushes; }
  2459. void setMsgFields(const unsigned _fields){ msgFields = _fields; }
  2460. //ILogMsgFilter fields
  2461. void setMsgAudiences(const unsigned _audiences){ msgAudiences = _audiences; }
  2462. void setMsgClasses(const unsigned _classes) { msgClasses = _classes; }
  2463. void setMaxDetail(const LogMsgDetail _maxDetail) { maxDetail = _maxDetail; }
  2464. void setLocal(const bool _local) { local = _local; }
  2465. //query methods (not valid until logging started)
  2466. const char * queryLogDir() const { return logDir.str(); }
  2467. const char * queryLogFileSpec() const { return expandedLogSpec.str(); }
  2468. const char * queryAliasFileSpec() const { return aliasFileSpec.str(); }
  2469. ILogMsgHandler * beginLogging()
  2470. {
  2471. //build directory path
  2472. StringBuffer logFileSpec;
  2473. if (!fullFileSpec.length())//user specify complete logfile specification?
  2474. {
  2475. if (!logDir.length())
  2476. {
  2477. appendCurrentDirectory(logDir,false).append(PATHSEPSTR).append("logs");
  2478. WARNLOG("No logfile directory specified - logs will be written locally to %s", logDir.str());
  2479. }
  2480. makeAbsolutePath(logDir);
  2481. //build log file name (without date string or extension)
  2482. StringBuffer logFileName;
  2483. if (prefix.length())
  2484. logFileName.append(prefix).append(".");
  2485. logFileName.append(name);
  2486. if (postfix.length())
  2487. logFileName.append(".").append(postfix);
  2488. //build log file spec
  2489. if (logDirSubdir.length())
  2490. logDir.append(PATHSEPCHAR).append(logDirSubdir);//user specified subfolder
  2491. logFileSpec.append(logDir).append(PATHSEPCHAR).append(logFileName);
  2492. //build alias file spec
  2493. if (createAlias)
  2494. {
  2495. if (aliasName.length()==0)
  2496. aliasName.set(logFileName);
  2497. aliasFileSpec.append(logDir).append(PATHSEPCHAR).append(aliasName).append(extension);
  2498. }
  2499. }
  2500. else
  2501. makeAbsolutePath(fullFileSpec);
  2502. ILogMsgHandler * lmh;
  2503. if (rolling)
  2504. {
  2505. lmh = getRollingFileLogMsgHandler(logFileSpec.str(), extension, msgFields, append, flushes, NULL, aliasFileSpec.str(), true);
  2506. }
  2507. else
  2508. {
  2509. StringBuffer lfs;
  2510. if (fullFileSpec.length())
  2511. lfs.set(fullFileSpec);
  2512. else
  2513. lfs.set(logFileSpec.append(extension).str());
  2514. lmh = getFileLogMsgHandler(lfs.str(), NULL, msgFields, false);
  2515. }
  2516. lmh->getLogName(expandedLogSpec);
  2517. queryLogMsgManager()->addMonitorOwn( lmh, getCategoryLogMsgFilter(msgAudiences, msgClasses, maxDetail, local));
  2518. return lmh;
  2519. }
  2520. };
  2521. IComponentLogFileCreator * createComponentLogFileCreator(IPropertyTree * _properties, const char *_component)
  2522. {
  2523. return new CComponentLogFileCreator(_properties, _component);
  2524. }
  2525. IComponentLogFileCreator * createComponentLogFileCreator(const char *_logDir, const char *_component)
  2526. {
  2527. return new CComponentLogFileCreator(_logDir, _component);
  2528. }
  2529. IComponentLogFileCreator * createComponentLogFileCreator(const char *_component)
  2530. {
  2531. return new CComponentLogFileCreator(_component);
  2532. }