1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054 |
- /*
- * Copyright 2005 - 2016 Zarafa and its licensors
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License, version 3,
- * as published by the Free Software Foundation.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
- *
- */
- #include <kopano/platform.h>
- #include <iostream>
- #include <climits>
- #include <cstdio>
- #include <cstdlib>
- #include <ctime>
- #include <getopt.h>
- #include <kopano/memory.hpp>
- #include <mapidefs.h>
- #include <mapispi.h>
- #include <mapix.h>
- #include <mapiutil.h>
- #include <kopano/automapi.hpp>
- #include <kopano/IECServiceAdmin.h>
- #include <kopano/IECSecurity.h>
- #include <kopano/IECUnknown.h>
- #include "SSLUtil.h"
- #include <kopano/ECConfig.h>
- #include <kopano/ECLogger.h>
- #include <kopano/ECTags.h>
- #include <kopano/EMSAbTag.h>
- #include <kopano/ECGuid.h>
- #include <kopano/ECABEntryID.h>
- #include <kopano/CommonUtil.h>
- #include <kopano/stringutil.h>
- #include <kopano/ecversion.h>
- #include <kopano/mapiext.h>
- #include <kopano/Util.h>
- #include <kopano/ECRestriction.h>
- #include <kopano/charset/convert.h>
- #include "ConsoleTable.h"
- #include <kopano/mapi_ptr.h>
- #include "ECFeatures.h"
- #include "ECACL.h"
- #include "charset/localeutil.h"
- #include <kopano/MAPIErrors.h>
- #include <edkmdb.h>
- #include <edkguid.h>
- #include <libintl.h>
- #include "Archiver.h"
- #include <kopano/MAPIErrors.h> // for declaration of GetMAPIErrorMessage()
- using namespace std;
- using namespace KCHL;
- static int forcedExitCode = 0;
- enum modes {
- MODE_INVALID = 0, MODE_LIST_USERS, MODE_CREATE_PUBLIC,
- MODE_CREATE_USER, MODE_CREATE_STORE, MODE_HOOK_STORE, MODE_UNHOOK_STORE,
- MODE_DELETE_STORE, MODE_REMOVE_STORE, MODE_UPDATE_USER, MODE_DELETE_USER,
- MODE_CREATE_GROUP, MODE_UPDATE_GROUP, MODE_DELETE_GROUP,
- MODE_LIST_GROUP, MODE_ADDUSER_GROUP, MODE_DELETEUSER_GROUP,
- MODE_CREATE_COMPANY, MODE_UPDATE_COMPANY, MODE_DELETE_COMPANY, MODE_LIST_COMPANY,
- MODE_ADD_VIEW, MODE_DEL_VIEW, MODE_LIST_VIEW,
- MODE_ADD_ADMIN, MODE_DEL_ADMIN, MODE_LIST_ADMIN,
- MODE_ADD_USERQUOTA_RECIPIENT, MODE_DEL_USERQUOTA_RECIPIENT, MODE_LIST_USERQUOTA_RECIPIENT,
- MODE_ADD_COMPANYQUOTA_RECIPIENT, MODE_DEL_COMPANYQUOTA_RECIPIENT, MODE_LIST_COMPANYQUOTA_RECIPIENT,
- MODE_SYNC_USERS, MODE_DETAILS, MODE_LIST_SENDAS, MODE_HELP,
- MODE_SYSTEM_ADMIN, MODE_PURGE_SOFTDELETE, MODE_PURGE_DEFERRED, MODE_CLEAR_CACHE, MODE_LIST_ORPHANS,
- MODE_FORCE_RESYNC, MODE_USER_COUNT, MODE_RESET_FOLDER_COUNT
- };
- enum {
- OPT_CREATE_STORE = UCHAR_MAX + 1, // high to avoid clashes with modes
- OPT_DELETE_STORE,
- OPT_HOOK_STORE,
- OPT_UNHOOK_STORE,
- OPT_REMOVE_STORE,
- OPT_COPYTO_PUBLIC,
- OPT_HELP,
- OPT_HOST,
- OPT_SYNC_USERS,
- OPT_DETAILS,
- OPT_DETAILS_TYPE,
- OPT_USER_QUOTA_HARD,
- OPT_USER_QUOTA_SOFT,
- OPT_USER_QUOTA_WARN,
- OPT_USER_QUOTA_OVERRIDE,
- OPT_USER_DEFAULT_QUOTA_HARD,
- OPT_USER_DEFAULT_QUOTA_SOFT,
- OPT_USER_DEFAULT_QUOTA_WARN,
- OPT_USER_DEFAULT_QUOTA_OVERRIDE,
- OPT_LANG,
- OPT_MR_ACCEPT,
- OPT_MR_DECLINE_CONFLICT,
- OPT_MR_DECLINE_RECURRING,
- OPT_ADD_SENDAS,
- OPT_DEL_SENDAS,
- OPT_LIST_SENDAS,
- OPT_UPDATE_GROUP,
- OPT_CREATE_COMPANY,
- OPT_UPDATE_COMPANY,
- OPT_DELETE_COMPANY,
- OPT_LIST_COMPANY,
- OPT_ADD_VIEW,
- OPT_DEL_VIEW,
- OPT_LIST_VIEW,
- OPT_ADD_ADMIN,
- OPT_DEL_ADMIN,
- OPT_LIST_ADMIN,
- OPT_SYSTEM_ADMIN,
- OPT_ADD_UQUOTA_RECIPIENT,
- OPT_DEL_UQUOTA_RECIPIENT,
- OPT_LIST_UQUOTA_RECIPIENT,
- OPT_ADD_CQUOTA_RECIPIENT,
- OPT_DEL_CQUOTA_RECIPIENT,
- OPT_LIST_CQUOTA_RECIPIENT,
- OPT_PURGE_SOFTDELETE,
- OPT_PURGE_DEFERRED,
- OPT_CLEAR_CACHE,
- OPT_LIST_ORPHANS,
- OPT_CONFIG,
- OPT_UTF8,
- OPT_FORCE_RESYNC,
- OPT_USER_COUNT,
- OPT_ENABLE_FEATURE,
- OPT_DISABLE_FEATURE,
- OPT_SELECT_NODE,
- OPT_RESET_FOLDER_COUNT,
- OPT_VERBOSITY,
- OPT_VERSION,
- };
- static const struct option long_options[] = {
- { "create-store", 1, NULL, OPT_CREATE_STORE },
- { "delete-store", 1, NULL, OPT_DELETE_STORE },
- { "hook-store", 1, NULL, OPT_HOOK_STORE },
- { "unhook-store", 1, NULL, OPT_UNHOOK_STORE },
- { "remove-store", 1, NULL, OPT_REMOVE_STORE },
- { "copyto-public", 0, NULL, OPT_COPYTO_PUBLIC },
- { "list-orphans", 0, NULL, OPT_LIST_ORPHANS },
- { "details", 1, NULL, OPT_DETAILS },
- { "type", 1, NULL, OPT_DETAILS_TYPE },
- { "help", 0, NULL, OPT_HELP },
- { "host", 1, NULL, OPT_HOST },
- { "sync", 0, NULL, OPT_SYNC_USERS },
- { "qh", 1, NULL, OPT_USER_QUOTA_HARD },
- { "qs", 1, NULL, OPT_USER_QUOTA_SOFT },
- { "qw", 1, NULL, OPT_USER_QUOTA_WARN },
- { "qo", 1, NULL, OPT_USER_QUOTA_OVERRIDE },
- { "udqh", 1, NULL, OPT_USER_DEFAULT_QUOTA_HARD },
- { "udqs", 1, NULL, OPT_USER_DEFAULT_QUOTA_SOFT },
- { "udqw", 1, NULL, OPT_USER_DEFAULT_QUOTA_WARN },
- { "udqo", 1, NULL, OPT_USER_DEFAULT_QUOTA_OVERRIDE },
- { "lang", 1, NULL, OPT_LANG },
- { "mr-accept", 1, NULL, OPT_MR_ACCEPT },
- { "mr-decline-conflict", 1, NULL, OPT_MR_DECLINE_CONFLICT },
- { "mr-decline-recurring", 1, NULL, OPT_MR_DECLINE_RECURRING },
- { "add-sendas", 1, NULL, OPT_ADD_SENDAS },
- { "del-sendas", 1, NULL, OPT_DEL_SENDAS },
- { "list-sendas", 1, NULL, OPT_LIST_SENDAS },
- { "update-group", 1, NULL, OPT_UPDATE_GROUP },
- { "create-company", 1, NULL, OPT_CREATE_COMPANY },
- { "update-company", 1, NULL, OPT_UPDATE_COMPANY },
- { "delete-company", 1, NULL, OPT_DELETE_COMPANY },
- { "list-companies", 0, NULL, OPT_LIST_COMPANY },
- { "add-to-viewlist", 1, NULL, OPT_ADD_VIEW },
- { "del-from-viewlist", 1, NULL, OPT_DEL_VIEW },
- { "list-view", 0, NULL, OPT_LIST_VIEW },
- { "add-to-adminlist", 1, NULL, OPT_ADD_ADMIN },
- { "del-from-adminlist", 1, NULL, OPT_DEL_ADMIN },
- { "list-admin", 0, NULL, OPT_LIST_ADMIN },
- { "set-system-admin", 1, NULL, OPT_SYSTEM_ADMIN },
- { "add-userquota-recipient", 1, NULL, OPT_ADD_UQUOTA_RECIPIENT },
- { "del-userquota-recipient", 1, NULL, OPT_DEL_UQUOTA_RECIPIENT },
- { "list-userquota-recipients", 0, NULL, OPT_LIST_UQUOTA_RECIPIENT },
- { "add-companyquota-recipient", 1, NULL, OPT_ADD_CQUOTA_RECIPIENT },
- { "del-companyquota-recipient", 1, NULL, OPT_DEL_CQUOTA_RECIPIENT },
- { "list-companyquota-recipients", 0, NULL, OPT_LIST_CQUOTA_RECIPIENT },
- { "purge-softdelete", 1, NULL, OPT_PURGE_SOFTDELETE },
- { "purge-deferred", 0, NULL, OPT_PURGE_DEFERRED },
- { "clear-cache", 2, NULL, OPT_CLEAR_CACHE },
- { "config", 1, NULL, OPT_CONFIG },
- { "utf8", 0, NULL, OPT_UTF8 },
- { "force-resync", 0, NULL, OPT_FORCE_RESYNC },
- { "user-count", 0, NULL, OPT_USER_COUNT },
- { "enable-feature", 1, NULL, OPT_ENABLE_FEATURE },
- { "disable-feature", 1, NULL, OPT_DISABLE_FEATURE },
- { "node", 1, NULL, OPT_SELECT_NODE },
- { "reset-folder-count", 1, NULL, OPT_RESET_FOLDER_COUNT },
- { "verbose", required_argument, NULL, OPT_VERBOSITY },
- { "version", no_argument, NULL, OPT_VERSION },
- { NULL, 0, NULL, 0 }
- };
- /**
- * Prints all options on screen. This should always be in sync with reality.
- *
- * @param[in] name The name of the program (arg[0])
- */
- static void print_help(const char *name)
- {
- ConsoleTable ct(0,0);
- cout << "Usage:" << endl;
- cout << name << " [action] [options]" << endl << endl;
- cout << "Actions: [-s] | [[-c|-u|-d|-b|-B|--details] username] | [[-g|-G] groupname] | [-l|-L]" << endl;
- ct.Resize(15, 2);
- ct.AddColumn(0, "-s"); ct.AddColumn(1, "Create public store.");
- ct.AddColumn(0, "--sync"); ct.AddColumn(1, "Synchronize users and groups with external source.");
- ct.AddColumn(0, "--clear-cache"); ct.AddColumn(1, "Clear all caches in the server.");
- ct.AddColumn(0, "--purge-softdelete N"); ct.AddColumn(1, "Purge items in marked as softdeleted that are older than N days.");
- ct.AddColumn(0, "--purge-deferred"); ct.AddColumn(1, "Purge all items in the deferred update table.");
- ct.AddColumn(0, "-l"); ct.AddColumn(1, "List users. Use -I to list users of a specific company, if applicable.");
- ct.AddColumn(0, "-L"); ct.AddColumn(1, "List groups. Use -I to list groups of a specific company, if applicable.");
- ct.AddColumn(0, "--list-sendas name"); ct.AddColumn(1, "List all users who are allowed to send-as 'name'. Use --type to indicate the object type.");
- ct.AddColumn(0, "--list-companies"); ct.AddColumn(1, "List all companies.");
- ct.AddColumn(0, "--list-view"); ct.AddColumn(1, "List all companies in the remote-view list.");
- ct.AddColumn(0, "--list-admin"); ct.AddColumn(1, "List all users in the remote-admin list.");
- ct.AddColumn(0, "--list-userquota-recipients"); ct.AddColumn(1, "List all additional recipients for a userquota warning email.");
- ct.AddColumn(0, "--list-companyquota-recipients"); ct.AddColumn(1, "List all additional recipients for a companyquota warning email.");
- ct.AddColumn(0, "--details"); ct.AddColumn(1, "Show object details, use --type to indicate the object type.");
- ct.AddColumn(0, "--type type"); ct.AddColumn(1, "Set object type for --details. Values can be \"user\", \"group\" or \"company\".");
- ct.AddColumn(0, "--user-count"); ct.AddColumn(1, "Output the system users counts.");
- ct.PrintTable();
- cout << endl;
- cout << "Additional Actions when using the DB user plugin:" << endl;
- ct.Resize(8,2);
- ct.AddColumn(0, "-c user"); ct.AddColumn(1, "Create user, -p, -f, -e options required, -a and -n are optional. Quota options are optional.");
- ct.AddColumn(0, "-u user"); ct.AddColumn(1, "Update user, -U, -p, -f, -e, -n and -a optional. Quota options are optional.");
- ct.AddColumn(0, "-d user"); ct.AddColumn(1, "Delete user.");
- ct.AddColumn(0, "-g group"); ct.AddColumn(1, "Create group, -e options optional.");
- ct.AddColumn(0, "--update-group group"); ct.AddColumn(1, "Update group, -e optional.");
- ct.AddColumn(0, "-G group"); ct.AddColumn(1, "Delete group.");
- ct.AddColumn(0, "-b user"); ct.AddColumn(1, "Add user to a group, -i required.");
- ct.AddColumn(0, "-B user"); ct.AddColumn(1, "Delete user from a group, -i required.");
- ct.PrintTable();
- cout << endl;
- cout << "Additional Actions when using the Unix user plugin:" << endl;
- ct.Resize(2,2);
- ct.AddColumn(0, "-u user"); ct.AddColumn(1, "Update user, -e and -a optional. Quota options are optional.");
- ct.AddColumn(0, "--update-group group"); ct.AddColumn(1, "Update group, -e optional.");
- ct.PrintTable();
- cout << endl;
- cout << "Additional Actions when using the DB or Unix user plugin:" << endl;
- ct.Resize(2,2);
- ct.AddColumn(0, "--enable-feature feature"); ct.AddColumn(1, "Update user to explicitly enable a feature.");
- ct.AddColumn(0, "--disable-feature feature"); ct.AddColumn(1, "Update user to explicitly disable a feature.");
- ct.PrintTable();
- cout << endl;
- cout << "Additional Actions when using the DB user plugin in hosted mode:" << endl;
- ct.Resize(12,2);
- ct.AddColumn(0, "--create-company name"); ct.AddColumn(1, "Create company space");
- ct.AddColumn(0, "--update-company name"); ct.AddColumn(1, "Update company space");
- ct.AddColumn(0, "--delete-company name"); ct.AddColumn(1, "Delete company space");
- ct.AddColumn(0, "--set-system-admin name"); ct.AddColumn(1, "Set system administrator for the company specified by -I (does not grant Admin privileges)");
- ct.AddColumn(0, "--add-to-viewlist name"); ct.AddColumn(1, "Add company 'name' to remote-view list of company specified by -I");
- ct.AddColumn(0, "--del-from-viewlist name"); ct.AddColumn(1, "Delete company 'name' from remote-view list of company specified by -I");
- ct.AddColumn(0, "--add-to-adminlist name"); ct.AddColumn(1, "Add user 'name' to remote-admin list of company specified by -I");
- ct.AddColumn(0, "--del-from-adminlist name"); ct.AddColumn(1, "Delete user 'name' from remote-admin list of company specified by -I");
- ct.AddColumn(0, "--add-userquota-recipient user"); ct.AddColumn(1, "Add 'user' as recipient to userquota warning emails.");
- ct.AddColumn(0, "--del-userquota-recipient user"); ct.AddColumn(1, "Delete 'user' as recipient to userquota warning emails.");
- ct.AddColumn(0, "--add-companyquota-recipient user"); ct.AddColumn(1, "Add 'user' as recipient to companyquota warning emails.");
- ct.AddColumn(0, "--del-companyquota-recipient user"); ct.AddColumn(1, "Delete 'user' as recipient to companyquota warning emails.");
- ct.PrintTable();
- cout << endl;
- cout << "Options: [-U new username] [-P|-p password] [-f fullname] [-e emailaddress]" << endl;
- cout << " [-a [y|n]] [-n [y|n]] [-h path] [-i group] [--qo [y|n]] [--qh value] [--qs value] [--qw value]" << endl;
- ct.Resize(22,2);
- ct.AddColumn(0, "-U new username"); ct.AddColumn(1, "Rename username to new username");
- ct.AddColumn(0, "-P"); ct.AddColumn(1, "Prompt for password, can be substituted by '-p pass'");
- ct.AddColumn(0, "-p pass"); ct.AddColumn(1, "Set password to pass, can be substituted by '-P'");
- ct.AddColumn(0, "-f full"); ct.AddColumn(1, "Set fullname to full");
- ct.AddColumn(0, "-e addr"); ct.AddColumn(1, "Set email address to addr");
- ct.AddColumn(0, "-a [y|n]"); ct.AddColumn(1, "Set administrator level for user. yes/no, y/n or 2/1/0.");
- ct.AddColumn(0, "-n [y|n]"); ct.AddColumn(1, "set user to non-active. yes/no, y/n or 1/0.");
- ct.AddColumn(0, "--qo [y|n]"); ct.AddColumn(1, "Override server quota limits. yes/no, y/n or 1/0.");
- ct.AddColumn(0, "--qh hardquota"); ct.AddColumn(1, "Set hardquota limit in Mb");
- ct.AddColumn(0, "--qs softquota"); ct.AddColumn(1, "Set softquota limit in Mb");
- ct.AddColumn(0, "--qw warnquota"); ct.AddColumn(1, "Set warnquota limit in Mb");
- ct.AddColumn(0, "--udqo [y|n]"); ct.AddColumn(1, "Override user default server quota limits for specific company. yes/no, y/n or 1/0.");
- ct.AddColumn(0, "--udqh hardquota"); ct.AddColumn(1, "Set user default hardquota limit for specific company in Mb");
- ct.AddColumn(0, "--udqs softquota"); ct.AddColumn(1, "Set user default softquota limit for specific company in Mb");
- ct.AddColumn(0, "--udqw warnquota"); ct.AddColumn(1, "Set user default warnquota limit for specific company in Mb");
- ct.AddColumn(0, "-i group"); ct.AddColumn(1, "Name of the group");
- ct.AddColumn(0, "-I company"); ct.AddColumn(1, "Name of the company");
- ct.AddColumn(0, "--mr-accept"); ct.AddColumn(1, "(resource) auto-accept meeting requests. yes/no");
- ct.AddColumn(0, "--mr-decline-conflict"); ct.AddColumn(1, "(resource) decline meeting requests for conflicting times. yes/no");
- ct.AddColumn(0, "--mr-decline-recurring"); ct.AddColumn(1, "(resource) decline meeting requests for all recurring items. yes/no");
- ct.AddColumn(0, "--add-sendas name"); ct.AddColumn(1, "Add user 'name' to send-as list of user specified by -u or --update-group");
- ct.AddColumn(0, "--del-sendas name"); ct.AddColumn(1, "Delete user 'name' from send-as list of user specified by -u or --update-group");
- ct.PrintTable();
- cout << endl;
- cout << "The following functions are to control stores of users:" << endl;
- ct.Resize(10,2);
- ct.AddColumn(0, "--list-orphans"); ct.AddColumn(1, "List all users without stores and stores without users.");
- ct.AddColumn(0, "--remove-store storeguid"); ct.AddColumn(1, "Delete orphaned store of user that is deleted from external source.");
- ct.AddColumn(0, "--hook-store storeguid"); ct.AddColumn(1, "Hook orphaned store to a user or copy to a public.");
- ct.AddColumn(0, " -u username"); ct.AddColumn(1, "Update user to received orphaned store given in --hook-store.");
- ct.AddColumn(0, " --type"); ct.AddColumn(1, "Type of the user to hook. Defaults to 'user', can be 'group' or 'company' for public store. Use 'archive' for archive stores.");
- ct.AddColumn(0, " --copyto-public"); ct.AddColumn(1, "Copy the orphan store to the public folder.");
- ct.AddColumn(0, "--unhook-store username"); ct.AddColumn(1, "Unhook store from user.");
- ct.AddColumn(0, " --type"); ct.AddColumn(1, "Type of the user to hook. Defaults to 'user', can be 'group' or 'company' for public store. Use 'archive' for archive stores.");
- ct.AddColumn(0, ""); ct.AddColumn(1, "Use 'Everyone' as username with type 'group' to unhook the public store, or use the company name and type 'company'.");
- ct.AddColumn(0, "--force-resync [username [username [...]]]"); ct.AddColumn(1, "Force a resynchronisation of offline profiles for specified users.");
- ct.AddColumn(0, "--reset-folder-count username"); ct.AddColumn(1, "Reset the counters on all folder in the store.");
- ct.PrintTable();
- cout << endl;
- cout << "The following functions are for use from the create/delete user/group scripts:" << endl;
- ct.Resize(2,2);
- ct.AddColumn(0, "--create-store user"); ct.AddColumn(1, "Create store for user that exists in external source.");
- ct.AddColumn(0, "--lang language"); ct.AddColumn(1, "Create folders in a new store in this language (e.g. en_EN.UTF-8).");
- ct.PrintTable();
- cout << endl;
- cout << "Note: the list-orphans and create/remove/hook/unhook-store functions only work on the server you're connected to. The commands will not be redirected in a multi-server environment." << endl;
- cout << endl;
- cout << "Global options: [-h|--host path]" << endl;
- ct.Resize(4,2);
- ct.AddColumn(0, "--config file"); ct.AddColumn(1, "Use a configuration file");
- ct.AddColumn(0, "-h path"); ct.AddColumn(1, "Connect through <path>, e.g. file:///var/run/socket");
- ct.AddColumn(0, "--node name"); ct.AddColumn(1, "Execute the command on cluster node <name>");
- ct.AddColumn(0, "--utf8"); ct.AddColumn(1, "Force the current locale to UTF-8");
- ct.AddColumn(0, "-v"); ct.AddColumn(1, "Increase verbosity. A maximum of 7 is possible where 1=fatal errors only, 6=debug and 7=everything.");
- ct.AddColumn(0, "--verbosity x"); ct.AddColumn(1, "Set verbosity to value 'x': 0...7 (0 = disable)");
- ct.AddColumn(0, "-V"); ct.AddColumn(1, "Print version info.");
- ct.AddColumn(0, "--version"); ct.AddColumn(1, "Print version info.");
- ct.AddColumn(0, "--help"); ct.AddColumn(1, "Show this help text.");
- ct.PrintTable();
- cout << endl;
- }
- /**
- * Reads a password twice from stdin, and doesn't print on stdout.
- *
- * @return char The typed password if ok, or NULL when failed.
- */
- static char *get_password(void)
- {
- static char password[80] = {0};
- char *s = NULL;
- s = get_password("Type password:");
- if(s == NULL)
- return NULL;
- strncpy(password, s, sizeof(password)-1);
- s = get_password("Retype password:");
- if (s == NULL)
- return NULL;
- if (strcmp(password, s) != 0)
- return NULL;
- return password;
- }
- /**
- * Parse given string to return 1 for true (yes) or 0 for false (no).
- *
- * @note Does not accept uppercase 'yes'
- *
- * @param[in] char* String containing a boolean
- * @return int 1 for true (yes) or 0 for false (no).
- */
- static int parse_yesno(const char *opt)
- {
- return opt[0] == 'y' || opt[0] == '1';
- }
- /**
- * Filetime to string
- *
- * @param ft time to convert to string
- *
- * @return time string "m / %d / %y %T"
- */
- static std::string FiletimeToString(FILETIME ft)
- {
- time_t timestamp;
- tm local;
- char d[64];
- memset(d, 0, sizeof(d));
- FileTimeToUnixTime(ft, ×tamp);
- localtime_r(×tamp, &local);
- strftime(d, sizeof(d), "%x %X", &local);
- return d;
- }
- static std::string UnixtimeToString(time_t timestamp)
- {
- tm local;
- char d[64];
- memset(d, 0, sizeof(d));
- localtime_r(×tamp, &local);
- strftime(d, sizeof(d), "%c", &local);
- return d;
- }
- /**
- * Print quota levels and/or store size.
- *
- * @param[in] lpQuota Optional ECQuota object with a users quota settings
- * @param[in] lpQuotaStatus Optional ECQuotaStatus object with the storesize of a user
- */
- static void print_quota(const ECQUOTA *lpQuota,
- const ECQUOTASTATUS *lpQuotaStatus, bool isPublic = false)
- {
- if (lpQuota) {
- // watch the not:
- if (!isPublic)
- cout << "Current user store quota settings:" << endl;
- else
- cout << "Current public store quota settings:" << endl;
- cout << " Quota overrides:\t" << (!lpQuota->bUseDefaultQuota?"yes":"no") << endl;
- cout << " Warning level:\t\t" << str_storage(lpQuota->llWarnSize) << endl;
- if(!isPublic) {
- cout << " Soft level:\t\t" << str_storage(lpQuota->llSoftSize) << endl;
- cout << " Hard level:\t\t" << str_storage(lpQuota->llHardSize) << endl;
- }
- }
- if (lpQuotaStatus == nullptr)
- return;
- if (!isPublic)
- cout << "Current store size:\t";
- else
- cout << "Public store size:\t";
- cout << str_storage(lpQuotaStatus->llStoreSize, false) << endl;
- }
- /**
- * Set new quota levels for a given user (EntryID). This only works
- * for DB and Unix plugin.
- *
- * @param[in] lpServiceAdmin IECServiceAdmin object on the Admin store
- * @param[in] cbEid Size of lpEid
- * @param[in] lpEid EntryID of a user
- * @param[in] quota New yes/no quota override flag setting, or -1 for default quota settings for user
- * @param[in] udefault IsUserDefaultQuota setting (Default quota for users within company)
- * @param[in] warn New warning level for user, or -1 for default system settings
- * @param[in] soft New soft level for user, or -1 for default system settings
- * @param[in] hard New hard level for user, or -1 for default system settings
- * @param[in] print Prints old and new quota settings for a user (optional, default false)
- */
- static HRESULT setQuota(IECServiceAdmin *lpServiceAdmin, ULONG cbEid,
- LPENTRYID lpEid, int quota, bool udefault, long long warn, long long soft,
- long long hard, bool print = false, bool company = false)
- {
- HRESULT hr = hrSuccess;
- memory_ptr<ECQUOTASTATUS> lpsQuotaStatus;
- memory_ptr<ECQUOTA> lpsQuota;
- ECQUOTA sQuota;
- if (lpEid == NULL)
- return MAPI_E_INVALID_PARAMETER;
- hr = lpServiceAdmin->GetQuota(cbEid, lpEid, false, &~lpsQuota);
- if (hr != hrSuccess) {
- cerr << "Unable to update quota, probably not found." << endl;
- return hr;
- }
- if (print) {
- cout << "Old quota settings:" << endl;
- print_quota(lpsQuota, NULL, company);
- cout << endl;
- }
- if (quota == -1)
- sQuota.bUseDefaultQuota = lpsQuota->bUseDefaultQuota;
- else
- sQuota.bUseDefaultQuota = !quota;
- sQuota.bIsUserDefaultQuota = udefault;
- sQuota.llHardSize = (hard >= 0) ? hard : lpsQuota->llHardSize;
- sQuota.llSoftSize = (soft >= 0) ? soft : lpsQuota->llSoftSize;
- sQuota.llWarnSize = (warn >= 0) ? warn : lpsQuota->llWarnSize;
- hr = lpServiceAdmin->SetQuota(cbEid, lpEid, &sQuota);
- if(hr != hrSuccess) {
- cerr << "Unable to update quota information." << endl;
- return hr;
- }
- if (!print)
- return hrSuccess;
- hr = lpServiceAdmin->GetQuotaStatus(cbEid, lpEid, &~lpsQuotaStatus);
- if(hr != hrSuccess) {
- cerr << "Unable to request updated quota information: " <<
- GetMAPIErrorMessage(hr) << " (" <<
- stringify(hr, true) << ")" << endl;
- return hr;
- }
- cout << "New quota settings:" << endl;
- print_quota(&sQuota, lpsQuotaStatus, company);
- cout << endl;
- return hrSuccess;
- }
- /**
- * Returns the string PR_* name for a set of addressbook properties,
- * not using the type of the tag to compare.
- *
- * @param[in] ulPropTag A MAPI proptag
- * @return string The PR_ string of a property, or the number.
- */
- static string getMapiPropertyString(ULONG ulPropTag)
- {
- #define PROP_TO_STRING(__proptag) \
- case PROP_ID(__proptag): return #__proptag
- switch (PROP_ID(ulPropTag))
- {
- PROP_TO_STRING(PR_MANAGER_NAME);
- PROP_TO_STRING(PR_GIVEN_NAME);
- PROP_TO_STRING(PR_INITIALS);
- PROP_TO_STRING(PR_SURNAME);
- PROP_TO_STRING(PR_DISPLAY_NAME);
- PROP_TO_STRING(PR_ACCOUNT);
- PROP_TO_STRING(PR_STREET_ADDRESS);
- PROP_TO_STRING(PR_LOCALITY);
- PROP_TO_STRING(PR_STATE_OR_PROVINCE);
- PROP_TO_STRING(PR_POSTAL_CODE);
- PROP_TO_STRING(PR_COUNTRY);
- PROP_TO_STRING(PR_TITLE);
- PROP_TO_STRING(PR_COMPANY_NAME);
- PROP_TO_STRING(PR_DEPARTMENT_NAME);
- PROP_TO_STRING(PR_OFFICE_LOCATION);
- PROP_TO_STRING(PR_ASSISTANT);
- PROP_TO_STRING(PR_BUSINESS_TELEPHONE_NUMBER);
- PROP_TO_STRING(PR_BUSINESS2_TELEPHONE_NUMBER);
- PROP_TO_STRING(PR_BUSINESS_FAX_NUMBER);
- PROP_TO_STRING(PR_HOME_TELEPHONE_NUMBER);
- PROP_TO_STRING(PR_HOME2_TELEPHONE_NUMBER);
- PROP_TO_STRING(PR_MOBILE_TELEPHONE_NUMBER);
- PROP_TO_STRING(PR_PAGER_TELEPHONE_NUMBER);
- PROP_TO_STRING(PR_PRIMARY_FAX_NUMBER);
- PROP_TO_STRING(PR_COMMENT);
- PROP_TO_STRING(PR_EMS_AB_MANAGER);
- PROP_TO_STRING(PR_EMS_AB_REPORTS);
- PROP_TO_STRING(PR_EMS_AB_IS_MEMBER_OF_DL);
- PROP_TO_STRING(PR_EMS_AB_PROXY_ADDRESSES);
- PROP_TO_STRING(PR_EMS_AB_OWNER);
- PROP_TO_STRING(PR_EMS_AB_MEMBER);
- PROP_TO_STRING(PR_EMS_AB_X509_CERT);
- PROP_TO_STRING(PR_EC_ENABLED_FEATURES);
- PROP_TO_STRING(PR_EC_DISABLED_FEATURES);
- PROP_TO_STRING(PR_EC_ARCHIVE_SERVERS);
- PROP_TO_STRING(PR_EC_ARCHIVE_COUPLINGS);
- default:
- return stringify(ulPropTag, true);
- }
- }
- /**
- * Prints a list of companies, enter or comma separated.
- *
- * @param[in] cCompanies Number of companies in lpECCompanies
- * @param[in] lpECCompanies Array of ECCompany structs
- * @param[in] bList true to list with comma's separation, otherwise enters are used.
- */
- static void print_companies(unsigned int cCompanies,
- const ECCOMPANY *lpECCompanies, bool bList)
- {
- for (unsigned int i = 0; i< cCompanies; ++i) {
- if (!bList)
- cout << ((i > 0) ? ", " : "");
- else
- cout << "\t";
- cout << (LPSTR)lpECCompanies[i].lpszCompanyname;
- if (bList)
- cout << endl;
- }
- }
- /**
- * Prints a list of groups, enter or comma separated.
- *
- * @param[in] cGroups Number of groups in lpECGroups
- * @param[in] lpECGroups Array of ECGroup structs
- * @param[in] bList true to list with comma's separation, otherwise enters are used.
- */
- static void print_groups(unsigned int cGroups, const ECGROUP *lpECGroups,
- bool bList)
- {
- for (unsigned int i = 0; i < cGroups; ++i) {
- if (!bList)
- cout << ((i > 0) ? ", " : "");
- else
- cout << "\t";
- cout << (LPSTR)lpECGroups[i].lpszGroupname;
- if (bList)
- cout << endl;
- }
- }
- /**
- * Prints a list of users, enter or comma separated.
- *
- * @param[in] cUsers Number of users in lpECUsers
- * @param[in] lpECUsers Array of ECUser structs
- * @param[in] bShowHomeServer true to print home server in multiserver environment if available
- */
- static void print_users(unsigned int cUsers, const ECUSER *lpECUsers,
- bool bShowHomeServer = false)
- {
- ConsoleTable ct(cUsers, bShowHomeServer?3:2);
- ct.SetHeader(0, "Username");
- ct.SetHeader(1, "Fullname");
- if (bShowHomeServer)
- ct.SetHeader(2, "Homeserver");
- for (unsigned int i = 0; i < cUsers; ++i) {
- ct.SetColumn(i, 0, (LPSTR)lpECUsers[i].lpszUsername);
- ct.SetColumn(i, 1, (LPSTR)lpECUsers[i].lpszFullName);
- if (!bShowHomeServer)
- continue;
- if (lpECUsers[i].lpszServername != NULL && *reinterpret_cast<LPSTR>(lpECUsers[i].lpszServername) != '\0')
- ct.SetColumn(i, 2, (LPSTR)lpECUsers[i].lpszServername);
- else
- // make sure we fill in all table parts. not using "<unknown>" tag,
- // since bShowHomeServer can be set to true even on non-multiserver environments
- ct.SetColumn(i, 2, string());
- }
- ct.PrintTable();
- }
- /**
- * Prints extra addressbook properties of an addressbook object, if present.
- *
- * @param[in] lpPropmap SPROPMAP struct, custom addressbook properties
- * @param[in] lpMVPropmap MVSPROPMAP struct, custom addressbook multi-valued properties
- * @param[in] group Indicate if the caller of this function is print_group_settings
- */
- static void print_extra_settings(const SPROPMAP *lpPropmap,
- const MVPROPMAP *lpMVPropmap, bool is_group = false)
- {
- unsigned int c = 0;
- if (!lpPropmap->cEntries && !lpMVPropmap->cEntries)
- return;
- ConsoleTable ct(lpPropmap->cEntries + lpMVPropmap->cEntries, 2);
- for (unsigned int i = 0; i < lpPropmap->cEntries; ++i) {
- ct.SetColumn(c, 0, getMapiPropertyString(lpPropmap->lpEntries[i].ulPropId));
- if (PROP_TYPE(lpPropmap->lpEntries[i].ulPropId) == PT_BINARY)
- ct.SetColumn(c, 1, stringify(strlen((LPSTR)lpPropmap->lpEntries[i].lpszValue)) + " bytes");
- else
- ct.SetColumn(c, 1, (LPSTR)lpPropmap->lpEntries[i].lpszValue);
- ++c;
- }
- for (unsigned int i = 0; i < lpMVPropmap->cEntries; ++i) {
- string strMVValues;
- if (is_group && (lpMVPropmap->lpEntries[i].ulPropId == PR_EC_ENABLED_FEATURES_A ||
- lpMVPropmap->lpEntries[i].ulPropId == PR_EC_DISABLED_FEATURES_A))
- continue;
- ct.SetColumn(c, 0, getMapiPropertyString(lpMVPropmap->lpEntries[i].ulPropId));
- if (PROP_TYPE(lpMVPropmap->lpEntries[i].ulPropId) == PT_MV_BINARY) {
- strMVValues = stringify(lpMVPropmap->lpEntries[i].cValues) + " values";
- } else {
- for (int j = 0; j < lpMVPropmap->lpEntries[i].cValues; ++j) {
- if (!strMVValues.empty())
- strMVValues += "; ";
- strMVValues += (LPSTR)lpMVPropmap->lpEntries[i].lpszValues[j];
- }
- }
- ct.SetColumn(c, 1, strMVValues);
- ++c;
- }
- if (c == 0)
- return;
- cout << "Mapped properties:" << endl;
- ct.PrintTable();
- }
- /**
- * Prints company details
- *
- * @param[in] lpECCompany ECCompany struct
- * @param[in] lpECAdministrator ECUser struct with the administrator of this company
- */
- static void print_company_settings(const ECCOMPANY *lpECCompany,
- const ECUSER *lpECAdministrator)
- {
- cout << "Companyname:\t\t" << (LPSTR)lpECCompany->lpszCompanyname << endl;
- cout << "Sysadmin:\t\t" << (LPSTR)lpECAdministrator->lpszUsername << endl;
- if (lpECCompany->lpszServername != NULL && *reinterpret_cast<LPSTR>(lpECCompany->lpszServername) != '\0')
- cout << "Home server:\t\t" << (LPSTR)lpECCompany->lpszServername << endl;
- cout << "Address book:\t\t" << (lpECCompany->ulIsABHidden ? "Hidden" : "Visible") << endl;
- print_extra_settings(&lpECCompany->sPropmap, &lpECCompany->sMVPropmap);
- }
- /**
- * Prints group details
- *
- * @param[in] lpECGroups ECGroup struct
- */
- static void print_group_settings(const ECGROUP *lpECGroup)
- {
- cout << "Groupname:\t\t" << (LPSTR)lpECGroup->lpszGroupname << endl;
- cout << "Fullname:\t\t" << (LPSTR)lpECGroup->lpszFullname << endl;
- cout << "Emailaddress:\t\t" << (LPSTR)lpECGroup->lpszFullEmail << endl;
- cout << "Address book:\t\t" << (lpECGroup->ulIsABHidden ? "Hidden" : "Visible") << endl;
- print_extra_settings(&lpECGroup->sPropmap, &lpECGroup->sMVPropmap, true);
- }
- /**
- * Converts an objectclass_t (common/ECDefs.h) to a string.
- *
- * @param[in] eClass Returns a user readable string for this objectclass
- * @return string
- */
- static string ClassToString(objectclass_t eClass)
- {
- switch (eClass) {
- case ACTIVE_USER:
- return string("User");
- case NONACTIVE_USER:
- return string("Shared store");
- case NONACTIVE_ROOM:
- return string("Room");
- case NONACTIVE_EQUIPMENT:
- return string("Equipment");
- case NONACTIVE_CONTACT:
- return string("Contact");
- case DISTLIST_GROUP:
- return string("Group");
- case DISTLIST_SECURITY:
- return string("Security group");
- case DISTLIST_DYNAMIC:
- return string("Dynamic group");
- case CONTAINER_COMPANY:
- return string("Company");
- case CONTAINER_ADDRESSLIST:
- return string("Addresslist");
- default:
- return string("Unknown");
- };
- }
- static int FileTimeToTimestamp(const FILETIME &ft, time_t& ts, char* buf, size_t size) {
- struct tm *tm;
- FileTimeToUnixTime(ft, &ts);
- if ((tm = localtime(&ts)) == NULL) {
- perror("localtime");
- return -1;
- }
- strftime(buf, size, "%F %T", tm);
- return 0;
- }
- static void adm_oof_status(const SPropValue *const prop)
- {
- char start_buf[64] = {'\0'}, end_buf[64] = {'\0'};
- time_t ts, now = time(NULL);
- /* o = out of office marked as switched on */
- bool o = prop[2].ulPropTag == PR_EC_OUTOFOFFICE && prop[2].Value.b;
- /* a = is it currently active with respect to time interval */
- bool a = o;
- if (prop[3].ulPropTag == PR_EC_OUTOFOFFICE_FROM) {
- if (FileTimeToTimestamp(prop[3].Value.ft, ts, start_buf, sizeof(start_buf)) == -1)
- return;
- a &= ts <= now;
- }
- if (prop[4].ulPropTag == PR_EC_OUTOFOFFICE_UNTIL) {
- if (FileTimeToTimestamp(prop[4].Value.ft, ts, end_buf, sizeof(start_buf)) == -1)
- return;
- a &= now <= ts;
- }
- if (!o) {
- printf("Out Of Office: disabled\n");
- return;
- }
- if (start_buf[0] == '\0' && end_buf[0] == '\0') {
- printf("Out Of Office: enabled\n");
- return;
- }
- printf("Out Of Office: ");
- if (start_buf[0] != '\0')
- printf("from %s ", start_buf);
- if (end_buf[0] != '\0')
- printf("until %s ", end_buf);
- printf("(currently %s)\n", a ? "active" : "inactive");
- }
- /**
- * Print user details
- *
- * @param[in] lpStore Store of the user
- * @param[in] lpECUser ECUser struct with user details
- * @param[in] bAutoAccept Meeting request settings of user
- * @param[in] bDeclineConflict Meeting request settings of user
- * @param[in] bDeclineRecurring Meeting request settings of user
- * @param[in] lstArchives List of attached archives
- */
- static void print_user_settings(IMsgStore *lpStore, const ECUSER *lpECUser,
- bool bAutoAccept, bool bDeclineConflict, bool bDeclineRecur,
- const ArchiveList &lstArchives, const ECUSERCLIENTUPDATESTATUS *lpECUCUS)
- {
- memory_ptr<SPropValue> lpProps;
- static constexpr const SizedSPropTagArray(5, sptaProps) =
- {5, {PR_LAST_LOGON_TIME, PR_LAST_LOGOFF_TIME,
- PR_EC_OUTOFOFFICE, PR_EC_OUTOFOFFICE_FROM,
- PR_EC_OUTOFOFFICE_UNTIL}};
- ULONG cValues = 0;
- if (lpStore)
- lpStore->GetProps(sptaProps, 0, &cValues, &~lpProps);
- cout << "Username:\t\t" << (LPSTR)lpECUser->lpszUsername << endl;
- cout << "Fullname:\t\t" << (LPSTR)lpECUser->lpszFullName << endl;
- cout << "Emailaddress:\t\t" << (LPSTR)lpECUser->lpszMailAddress << endl;
- cout << "Active:\t\t\t" << ((lpECUser->ulObjClass==ACTIVE_USER) ? "yes" : "no") << endl;
- if (lpECUser->ulObjClass != ACTIVE_USER)
- cout << "Non-active type:\t" << ClassToString((objectclass_t)lpECUser->ulObjClass) << endl;
- if (lpECUser->ulObjClass == NONACTIVE_ROOM || lpECUser->ulObjClass == NONACTIVE_EQUIPMENT)
- cout << "Resource capacity:\t" << lpECUser->ulCapacity << endl;
- cout << "Administrator:\t\t" << ((lpECUser->ulIsAdmin >= 1) ? "yes" : "no") << ((lpECUser->ulIsAdmin == 2) ? " (system)" : "") << endl;
- cout << "Address book:\t\t" << (lpECUser->ulIsABHidden ? "Hidden" : "Visible") << endl;
- cout << "Auto-accept meeting req:" << (bAutoAccept ? "yes" : "no") << endl;
- if (bAutoAccept) {
- cout << "Decline dbl meetingreq:\t" << (bDeclineConflict ? "yes" : "no") << endl;
- cout << "Decline recur meet.req:\t" << (bDeclineRecur ? "yes" : "no") << endl;
- }
- if (lpECUser->lpszServername != NULL && *reinterpret_cast<LPSTR>(lpECUser->lpszServername) != '\0')
- cout << "Home server:\t\t" << (LPSTR)lpECUser->lpszServername << endl;
- if (lpProps) {
- time_t logon = 0, logoff = 0;
- char d[64];
- adm_oof_status(lpProps);
- if(lpProps[0].ulPropTag == PR_LAST_LOGON_TIME)
- FileTimeToUnixTime(lpProps[0].Value.ft, &logon);
- if(lpProps[1].ulPropTag == PR_LAST_LOGOFF_TIME)
- FileTimeToUnixTime(lpProps[1].Value.ft, &logoff);
- if(logon) {
- strftime(d, sizeof(d), "%x %X", localtime(&logon));
- cout << "Last logon:\t\t" << d << std::endl;
- }
- if(logoff) {
- strftime(d, sizeof(d), "%x %X", localtime(&logoff));
- cout << "Last logoff:\t\t" << d << std::endl;
- }
- }
- print_extra_settings(&lpECUser->sPropmap, &lpECUser->sMVPropmap);
- if (!lstArchives.empty()) {
- cout << "Attached archives:\t" << lstArchives.size() << endl;
- for (const auto &arc : lstArchives) {
- cout << "\t" << arc.FolderName << " in " << arc.StoreName << " (" << arc.StoreGuid << ")";
- if (arc.Rights != ARCHIVE_RIGHTS_ABSENT) {
- if (arc.Rights == ROLE_OWNER)
- cout << " [Read Write]";
- else if (arc.Rights == ROLE_REVIEWER)
- cout << " [Read Only]";
- else
- cout << " [Modified: " << AclRightsToString(arc.Rights) << "]";
- }
- cout << endl;
- }
- }
- if (lpECUCUS == nullptr || lpECUCUS->ulTrackId <= 0)
- return;
- cout << "Client update Information:" << endl;
- cout << " Trackid:\t\t" << ((lpECUCUS->ulTrackId != 0 ) ? stringify(lpECUCUS->ulTrackId, true).c_str() : "-" ) << endl;
- cout << " Last update:\t\t" << ( (lpECUCUS->tUpdatetime>0) ? UnixtimeToString(lpECUCUS->tUpdatetime) : "-" ) << endl;
- cout << " From version:\t\t" << ( (lpECUCUS->lpszCurrentversion) ? (LPSTR)lpECUCUS->lpszCurrentversion : "-" ) << endl;
- cout << " To version:\t\t" << ( (lpECUCUS->lpszLatestversion) ? (LPSTR)lpECUCUS->lpszLatestversion : "-" ) << endl;
- cout << " Computername:\t\t" << ( (lpECUCUS->lpszComputername) ? (LPSTR)lpECUCUS->lpszComputername : "-" ) << endl;
- if (lpECUCUS->ulStatus == UPDATE_STATUS_SUCCESS)
- cout << " Update:\t\tSuccess" << endl;
- else if (lpECUCUS->ulStatus == UPDATE_STATUS_PENDING)
- cout << " Update:\t\tPending" << endl;
- else if (lpECUCUS->ulStatus == UPDATE_STATUS_UNKNOWN)
- cout << " Update: \t\tUnknown" << endl;
- else
- cout << " Update:\t\tFailed" << endl;
- }
- /**
- * Print archive store details on local server
- *
- * @param[in] lpSession MAPI session of the internal Kopano System adminstrator user
- * @param[in] lpECMsgStore The IECUnknown PR_EC_OBJECT pointer, used as IECServiceAdmin and IExchangeManageStore interface
- * @param[in] lpszName Name to resolve, using type in ulClass
- * @return MAPI error code
- */
- static HRESULT print_archive_details(LPMAPISESSION lpSession,
- IECUnknown *lpECMsgStore, const char *lpszName)
- {
- HRESULT hr;
- ECServiceAdminPtr ptrServiceAdmin;
- ULONG cbArchiveId = 0;
- EntryIdPtr ptrArchiveId;
- MsgStorePtr ptrArchive;
- SPropValuePtr ptrArchiveSize;
- hr = lpECMsgStore->QueryInterface(IID_IECServiceAdmin, &~ptrServiceAdmin);
- if (hr != hrSuccess) {
- cerr << "Unable to get admin interface." << endl;
- return hr;
- }
- hr = ptrServiceAdmin->GetArchiveStoreEntryID((LPCTSTR)lpszName, NULL, 0, &cbArchiveId, &~ptrArchiveId);
- if (hr != hrSuccess) {
- cerr << "No archive found for user '" << lpszName << "'." << endl;
- return hr;
- }
- hr = lpSession->OpenMsgStore(0, cbArchiveId, ptrArchiveId, &ptrArchive.iid(), 0, &~ptrArchive);
- if (hr != hrSuccess) {
- cerr << "Unable to open archive." << endl;
- return hr;
- }
- hr = HrGetOneProp(ptrArchive, PR_MESSAGE_SIZE_EXTENDED, &~ptrArchiveSize);
- if (hr != hrSuccess) {
- cerr << "Unable to get archive store size." << endl;
- return hr;
- }
- cout << "Current store size:\t";
- cout << stringify_double((double)ptrArchiveSize->Value.li.QuadPart /1024.0 /1024.0, 2, true) << " MiB" << endl;
- return hrSuccess;
- }
- /**
- * Parse a server store entryid to client store entryid.
- *
- * This is a hack to open an orphan store. It will convert a server store entryid,
- * which doesn't include a server url and is not wrapped by the support object, to
- * a client side store entryid.
- *
- * @param[in] lpServerUrl ServerURL for open the orphan store.
- * @param[in] cbEntryID Size of the unwrapped orphan store entryid.
- * @param[in] lpEntryID unwrapped orphan store entryid without server URL.
- * @param[out] lpcbEntryID Size of the wrapped orphan store entryid.
- * @param[out] lppEntryID Pointer to the wrapped entryid from the orphan store entryid.
- */
- static HRESULT CreateOrphanStoreEntryID(const char *lpServerUrl,
- ULONG cbEntryID, LPENTRYID lpEntryID, ULONG *lpcbEntryID,
- LPENTRYID *lppEntryID)
- {
- HRESULT hr = hrSuccess;
- ULONG cbNewEntryID = 0;
- memory_ptr<ENTRYID> lpNewEntryID;
- ULONG cbServerURL = 0;
- if (lpServerUrl == nullptr || lpEntryID == nullptr ||
- lpcbEntryID == nullptr || lppEntryID == nullptr)
- return MAPI_E_INVALID_PARAMETER;
- cbServerURL = strlen(lpServerUrl);
- cbNewEntryID = cbEntryID + cbServerURL;
- hr = MAPIAllocateBuffer(cbNewEntryID, &~lpNewEntryID);
- if (hr != hrSuccess)
- return hr;
- memcpy(lpNewEntryID, lpEntryID, cbEntryID);
- memcpy(reinterpret_cast<unsigned char *>(lpNewEntryID.get()) + cbEntryID - 4, lpServerUrl, cbServerURL + 4);
- return WrapStoreEntryID(0, (LPTSTR)"zarafa6client.dll", cbNewEntryID,
- lpNewEntryID, lpcbEntryID, lppEntryID);
- }
- /**
- * Get the information for an orphan store
- *
- * @param[in] lpServiceAdmin Pointer to the service admin.
- * @param[in] lpStoreGuid Indentifier to the orphan store.
- * @param[in] lpServerUrl ServerURL for open the orphan store.
- * @param[out] strUsername A guess of the user name belongs to the orphan store.
- * @param[out] strCompanyName Company name belongs to the orphan store.
- * @param[out] lpcbEntryID The orphan store entryid size.
- * @param[out] lppEntryID Pointer to the entry of the orphan store.
- */
- static HRESULT GetOrphanStoreInfo(IECServiceAdmin *lpServiceAdmin,
- GUID *lpStoreGuid, const char *lpServerUrl, wstring &strUsername,
- wstring &strCompanyName, ULONG *lpcbEntryID, LPENTRYID *lppEntryID)
- {
- HRESULT hr;
- MAPITablePtr ptrTable;
- SRowSetPtr ptrRowSet;
- SPropValue sStoreGuid;
- static constexpr const SizedSSortOrderSet(1, tableSort) =
- { 1, 0, 0,
- {
- { PR_EC_STOREGUID, TABLE_SORT_ASCEND },
- }
- };
- hr = lpServiceAdmin->OpenUserStoresTable(MAPI_UNICODE, &~ptrTable);
- if (hr != hrSuccess)
- return hr;
- hr = ptrTable->SortTable(tableSort, 0);
- if (hr != hrSuccess)
- return hr;
- sStoreGuid.ulPropTag = PR_EC_STOREGUID;
- sStoreGuid.Value.bin.cb = sizeof(GUID);
- sStoreGuid.Value.bin.lpb = (BYTE*)lpStoreGuid;
- hr = ECPropertyRestriction(RELOP_EQ, PR_EC_STOREGUID, &sStoreGuid, ECRestriction::Cheap)
- .FindRowIn(ptrTable, BOOKMARK_BEGINNING, 0);
- if (hr != hrSuccess)
- return hr;
- hr = ptrTable->QueryRows(1, 0, &ptrRowSet);
- if (hr != hrSuccess)
- return hr;
- if (ptrRowSet.empty())
- return MAPI_E_NOT_FOUND;
- auto lpsName = PCpropFindProp(ptrRowSet[0].lpProps, ptrRowSet[0].cValues, PR_DISPLAY_NAME_W);
- if (lpsName != nullptr)
- strUsername = lpsName->Value.lpszW;
- lpsName = PCpropFindProp(ptrRowSet[0].lpProps, ptrRowSet[0].cValues, PR_EC_COMPANY_NAME_W);
- if (lpsName != nullptr)
- strCompanyName = lpsName->Value.lpszW;
- auto lpsPropEntryId = PCpropFindProp(ptrRowSet[0].lpProps, ptrRowSet[0].cValues, PR_STORE_ENTRYID);
- if (lpsPropEntryId == NULL)
- return MAPI_E_NOT_FOUND;
- return CreateOrphanStoreEntryID(lpServerUrl,lpsPropEntryId->Value.bin.cb,
- reinterpret_cast<LPENTRYID>(lpsPropEntryId->Value.bin.lpb),
- lpcbEntryID, lppEntryID);
- }
- /**
- * Open/create deleted stores folder in the public store.
- *
- * Open the deleted admin folder in a public store, if the folder not exist it will create the folder.
- * First it creates a folder 'Admin' in the top-level tree (IPM_SUBTREE). The permissions on the folder
- * are set to 'everyone' can not read the folder except an admin. A second folder 'Deleted stores' will
- * create without permissions because the inheritance of the permissions.
- *
- * @param[in] lpPublicStore Public store to open or create the 'deleted stores' folder
- * @param[out] Pointer to a pointer of folder 'Deleted stores'.
- */
- static HRESULT OpenDeletedStoresFolder(LPMDB lpPublicStore,
- LPMAPIFOLDER *lppFolderStores)
- {
- HRESULT hr = hrSuccess;
- object_ptr<IMAPIFolder> lpFolderSubTree, lpFolderAdmin, lpFolderDeletedStores;
- memory_ptr<SPropValue> lpsPropSubTree, lpPropValue, lpsPropMDB;
- ULONG ulObjType;
- IECUnknown *lpECFolder = NULL; // non reference
- ECPERMISSION sPermission = {0};
- object_ptr<IECSecurity> lpSecurity;
- ULONG ulPropTagSubtree = 0;
- if (lpPublicStore == nullptr || lppFolderStores == nullptr)
- return MAPI_E_INVALID_PARAMETER;
- hr = HrGetOneProp(lpPublicStore, PR_MDB_PROVIDER, &~lpsPropMDB);
- if (hr != hrSuccess)
- return hr;
- // Workaround for companies, because a company is a delegate store!
- if (lpsPropMDB->Value.bin.cb == sizeof(MAPIUID) && memcmp(lpsPropMDB->Value.bin.lpb, &KOPANO_STORE_PUBLIC_GUID, sizeof(MAPIUID)) == 0)
- ulPropTagSubtree = PR_IPM_PUBLIC_FOLDERS_ENTRYID;
- else
- ulPropTagSubtree = PR_IPM_SUBTREE_ENTRYID;
- // Open IPM_subtree
- hr = HrGetOneProp(lpPublicStore, ulPropTagSubtree, &~lpsPropSubTree);
- if (hr != hrSuccess)
- return hr;
- hr = lpPublicStore->OpenEntry(lpsPropSubTree->Value.bin.cb, reinterpret_cast<ENTRYID *>(lpsPropSubTree->Value.bin.lpb), nullptr, MAPI_MODIFY, &ulObjType, &~lpFolderSubTree);
- if (hr != hrSuccess)
- return hr;
- // Create/open folder Admin
- hr = lpFolderSubTree->CreateFolder(FOLDER_GENERIC, (LPTSTR)"Admin", nullptr, nullptr, 0, &~lpFolderAdmin);
- if (hr == hrSuccess) {
- // Set permissions
- hr = HrGetOneProp(lpFolderAdmin, PR_EC_OBJECT, &~lpPropValue);
- if(hr != hrSuccess)
- return hr;
- lpECFolder = reinterpret_cast<IECUnknown *>(lpPropValue->Value.lpszA);
- hr = lpECFolder->QueryInterface(IID_IECSecurity, &~lpSecurity);
- if (hr != hrSuccess)
- return hr;
- sPermission.ulRights = 0;// No rights, only for admin
- sPermission.sUserId.lpb = g_lpEveryoneEid; // group everyone
- sPermission.sUserId.cb = g_cbEveryoneEid;
- sPermission.ulState = RIGHT_NEW|RIGHT_AUTOUPDATE_DENIED;
- sPermission.ulType = ACCESS_TYPE_GRANT;
- hr = lpSecurity->SetPermissionRules(1, &sPermission);
- } else if (hr == MAPI_E_COLLISION) {
- hr = lpFolderSubTree->CreateFolder(FOLDER_GENERIC, (LPTSTR)"Admin", nullptr, nullptr, OPEN_IF_EXISTS, &~lpFolderAdmin);
- }
- if (hr != hrSuccess)
- return hr;
- // Create/open folder Deleted Stores
- hr = lpFolderAdmin->CreateFolder(FOLDER_GENERIC, (LPTSTR)"Deleted stores", nullptr, nullptr, OPEN_IF_EXISTS, &~lpFolderDeletedStores);
- if (hr != hrSuccess)
- return hr;
- return lpFolderDeletedStores->QueryInterface(IID_IMAPIFolder, (void**)lppFolderStores);
- }
- /**
- * Get the public store
- *
- * Get the public store from a company or just the default public store. If a company name is given
- * it will try to open the companies store. if it fails it won't fall back to the default store.
- *
- * @param[in] lpSession Pointer to a mapi session.
- * @param[in] lpMsgStore Pointer to a random store to open the ExchangeManageStore object.
- * @param[in] strCompanyname name whose belongs the public store. If empty it opens the default public store.
- * @param[out] lppPublicStore Pointer to the public store
- */
- static HRESULT GetPublicStore(LPMAPISESSION lpSession, LPMDB lpMsgStore,
- const wstring &strCompanyname, LPMDB *lppPublicStore)
- {
- HRESULT hr = hrSuccess;
- ULONG cbEntryID = 0;
- memory_ptr<ENTRYID> lpEntryID;
- object_ptr<IExchangeManageStore> lpIEMS;
- if (strCompanyname.empty())
- return HrOpenECPublicStore(lpSession, lppPublicStore);
- hr = lpMsgStore->QueryInterface(IID_IExchangeManageStore, &~lpIEMS);
- if (hr != hrSuccess)
- return hr;
- hr = lpIEMS->CreateStoreEntryID((LPTSTR)L"", (LPTSTR)strCompanyname.c_str(), MAPI_UNICODE, &cbEntryID, &~lpEntryID);
- if (hr != hrSuccess)
- return hr;
- return lpSession->OpenMsgStore(0, cbEntryID, lpEntryID, &IID_IMsgStore, MDB_WRITE, lppPublicStore);
- }
- static const char *StoreTypeToString(ULONG ulStoreType)
- {
- switch (ulStoreType) {
- case ECSTORE_TYPE_PRIVATE:
- return "private";
- case ECSTORE_TYPE_ARCHIVE:
- return "archive";
- case ECSTORE_TYPE_PUBLIC:
- return "public";
- default:
- return "unknown";
- };
- }
- /**
- * List users without a store, and stores without a user.
- *
- * Gets a list of users and stores. Because of the sorting chosen,
- * stores without a user will be printed first, until the first user
- * without a store is found. Then those are printed, until the first
- * user with a store is found.
- *
- * @param[in] lpServiceAdmin Kopano Administrator service object
- * @result HRESULT MAPI Error code
- */
- static HRESULT list_orphans(IECServiceAdmin *lpServiceAdmin)
- {
- HRESULT hr = hrSuccess;
- ULONG i = 0;
- object_ptr<IMAPITable> lpTable;
- std::string strUsername;
- bool bHeader = true;
- ConsoleTable ct(50, 5);
- static constexpr const SizedSSortOrderSet(2, tableSort) =
- { 2, 0, 0,
- {
- { PR_EC_USERNAME, TABLE_SORT_ASCEND },
- { PR_EC_STOREGUID, TABLE_SORT_ASCEND },
- }
- };
- hr = lpServiceAdmin->OpenUserStoresTable(0, &~lpTable);
- if (hr != hrSuccess) {
- cerr << "Unable to open user/stores table" << endl;
- return hr;
- }
- hr = lpTable->SortTable(tableSort, 0);
- if (hr != hrSuccess) {
- cerr << "Unable to sort user/stores table" << endl;
- return hr;
- }
- ct.SetHeader(0, "Store guid");
- ct.SetHeader(1, "Guessed username");
- ct.SetHeader(2, "Last login");
- ct.SetHeader(3, "Store size");
- ct.SetHeader(4, "Store type");
- // Because of the sort, we start with these stores
- cout << "Stores without users:" << endl;
- while(TRUE) {
- rowset_ptr lpRowSet;
- hr = lpTable->QueryRows(50, 0, &~lpRowSet);
- if(hr != hrSuccess) {
- cerr << "Unable to load user/stores table" << endl;
- return hr;
- }
- if(lpRowSet->cRows == 0)
- break;
- for (i = 0; i < lpRowSet->cRows; ++i) {
- auto lpStoreGuid = PCpropFindProp(lpRowSet->aRow[i].lpProps, lpRowSet->aRow[i].cValues, PR_EC_STOREGUID);
- auto lpUserName = PCpropFindProp(lpRowSet->aRow[i].lpProps, lpRowSet->aRow[i].cValues, PR_EC_USERNAME_A);
- auto lpModTime = PCpropFindProp(lpRowSet->aRow[i].lpProps, lpRowSet->aRow[i].cValues, PR_LAST_MODIFICATION_TIME);
- auto lpStoreSize = PCpropFindProp(lpRowSet->aRow[i].lpProps, lpRowSet->aRow[i].cValues, PR_MESSAGE_SIZE_EXTENDED);
- auto lpStoreType = PCpropFindProp(lpRowSet->aRow[i].lpProps, lpRowSet->aRow[i].cValues, PR_EC_STORETYPE);
- if (lpStoreGuid && lpUserName)
- continue;
- if (!lpUserName) {
- // find "guessed" named
- lpUserName = PCpropFindProp(lpRowSet->aRow[i].lpProps, lpRowSet->aRow[i].cValues, PR_DISPLAY_NAME_A);
- if (lpUserName)
- strUsername = lpUserName->Value.lpszA;
- else
- strUsername = "<unknown>";
- } else {
- // we had all stores without users, now the users without stores
- if (bHeader) {
- ct.PrintTable();
- ct.Resize(50, 1);
- ct.SetHeader(0, "Username");
- cout << endl << "Users without stores:" << endl;
- bHeader = false;
- }
- strUsername = lpUserName->Value.lpszA;
- }
- if (lpStoreGuid == nullptr) {
- ct.AddColumn(0, strUsername);
- continue;
- }
- ct.AddColumn(0, bin2hex(lpStoreGuid->Value.bin.cb, lpStoreGuid->Value.bin.lpb));
- ct.AddColumn(1, strUsername);
- if (lpModTime)
- ct.AddColumn(2, FiletimeToString(lpModTime->Value.ft));
- else
- ct.AddColumn(2, "<unknown>");
- if (lpStoreSize)
- ct.AddColumn(3, str_storage(lpStoreSize->Value.li.QuadPart, false));
- else
- ct.AddColumn(3, "<unknown>");
- if (lpStoreType)
- ct.AddColumn(4, StoreTypeToString(lpStoreType->Value.ul));
- else
- ct.AddColumn(4, "<unknown>");
- }
- }
- ct.PrintTable();
- return hrSuccess;
- }
- static LPMVPROPMAPENTRY FindMVPropmapEntry(ECUSER *lpUser, ULONG ulPropTag)
- {
- for (unsigned i = 0; i < lpUser->sMVPropmap.cEntries; ++i)
- if (lpUser->sMVPropmap.lpEntries[i].ulPropId == ulPropTag)
- return &lpUser->sMVPropmap.lpEntries[i];
- return NULL;
- }
- /**
- * Print the defaults of any user object (user/group/company)
- *
- * Depending on the input ulClass, find the object on the server, and
- * print the details of the object if found.
- *
- * @param[in] lpSession MAPI session of the internal Kopano System adminstrator user
- * @param[in] lpECMsgStore The IECUnknown PR_EC_OBJECT pointer, used as IECServiceAdmin and IExchangeManageStore interface
- * @param[in] ulClass addressbook objectclass of input lpszName
- * @param[in] lpszName Name to resolve, using type in ulClass
- * @return MAPI error code
- */
- static HRESULT print_details(LPMAPISESSION lpSession, IECUnknown *lpECMsgStore,
- objectclass_t ulClass, const char *lpszName)
- {
- HRESULT hr = hrSuccess;
- memory_ptr<ECUSER> lpECUser;
- memory_ptr<ECGROUP> lpECGroup, lpECGroups;
- memory_ptr<ECCOMPANY> lpECCompany, lpECViews;
- memory_ptr<ECQUOTASTATUS> lpsQuotaStatus;
- memory_ptr<ECQUOTA> lpsQuota;
- memory_ptr<ECUSER> lpECUsers, lpECAdmins;
- ULONG cGroups = 0;
- ULONG cUsers = 0;
- ULONG cAdmins = 0;
- ULONG cViews = 0;
- ULONG cbEntryID = 0;
- memory_ptr<ENTRYID> lpEntryID;
- object_ptr<IMsgStore> lpStore;
- object_ptr<IExchangeManageStore> lpIEMS;
- object_ptr<IECServiceAdmin> lpServiceAdmin;
- bool bAutoAccept = false;
- bool bDeclineConflict = false;
- bool bDeclineRecurring = false;
- ULONG cbObjectId = 0;
- LPENTRYID lpObjectId = NULL;
- ArchiveManagePtr ptrArchiveManage;
- ArchiveList lstArchives;
- ECUSERCLIENTUPDATESTATUS *lpECUCUS = NULL;
- convert_context converter;
- hr = lpECMsgStore->QueryInterface(IID_IECServiceAdmin, &~lpServiceAdmin);
- if (hr != hrSuccess) {
- cerr << "Unable to get admin interface." << endl;
- return hr;
- }
- switch (ulClass) {
- case OBJECTCLASS_CONTAINER:
- case CONTAINER_COMPANY:
- hr = lpServiceAdmin->ResolveCompanyName((LPTSTR)lpszName, 0, &cbObjectId, &lpObjectId);
- if (hr != hrSuccess) {
- cerr << "Unable to resolve company, " << getMapiCodeString(hr, lpszName) << endl;
- return hr;
- }
- hr = lpServiceAdmin->GetCompany(cbObjectId, lpObjectId, 0, &~lpECCompany);
- if (hr != hrSuccess) {
- cerr << "Unable to show company details, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- hr = lpECMsgStore->QueryInterface(IID_IExchangeManageStore, &~lpIEMS);
- if (hr != hrSuccess) {
- cerr << "Unable to get admin interface." << endl;
- return hr;
- }
- hr = lpIEMS->CreateStoreEntryID((LPTSTR)"", lpECCompany->lpszCompanyname, 0, &cbEntryID, &~lpEntryID);
- if (hr != hrSuccess) {
- cerr << "Unable to get company store entry id. Company possibly has no store." << endl;
- return hr;
- }
- hr = lpSession->OpenMsgStore(0, cbEntryID, lpEntryID, &IID_IMsgStore, MDB_WRITE, &~lpStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open company store." << endl;
- return hr;
- }
- hr = lpServiceAdmin->GetUser(lpECCompany->sAdministrator.cb, (LPENTRYID)lpECCompany->sAdministrator.lpb, 0, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Unable to resolve company administrator, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- hr = lpServiceAdmin->GetRemoteAdminList(cbObjectId, lpObjectId, 0, &cAdmins, &~lpECAdmins);
- if (hr != hrSuccess) {
- cerr << "Unable to display remote-admin list, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- }
- hr = lpServiceAdmin->GetRemoteViewList(cbObjectId, lpObjectId, 0, &cViews, &~lpECViews);
- if (hr != hrSuccess) {
- cerr << "Unable to display remote-view list, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- }
- print_company_settings(lpECCompany, lpECUser);
- break;
- case OBJECTCLASS_DISTLIST:
- case DISTLIST_GROUP:
- case DISTLIST_SECURITY:
- case DISTLIST_DYNAMIC:
- hr = lpServiceAdmin->ResolveGroupName((LPTSTR)lpszName, 0, &cbObjectId, &lpObjectId);
- if (hr != hrSuccess) {
- cerr << "Unable to resolve group, " << getMapiCodeString(hr, lpszName) << endl;
- return hr;
- }
- hr = lpServiceAdmin->GetGroup(cbObjectId, lpObjectId, 0, &~lpECGroup);
- if (hr != hrSuccess) {
- cerr << "Unable to show group details, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- hr = lpServiceAdmin->GetUserListOfGroup(cbObjectId, lpObjectId, 0, &cUsers, &~lpECUsers);
- if (hr != hrSuccess) {
- cerr << "Unable to request users for group, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- }
- print_group_settings(lpECGroup);
- break;
- case OBJECTCLASS_USER:
- case ACTIVE_USER:
- case NONACTIVE_USER:
- case NONACTIVE_ROOM:
- case NONACTIVE_EQUIPMENT:
- case NONACTIVE_CONTACT:
- default:
- hr = lpServiceAdmin->ResolveUserName((LPTSTR)lpszName, 0, &cbObjectId, &lpObjectId);
- if (hr != hrSuccess) {
- cerr << "Unable to resolve user, " << getMapiCodeString(hr, lpszName) << endl;
- return hr;
- }
- hr = lpServiceAdmin->GetUser(cbObjectId, lpObjectId, 0, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Unable to show user details, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- hr = lpECMsgStore->QueryInterface(IID_IExchangeManageStore, &~lpIEMS);
- if (hr != hrSuccess) {
- cerr << "Unable to get admin interface." << endl;
- return hr;
- }
- hr = lpIEMS->CreateStoreEntryID((LPTSTR)"", lpECUser->lpszUsername, 0, &cbEntryID, &~lpEntryID);
- if (hr != hrSuccess) {
- cerr << "WARNING: Unable to get user store entry id. User possibly has no store." << endl << endl;
- lpStore.reset();
- forcedExitCode = 1;
- }
- else {
- hr = lpSession->OpenMsgStore(0, cbEntryID, lpEntryID, &IID_IMsgStore, MDB_WRITE, &~lpStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open user store." << endl;
- return hr;
- }
- GetAutoAcceptSettings(lpStore, &bAutoAccept, &bDeclineConflict, &bDeclineRecurring);
- /* Ignore return value */
- }
- hr = lpServiceAdmin->GetGroupListOfUser(cbObjectId, lpObjectId, 0, &cGroups, &~lpECGroups);
- if (hr != hrSuccess) {
- cerr << "Unable to request groups for user, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- }
- hr = ArchiveManage::Create(lpSession, NULL, converter.convert_to<LPTSTR>(lpszName), &ptrArchiveManage);
- if (hr != hrSuccess) {
- if (hr != MAPI_E_NOT_FOUND)
- cerr << "Error while obtaining archive details, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- }
- if (ptrArchiveManage.get() != NULL) {
- hr = ptrArchiveManage->ListArchives(&lstArchives, "Root Folder");
- if (hr != hrSuccess) {
- cerr << "Error while obtaining archive list, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- }
- }
- hr = lpServiceAdmin->GetUserClientUpdateStatus(cbObjectId, lpObjectId, 0, &lpECUCUS);
- if (hr != hrSuccess) {
- cerr << "Unable to get auto update status: " <<
- GetMAPIErrorMessage(hr) << " (" <<
- stringify(hr, true) << ")" << endl;
- hr = hrSuccess;
- }
- print_user_settings(lpStore, lpECUser, bAutoAccept, bDeclineConflict, bDeclineRecurring, lstArchives, lpECUCUS);
- break;
- }
- /* Group quota is not completely implemented at this time on the server... */
- if (ulClass != DISTLIST_GROUP) {
- hr = lpServiceAdmin->GetQuota(cbObjectId, lpObjectId, false, &~lpsQuota);
- if (hr != hrSuccess) {
- cerr << "Unable to show object quota, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- } else {
- hr = Util::HrGetQuotaStatus(lpStore, lpsQuota, &~lpsQuotaStatus);
- if (hr != hrSuccess) {
- cerr << "Unable to show object quota information, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* Don't make error fatal */
- } else
- print_quota(lpsQuota, lpsQuotaStatus, (ulClass == CONTAINER_COMPANY));
- }
- }
- if (ulClass == CONTAINER_COMPANY) {
- hr = lpServiceAdmin->GetQuota(cbObjectId, lpObjectId, true, &~lpsQuota);
- if (hr != hrSuccess) {
- cerr << "Unable to get user default quota for company, " << getMapiCodeString(hr) << endl;
- hr = hrSuccess; /* not fatal */
- } else
- print_quota(lpsQuota, NULL, false);
- }
- if (cUsers) {
- cout << "Users (" << cUsers << "):" << endl;
- print_users(cUsers, lpECUsers, true);
- cout << endl;
- }
- if (cGroups) {
- cout << "Groups (" << cGroups << "):" << endl;
- print_groups(cGroups, lpECGroups, true);
- cout << endl;
- }
- if (cAdmins) {
- cout << "Remote admins (" << cAdmins << "):" << endl;
- print_users(cAdmins, lpECAdmins);
- cout << endl;
- }
- if (cViews) {
- cout << "Remote viewers (" << cViews << "):" << endl;
- print_companies(cViews, lpECViews, true);
- cout << endl;
- }
- if (lpECUser == nullptr)
- return hr;
- LPMVPROPMAPENTRY lpArchiveServers = FindMVPropmapEntry(lpECUser, PR_EC_ARCHIVE_SERVERS_A);
- if (lpArchiveServers == nullptr || lpArchiveServers->cValues == 0)
- return hr;
- MsgStorePtr ptrAdminStore;
- hr = lpECMsgStore->QueryInterface(IID_IMsgStore, &~ptrAdminStore);
- if (hr != hrSuccess)
- return hr;
- for (int i = 0; i < lpArchiveServers->cValues; ++i) {
- MsgStorePtr ptrRemoteAdminStore;
- SPropValuePtr ptrPropValue;
- IECUnknown *lpECRemoteAdminStore = NULL;
- HRESULT hrTmp;
- cout << "Archive details on node '" << (LPSTR)lpArchiveServers->lpszValues[i] << "':" << endl;
- hrTmp = HrGetRemoteAdminStore(lpSession, ptrAdminStore, lpArchiveServers->lpszValues[i], 0, &~ptrRemoteAdminStore);
- if (FAILED(hrTmp)) {
- cerr << "Unable to access node '" <<
- (LPSTR)lpArchiveServers->lpszValues[i] <<
- "': " << GetMAPIErrorMessage(hr) <<
- "(" << stringify(hrTmp, true) <<
- ")" << endl;
- continue;
- }
- hr = HrGetOneProp(ptrRemoteAdminStore, PR_EC_OBJECT, &~ptrPropValue);
- if (hr != hrSuccess || !ptrPropValue || !ptrPropValue->Value.lpszA) {
- cerr << "Admin object not found." << endl;
- return hr;
- }
- lpECRemoteAdminStore = reinterpret_cast<IECUnknown *>(ptrPropValue->Value.lpszA);
- print_archive_details(lpSession, lpECRemoteAdminStore, lpszName);
- cout << endl;
- }
- return hr;
- }
- /**
- * Print a list of all users within a company.
- *
- * @param[in] lpServiceAdmin IECServiceAdmin on SYSTEM store
- * @param[in] lpCompany The company to request users from. NULL EntryID in a non-hosted environment
- * @return MAPI Error code
- */
- static HRESULT ListUsers(IECServiceAdmin *lpServiceAdmin, ECCOMPANY *lpCompany)
- {
- HRESULT hr = hrSuccess;
- memory_ptr<ECUSER> lpECUsers;
- ULONG cUsers = 0;
- hr = lpServiceAdmin->GetUserList(lpCompany->sCompanyId.cb, (LPENTRYID)lpCompany->sCompanyId.lpb, 0, &cUsers, &~lpECUsers);
- if (hr != hrSuccess) {
- cerr << "Unable to list users, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- cout << "User list for " << (LPSTR)lpCompany->lpszCompanyname << "("<< cUsers <<"):" << endl;
- print_users(cUsers, lpECUsers, true);
- cout << endl;
- return hrSuccess;
- }
- /**
- * Print a list of all groups within a company.
- *
- * @param[in] lpServiceAdmin IECServiceAdmin on SYSTEM store
- * @param[in] lpCompany The company to request users from. NULL EntryID in a non-hosted environment
- * @return HRESULT MAPI Error code
- */
- static HRESULT ListGroups(IECServiceAdmin *lpServiceAdmin,
- ECCOMPANY *lpCompany)
- {
- HRESULT hr = hrSuccess;
- memory_ptr<ECGROUP> lpECGroups;
- ULONG cGroups = 0;
- hr = lpServiceAdmin->GetGroupList(lpCompany->sCompanyId.cb, (LPENTRYID)lpCompany->sCompanyId.lpb, 0, &cGroups, &~lpECGroups);
- if (hr != hrSuccess) {
- cerr << "Unable to list groups, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- cout << "Group list for " << (LPSTR)lpCompany->lpszCompanyname << "("<< cGroups <<"):" << endl;
- cout << "\t" << "groupname" << "" << endl;
- cout << "\t-------------------------------------" << endl;
- print_groups(cGroups, lpECGroups, true);
- cout << endl;
- return hrSuccess;
- }
- /**
- * Call the sync function to flush user cache on the server.
- *
- * @param[in] lpServiceAdmin IECServiceAdmin on SYSTEM store
- * @return HRESULT MAPI Error code
- */
- static HRESULT SyncUsers(IECServiceAdmin *lpServiceAdmin)
- {
- HRESULT hr;
- // we don't sync one company, since the complete cache is flushed in the server
- hr = lpServiceAdmin->SyncUsers(0, NULL);
- if (hr != hrSuccess)
- cerr << "User/group synchronization failed, " << getMapiCodeString(hr) << endl;
- return hr;
- }
- /**
- * Loop a function over one or a list of companies.
- *
- * @param[in] lpServiceAdmin IECServiceAdmin on SYSTEM store
- * @param[in] lpszCompanyname Only work on given company. NULL to work on all companies available.
- * @param[in] lpWork Function to call given any company found in this function.
- * @return HRESULT MAPI Error code
- */
- static HRESULT ForEachCompany(IECServiceAdmin *lpServiceAdmin,
- const char *lpszCompanyName,
- HRESULT (*lpWork)(IECServiceAdmin *, ECCOMPANY *))
- {
- HRESULT hr = hrSuccess;
- ULONG cbCompanyId = 0;
- memory_ptr<ENTRYID> lpCompanyId;
- ULONG cCompanies = 0;
- ECCOMPANY *lpECCompanies = NULL;
- memory_ptr<ECCOMPANY> lpECCompaniesAlloc;
- ECCOMPANY sRootCompany = {{g_cbSystemEid, g_lpSystemEid}, (LPTSTR)"Default", NULL, {0, NULL}};
- if (lpszCompanyName) {
- hr = lpServiceAdmin->ResolveCompanyName((LPTSTR)lpszCompanyName, 0, &cbCompanyId, &~lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to resolve company name, " << getMapiCodeString(hr, lpszCompanyName) << endl;
- return hr;
- }
- cCompanies = 1;
- sRootCompany.sCompanyId.cb = cbCompanyId;
- sRootCompany.sCompanyId.lpb = reinterpret_cast<unsigned char *>(lpCompanyId.get());
- sRootCompany.lpszCompanyname = (LPTSTR)lpszCompanyName;
- lpECCompanies = &sRootCompany;
- } else {
- hr = lpServiceAdmin->GetCompanyList(0, &cCompanies, &~lpECCompaniesAlloc);
- if (hr != hrSuccess) {
- cCompanies = 1;
- lpECCompanies = &sRootCompany;
- hr = hrSuccess;
- } else {
- lpECCompanies = lpECCompaniesAlloc;
- }
- }
- if (cCompanies == 0) {
- cerr << "No companies found." << endl;
- return hr;
- }
- for (unsigned int i = 0; i < cCompanies; ++i) {
- hr = lpWork(lpServiceAdmin, &lpECCompanies[i]);
- if (hr != hrSuccess)
- return hr;
- }
- return hr;
- }
- static HRESULT ForceResyncFor(LPMAPISESSION lpSession, LPMDB lpAdminStore,
- const char *lpszAccount, const char *lpszHomeMDB)
- {
- HRESULT hr;
- ExchangeManageStorePtr ptrEMS;
- ULONG cbEntryID = 0;
- EntryIdPtr ptrEntryID;
- MsgStorePtr ptrUserStore;
- MAPIFolderPtr ptrRoot;
- SPropValuePtr ptrPropResyncID;
- ULONG ulType = 0;
- hr = lpAdminStore->QueryInterface(ptrEMS.iid(), &~ptrEMS);
- if (hr != hrSuccess)
- return hr;
- hr = ptrEMS->CreateStoreEntryID((LPTSTR)lpszHomeMDB, (LPTSTR)lpszAccount, 0, &cbEntryID, &~ptrEntryID);
- if (hr != hrSuccess)
- return hr;
- hr = lpSession->OpenMsgStore(0, cbEntryID, ptrEntryID, NULL, MDB_WRITE|MAPI_DEFERRED_ERRORS, &~ptrUserStore);
- if (hr != hrSuccess)
- return hr;
- hr = ptrUserStore->OpenEntry(0, nullptr, &ptrRoot.iid(), MAPI_MODIFY, &ulType, &~ptrRoot);
- if (hr != hrSuccess)
- return hr;
- hr = HrGetOneProp(ptrRoot, PR_EC_RESYNC_ID, &~ptrPropResyncID);
- if (hr == MAPI_E_NOT_FOUND) {
- SPropValue sPropResyncID;
- sPropResyncID.ulPropTag = PR_EC_RESYNC_ID;
- sPropResyncID.Value.ul = 1;
- return HrSetOneProp(ptrRoot, &sPropResyncID);
- } else if (hr != hrSuccess) {
- return hr;
- }
- ++ptrPropResyncID->Value.ul;
- return HrSetOneProp(ptrRoot, ptrPropResyncID);
- }
- static HRESULT ForceResyncAll(LPMAPISESSION lpSession, LPMDB lpAdminStore)
- {
- HRESULT hr = hrSuccess;
- AddrBookPtr ptrAdrBook;
- ABContainerPtr ptrABContainer;
- MAPITablePtr ptrTable;
- SRowSetPtr ptrRows;
- ULONG ulType = 0;
- bool bFail = false;
- static constexpr const SizedSPropTagArray(1, sGALProps) = {1, {PR_ENTRYID}};
- SPropValue sGALPropVal;
- static constexpr const SizedSPropTagArray(2, sContentsProps) =
- {2, {PR_ACCOUNT, PR_EMS_AB_HOME_MDB}};
- SPropValue sObjTypePropVal;
- SPropValue sDispTypePropVal;
- hr = lpSession->OpenAddressBook(0, &ptrAdrBook.iid(), AB_NO_DIALOG, &~ptrAdrBook);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrAdrBook->OpenEntry(0, nullptr, &ptrABContainer.iid(), 0, &ulType, &~ptrABContainer);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrABContainer->GetHierarchyTable(0, &~ptrTable);
- if (hr != hrSuccess)
- goto exit;
- sGALPropVal.ulPropTag = PR_AB_PROVIDER_ID;
- sGALPropVal.Value.bin.cb = sizeof(GUID);
- sGALPropVal.Value.bin.lpb = (LPBYTE)&MUIDECSAB;
- hr = ptrTable->SetColumns(sGALProps, TBL_BATCH);
- if (hr != hrSuccess)
- goto exit;
- hr = ECPropertyRestriction(RELOP_EQ, PR_AB_PROVIDER_ID, &sGALPropVal, ECRestriction::Cheap)
- .RestrictTable(ptrTable, TBL_BATCH);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrTable->QueryRows(1, 0, &ptrRows);
- if (hr != hrSuccess)
- goto exit;
- if (ptrRows.size() != 1 || ptrRows[0].lpProps[0].ulPropTag != PR_ENTRYID) {
- hr = MAPI_E_NOT_FOUND;
- goto exit;
- }
- hr = ptrAdrBook->OpenEntry(ptrRows[0].lpProps[0].Value.bin.cb, reinterpret_cast<ENTRYID *>(ptrRows[0].lpProps[0].Value.bin.lpb),
- &ptrABContainer.iid(), MAPI_BEST_ACCESS, &ulType, &~ptrABContainer);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrABContainer->GetContentsTable(0, &~ptrTable);
- if (hr != hrSuccess)
- goto exit;
- sObjTypePropVal.ulPropTag = PR_OBJECT_TYPE;
- sObjTypePropVal.Value.l = MAPI_MAILUSER;
- sDispTypePropVal.ulPropTag = PR_DISPLAY_TYPE;
- sDispTypePropVal.Value.l = DT_MAILUSER;
- hr = ptrTable->SetColumns(sContentsProps, TBL_BATCH);
- if (hr != hrSuccess)
- goto exit;
- hr = ECAndRestriction(
- ECPropertyRestriction(RELOP_EQ, PR_OBJECT_TYPE, &sObjTypePropVal, ECRestriction::Cheap) +
- ECPropertyRestriction(RELOP_EQ, PR_DISPLAY_TYPE, &sDispTypePropVal, ECRestriction::Cheap)
- ).RestrictTable(ptrTable, TBL_BATCH);
- if (hr != hrSuccess)
- goto exit;
- while (true) {
- hr = ptrTable->QueryRows(50, 0, &ptrRows);
- if (hr != hrSuccess)
- goto exit;
- if (ptrRows.empty())
- break;
- for (SRowSetPtr::size_type i = 0; i < ptrRows.size(); ++i) {
- if (PROP_TYPE(ptrRows[i].lpProps[0].ulPropTag) == PT_ERROR ||
- PROP_TYPE(ptrRows[i].lpProps[1].ulPropTag) == PT_ERROR)
- {
- cerr << "Ignoring incomplete entry." << endl;
- continue;
- }
- hr = ForceResyncFor(lpSession, lpAdminStore, ptrRows[i].lpProps[0].Value.lpszA, ptrRows[i].lpProps[1].Value.lpszA);
- if (hr == hrSuccess)
- continue;
- cerr << "Failed to force resync for user " <<
- ptrRows[i].lpProps[0].Value.lpszA <<
- ": " << GetMAPIErrorMessage(hr) <<
- " (" << stringify(hr, true) << ")" <<
- endl;
- bFail = true;
- }
- }
- exit:
- if (!FAILED(hr) && bFail)
- hr = MAPI_W_ERRORS_RETURNED;
- return hr;
- }
- static HRESULT ForceResync(LPMAPISESSION lpSession, LPMDB lpAdminStore,
- const list<string> &lstUsernames)
- {
- HRESULT hr = hrSuccess;
- bool bFail = false;
- for (const auto &user : lstUsernames) {
- hr = ForceResyncFor(lpSession, lpAdminStore, user.c_str(), NULL);
- if (hr == hrSuccess)
- continue;
- cerr << "Failed to force resync for user " <<
- user << ": " << GetMAPIErrorMessage(hr) <<
- " (" << stringify(hr, true) << ")" << endl;
- bFail = true;
- }
- if (!FAILED(hr) && bFail)
- hr = MAPI_W_ERRORS_RETURNED;
- return hr;
- }
- static HRESULT DisplayUserCount(LPMDB lpAdminStore)
- {
- HRESULT hr;
- MAPITablePtr ptrSystemTable;
- SPropValue sPropDisplayName;
- SRowSetPtr ptrRows;
- ULONG ulLicensedUsers = (ULONG)-1; //!< active users allowed by license
- ULONG ulActiveUsers = (ULONG)-1; //!< used active users
- ULONG ulNonActiveTotal = (ULONG)-1; //!< used non-active users
- ULONG ulNonActiveUsers = (ULONG)-1; //!< used sharedstores, subset of used non-active users
- ULONG ulRooms = (ULONG)-1; //!< used rooms, subset of used non-active users
- ULONG ulEquipment = (ULONG)-1; //!< used equipment, subset of used non-active users
- ULONG ulMaxTotal = 0; //!< complete total of user objects allowed by license, aka ulNonActiveHigh limit
- ULONG ulNonActiveLow = 0; //!< atleast non-active users allowed
- ULONG ulActiveAsNonActive = 0; //!< non-active users taken from active count
- ConsoleTable ct(3, 4);
- ULONG ulExtraRow = 0;
- ULONG ulExtraRows = 0;
- static constexpr const SizedSPropTagArray(2, sptaStatsProps) =
- {2, {PR_DISPLAY_NAME_A, PR_EC_STATS_SYSTEM_VALUE}};
- enum {IDX_DISPLAY_NAME_A, IDX_EC_STATS_SYSTEM_VALUE};
- enum {COL_ALLOWED=1, COL_USED, COL_AVAILABLE};
- hr = lpAdminStore->OpenProperty(PR_EC_STATSTABLE_SYSTEM, &ptrSystemTable.iid(), 0, 0, &~ptrSystemTable);
- if (hr != hrSuccess)
- return hr;
- sPropDisplayName.ulPropTag = PR_DISPLAY_NAME_A;
- sPropDisplayName.Value.lpszA = const_cast<char *>("usercnt_");
- hr = ECContentRestriction(FL_PREFIX, PR_DISPLAY_NAME_A, &sPropDisplayName, ECRestriction::Cheap)
- .RestrictTable(ptrSystemTable, TBL_BATCH);
- if (hr != hrSuccess)
- return hr;
- hr = ptrSystemTable->SetColumns(sptaStatsProps, TBL_BATCH);
- if (hr != hrSuccess)
- return hr;
- hr = ptrSystemTable->QueryRows(0xffff, 0, &ptrRows);
- if (hr != hrSuccess)
- return hr;
- // We expect at least the first 3
- if (ptrRows.size() < 3)
- return MAPI_E_NOT_FOUND;
- for (SRowSetPtr::size_type i = 0; i < ptrRows.size(); ++i) {
- const char *lpszDisplayName = ptrRows[i].lpProps[IDX_DISPLAY_NAME_A].Value.lpszA;
- if (strcmp(lpszDisplayName, "usercnt_licensed") == 0)
- ulLicensedUsers = atoui(ptrRows[i].lpProps[IDX_EC_STATS_SYSTEM_VALUE].Value.lpszA);
- else if (strcmp(lpszDisplayName, "usercnt_active") == 0)
- ulActiveUsers = atoui(ptrRows[i].lpProps[IDX_EC_STATS_SYSTEM_VALUE].Value.lpszA);
- else if (strcmp(lpszDisplayName, "usercnt_nonactive") == 0)
- ulNonActiveTotal = atoui(ptrRows[i].lpProps[IDX_EC_STATS_SYSTEM_VALUE].Value.lpszA);
- else if (strcmp(lpszDisplayName, "usercnt_na_user") == 0)
- ulNonActiveUsers = atoui(ptrRows[i].lpProps[IDX_EC_STATS_SYSTEM_VALUE].Value.lpszA);
- else if (strcmp(lpszDisplayName, "usercnt_room") == 0)
- ulRooms = atoui(ptrRows[i].lpProps[IDX_EC_STATS_SYSTEM_VALUE].Value.lpszA);
- else if (strcmp(lpszDisplayName, "usercnt_equipment") == 0)
- ulEquipment = atoui(ptrRows[i].lpProps[IDX_EC_STATS_SYSTEM_VALUE].Value.lpszA);
- }
- if (ulLicensedUsers == static_cast<ULONG>(-1) ||
- ulActiveUsers == static_cast<ULONG>(-1) ||
- ulNonActiveTotal == static_cast<ULONG>(-1))
- return MAPI_E_NOT_FOUND;
- if (ulNonActiveUsers != (ULONG)-1)
- ++ulExtraRows;
- if (ulRooms != (ULONG)-1)
- ++ulExtraRows;
- if (ulEquipment != (ULONG)-1)
- ++ulExtraRows;
- if (ulExtraRows > 0)
- ct.Resize(3 + ulExtraRows, 4);
- ulMaxTotal = std::max(ulLicensedUsers + 25, (ulLicensedUsers *5)/ 2);
- ulNonActiveLow = ulMaxTotal - ulLicensedUsers;
- ulActiveAsNonActive = (ulNonActiveTotal > ulNonActiveLow) ? ulNonActiveTotal - ulNonActiveLow : 0;
- cout << "User counts:" << endl;
- ct.SetHeader(COL_ALLOWED, "Allowed");
- ct.SetHeader(COL_USED, "Used");
- ct.SetHeader(COL_AVAILABLE, "Available");
- ct.SetColumn(0, 0, "Active");
- ct.SetColumn(0, COL_USED, stringify(ulActiveUsers));
- if (ulLicensedUsers == 0) {
- ct.SetColumn(0, COL_ALLOWED, "no limit");
- ct.SetColumn(0, COL_AVAILABLE, "-");
- } else {
- ct.SetColumn(0, COL_ALLOWED, stringify(ulLicensedUsers));
- ct.SetColumn(0, COL_AVAILABLE, stringify(ulLicensedUsers - ulActiveUsers - ulActiveAsNonActive, false, true));
- }
- ct.SetColumn(1, 0, "Non-active");
- if (ulNonActiveTotal > ulNonActiveLow)
- ct.SetColumn(1, COL_USED, stringify(ulNonActiveLow) + " + " + stringify(ulActiveAsNonActive));
- else
- ct.SetColumn(1, COL_USED, stringify(ulNonActiveTotal));
- if (ulLicensedUsers == 0) {
- ct.SetColumn(1, COL_ALLOWED, "no limit");
- ct.SetColumn(1, COL_AVAILABLE, "-");
- } else {
- ct.SetColumn(1, COL_ALLOWED, stringify(ulMaxTotal - ulLicensedUsers, false, true));
- if (ulNonActiveTotal > ulNonActiveLow)
- ct.SetColumn(1, COL_AVAILABLE, "0 (+" + stringify(ulLicensedUsers - ulActiveUsers - ulActiveAsNonActive, false, true) + ")");
- else
- ct.SetColumn(1, COL_AVAILABLE, stringify(ulNonActiveLow - ulNonActiveTotal, false, true) +
- " (+" + stringify(ulLicensedUsers - ulActiveUsers - ulActiveAsNonActive, false, true) + ")");
- }
- if (ulNonActiveUsers != (ULONG)-1) {
- ct.SetColumn(2 + ulExtraRow, 0, " Users");
- ct.SetColumn(2 + ulExtraRow, COL_USED, stringify(ulNonActiveUsers));
- ++ulExtraRow;
- }
- if (ulRooms != (ULONG)-1) {
- ct.SetColumn(2 + ulExtraRow, 0, " Rooms");
- ct.SetColumn(2 + ulExtraRow, COL_USED, stringify(ulRooms));
- ++ulExtraRow;
- }
- if (ulEquipment != (ULONG)-1) {
- ct.SetColumn(2 + ulExtraRow, 0, " Equipment");
- ct.SetColumn(2 + ulExtraRow, COL_USED, stringify(ulEquipment));
- ++ulExtraRow;
- }
- ct.SetColumn(2 + ulExtraRows, 0, "Total");
- ct.SetColumn(2 + ulExtraRows, COL_USED, stringify(ulActiveUsers + ulNonActiveTotal));
- // available & allowed columns are too confusing in totals field.
- ct.SetColumn(2 + ulExtraRows, COL_AVAILABLE, string()); // add empty last column to make sure we print this row
- ct.PrintTable();
- return hrSuccess;
- }
- static HRESULT ResetFolderCount(LPMAPISESSION lpSession, LPMDB lpAdminStore,
- const char *lpszAccount)
- {
- HRESULT hr = hrSuccess;
- ExchangeManageStorePtr ptrEMS;
- ULONG cbEntryID;
- EntryIdPtr ptrEntryID;
- ULONG ulType = 0;
- MsgStorePtr ptrUserStore;
- MAPIFolderPtr ptrRoot;
- ECServiceAdminPtr ptrServiceAdmin;
- SPropValuePtr ptrPropEntryID;
- ULONG ulUpdates = 0;
- ULONG bFailures = false;
- ULONG ulTotalUpdates = 0;
- MAPITablePtr ptrTable;
- SRowSetPtr ptrRows;
- static constexpr const SizedSPropTagArray(2, sptaTableProps) =
- {2, {PR_DISPLAY_NAME_A, PR_ENTRYID}};
- enum {IDX_DISPLAY_NAME, IDX_ENTRYID};
- hr = lpAdminStore->QueryInterface(ptrEMS.iid(), &~ptrEMS);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrEMS->CreateStoreEntryID(NULL, (LPTSTR)lpszAccount, 0, &cbEntryID, &~ptrEntryID);
- if (hr != hrSuccess) {
- cerr << "Unable to resolve store for '" << lpszAccount << "'." << endl;
- goto exit;
- }
- hr = lpSession->OpenMsgStore(0, cbEntryID, ptrEntryID, nullptr, MDB_WRITE, &~ptrUserStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open store for '" << lpszAccount << "'." << endl;
- goto exit;
- }
- hr = ptrUserStore->QueryInterface(ptrServiceAdmin.iid(), &~ptrServiceAdmin);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrUserStore->OpenEntry(0, nullptr, &ptrRoot.iid(), 0, &ulType, &~ptrRoot);
- if (hr != hrSuccess)
- goto exit;
- hr = HrGetOneProp(ptrRoot, PR_ENTRYID, &~ptrPropEntryID);
- if (hr != hrSuccess)
- goto exit;
- hr = ptrServiceAdmin->ResetFolderCount(ptrPropEntryID->Value.bin.cb, (LPENTRYID)ptrPropEntryID->Value.bin.lpb, &ulUpdates);
- if (hr != hrSuccess) {
- cerr << "Failed to update counters in the root folder." << endl;
- bFailures = true;
- hr = hrSuccess;
- } else if (ulUpdates) {
- cerr << "Updated " << ulUpdates << " counters in the root folder." << endl;
- ulTotalUpdates += ulUpdates;
- }
- hr = ptrRoot->GetHierarchyTable(CONVENIENT_DEPTH, &~ptrTable);
- if (hr != hrSuccess)
- goto exit;
- hr = HrQueryAllRows(ptrTable, sptaTableProps, NULL, NULL, 0, &ptrRows);
- if (hr != hrSuccess)
- goto exit;
- for (SRowSetPtr::size_type i = 0; i < ptrRows.size(); ++i) {
- const SRow &row = ptrRows[i];
- const char* lpszName = "<Unknown>";
- if (PROP_TYPE(row.lpProps[IDX_DISPLAY_NAME].ulPropTag) != PT_ERROR)
- lpszName = row.lpProps[IDX_DISPLAY_NAME].Value.lpszA;
- hr = ptrServiceAdmin->ResetFolderCount(row.lpProps[IDX_ENTRYID].Value.bin.cb,
- (LPENTRYID)row.lpProps[IDX_ENTRYID].Value.bin.lpb,
- &ulUpdates);
- if (hr != hrSuccess) {
- cerr << "Failed to update counters in folder '" << lpszName << "'." << endl;
- bFailures = true;
- hr = hrSuccess;
- } else if (ulUpdates) {
- cerr << "Updated " << ulUpdates << " counters in folder '" << lpszName << "'." << endl;
- ulTotalUpdates += ulUpdates;
- }
- }
- if (ulTotalUpdates == 0)
- cerr << "No counters needed to be updated." << endl;
- exit:
- if (hr == hrSuccess && bFailures)
- hr = MAPI_W_ERRORS_RETURNED;
- return hr;
- }
- class InputValidator {
- public:
- bool Failed() const { return m_bFailure; }
- /**
- * Checks for 'invalid' input from the command prompt. Any
- * non-printable or contol ascii character is not allowed.
- *
- * @param[in] szInput command line input string
- *
- * @return validated input or NULL
- */
- char* operator()(char *szInput) {
- m_bFailure = true;
- wstring strInput;
- if (!szInput)
- return NULL;
- if (TryConvert(szInput, strInput) != hrSuccess)
- return NULL;
- for (auto c : strInput)
- if (!iswprint(c))
- return NULL;
- m_bFailure = false;
- return szInput;
- }
- private:
- bool m_bFailure = false;
- };
- // compare function for set<tstring, ltstr>, fixes default wchar_t compare, and makes it case-insensitive
- // used for PR_EC_*ABLED_FEATURES_A properties from TCHAR* strings in ECUSER struct
- struct lstr {
- bool operator()(const string &t1, const string &t2) const
- {
- return strcasecmp((char*)t1.c_str(), (char*)t2.c_str()) < 0;
- }
- };
- static HRESULT fillMVPropmap(ECUSER &sECUser, ULONG ulPropTag, int index,
- set<string, lstr> &sFeatures, void *lpBase)
- {
- HRESULT hr;
- sECUser.sMVPropmap.lpEntries[index].ulPropId = ulPropTag;
- sECUser.sMVPropmap.lpEntries[index].cValues = sFeatures.size();
- sECUser.sMVPropmap.lpEntries[index].lpszValues = NULL;
- if (sFeatures.size() == 0)
- return hrSuccess;
- hr = MAPIAllocateMore(sizeof(LPTSTR) * sFeatures.size(), lpBase, (void**)&sECUser.sMVPropmap.lpEntries[index].lpszValues);
- if (hr != hrSuccess) {
- cerr << "Memory error" << endl;
- return hr;
- }
- int n = 0;
- auto i = sFeatures.cbegin();
- // @note we store char* data in a LPTSTR (whcar_t by -DUNICODE) pointer.
- for (n = 0; i != sFeatures.cend(); ++i, ++n)
- sECUser.sMVPropmap.lpEntries[index].lpszValues[n] = (TCHAR*)i->c_str();
- return hrSuccess;
- }
- static void missing_quota(int hard, int warn, int soft)
- {
- if (hard == -1)
- cerr << " hard quota (--qh)";
- if (warn == -1)
- cerr << " warn quota (--qw)";
- if (soft == -1)
- cerr << " soft quota (--qs)";
- }
- int main(int argc, char* argv[])
- {
- HRESULT hr = hrSuccess;
- AutoMAPI mapiinit;
- object_ptr<IMAPISession> lpSession;
- object_ptr<IECUnknown> lpECMsgStore;
- object_ptr<IMsgStore> lpMsgStore;
- object_ptr<IECServiceAdmin> lpServiceAdmin;
- ULONG cbUserId = 0;
- memory_ptr<ENTRYID> lpUserId, lpSenderId, lpGroupId, lpCompanyId;
- memory_ptr<ENTRYID> lpSetCompanyId, lpEntryID, lpUnWrappedEntry;
- memory_ptr<ENTRYID> lpStoreId, lpRootId;
- ULONG cbGroupId = 0;
- ULONG cbSenderId = 0;
- ULONG cbCompanyId = 0;
- ULONG cbSetCompanyId = 0;
- ULONG ulDays = 0;
- memory_ptr<SPropValue> lpPropValue;
- ULONG cbGUID = 0;
- memory_ptr<GUID> lpGUID;
- ECUSER sECUser;
- memory_ptr<ECUSER> lpECUser, lpSenders;
- ULONG cbStoreId = 0;
- ULONG cbRootId = 0;
- ULONG cbUnWrappedEntry = 0;
- ECGROUP sECGroup;
- memory_ptr<ECGROUP> lpECGroups;
- ULONG cCompanies = 0;
- ULONG cUsers = 0;
- ECCOMPANY sECCompany;
- ECCOMPANY *lpECCompanies = NULL;
- memory_ptr<ECQUOTASTATUS> lpsQuotaStatus;
- memory_ptr<ECQUOTA> lpsQuota;
- memory_ptr<ECSVRNAMELIST> lpsServer;
- memory_ptr<ECSERVERLIST> lpServerDetails;
- ULONG cSenders = 0;
- objectclass_t ulClass = OBJECTCLASS_UNKNOWN;
- const char *detailstype = NULL;
- char *username = NULL;
- char *groupname = NULL;
- char *companyname = NULL;
- char *set_companyname = NULL;
- char *password = NULL;
- char *emailadr = NULL;
- char *fullname = NULL;
- char *new_username = NULL;
- char *storeguid = NULL;
- const char *path = NULL;
- char *lang = NULL;
- char *feature = NULL;
- char *node = NULL;
- bool bFeature = true;
- set<string, lstr> sEnabled;
- set<string, lstr> sDisabled;
- int quota = -1;
- long long quotahard = -1;
- long long quotasoft = -1;
- long long quotawarn = -1;
- int ud_quota = -1;
- long long ud_quotahard = -1;
- long long ud_quotasoft = -1;
- long long ud_quotawarn = -1;
- int isadmin = -1;
- int isnonactive = -1;
- int mr_accept = -1;
- int mr_decline_conflict = -1;
- int mr_decline_recurring = -1;
- char *sendas_user = NULL;
- int sendas_action = -1;
- modes mode = MODE_INVALID;
- int passprompt = 0;
- bool bCopyToPublic = false;
- int nFolderId = 0;
- list<string> lstUsernames;
- bool bAutoAccept = false, bDeclineConflict = false, bDeclineRecurring = false;
- ULONG cbEntryID = 0;
- object_ptr<IMsgStore> lpPublicStore, lpUserStore;
- object_ptr<IExchangeManageStore> lpIEMS;
- wstring strUsername;
- wstring strStorename;
- wstring strStorenameTMP;
- wstring strCompanyName;
- object_ptr<IMAPIFolder> lpDeletedStoresFolder, lpRootFolder;
- ULONG ulObjType = 0;
- ULONG ulCachePurgeMode = PURGE_CACHE_ALL;
- unsigned int loglevel = EC_LOGLEVEL_NONE;
- object_ptr<ECLogger> lpLogger;
- const configsetting_t lpDefaults[] = {
- { "server_socket", "default:" },
- { "sslkey_file", "" },
- { "sslkey_pass", "", CONFIGSETTING_EXACT },
- { NULL, NULL },
- };
- std::unique_ptr<ECConfig> lpsConfig(ECConfig::Create(lpDefaults));
- bool bExplicitConfig = false;
- ConsoleTable ct(0,0);
- const char *szConfig = ECConfig::GetDefaultPath("admin.cfg");
- // Set locale to system variables
- setlocale(LC_MESSAGES, "");
- setlocale(LC_CTYPE, "");
- setlocale(LC_TIME, "");
- if(argc < 2) {
- print_help(argv[0]);
- return 1;
- }
- int c;
- while (1) {
- InputValidator validateInput;
- c = getopt_long(argc, argv, "VlLsc:u:d:U:Pp:f:e:a:h:g:G:b:B:i:I:n:v", long_options, NULL);
- if (c == -1)
- break;
- switch (c) {
- case OPT_VERBOSITY:
- loglevel = strtoul(optarg, NULL, 0);
- break;
- case 'v':
- if (loglevel < EC_LOGLEVEL_DEBUG + 1)
- ++loglevel;
- break;
- case 'l':
- mode = MODE_LIST_USERS;
- break;
- case 's':
- mode = MODE_CREATE_PUBLIC;
- break;
- case 'c':
- mode = MODE_CREATE_USER;
- username = validateInput(optarg);
- break;
- case 'u':
- if (mode != MODE_HOOK_STORE)
- mode = MODE_UPDATE_USER;
- username = validateInput(optarg);
- break;
- case 'd':
- mode = MODE_DELETE_USER;
- username = validateInput(optarg);
- break;
- case 'g':
- mode = MODE_CREATE_GROUP;
- groupname = validateInput(optarg);
- break;
- case 'G':
- mode = MODE_DELETE_GROUP;
- groupname = validateInput(optarg);
- break;
- case 'L':
- mode = MODE_LIST_GROUP;
- break;
- case 'b':
- mode = MODE_ADDUSER_GROUP;
- username = validateInput(optarg);
- break;
- case 'B':
- mode = MODE_DELETEUSER_GROUP;
- username = validateInput(optarg);
- break;
- case 'U':
- new_username = validateInput(optarg);
- break;
- case 'P':
- passprompt = 1;
- break;
- case 'p':
- password = validateInput(optarg);
- break;
- case 'f':
- fullname = validateInput(optarg);
- break;
- case 'e':
- emailadr = validateInput(optarg);
- break;
- case 'a':
- isadmin = atoi(optarg);
- if (isadmin == 0)
- isadmin = parse_yesno(optarg);
- else
- isadmin = min(2, isadmin);
- break;
- case 'n':
- isnonactive = parse_yesno(optarg);
- break;
- case 'i':
- groupname = validateInput(optarg);
- break;
- case 'I':
- companyname = validateInput(optarg);
- break;
- // error handling?
- case '?':
- break;
- case OPT_HOST:
- case 'h':
- path = validateInput(optarg);
- break;
- case OPT_HELP:
- mode = MODE_HELP;
- break;
- case OPT_CREATE_STORE:
- mode = MODE_CREATE_STORE;
- username = validateInput(optarg);
- break;
- case OPT_DELETE_STORE:
- mode = MODE_DELETE_STORE;
- username = validateInput(optarg);
- break;
- case OPT_HOOK_STORE:
- mode = MODE_HOOK_STORE;
- storeguid = validateInput(optarg);
- break;
- case OPT_UNHOOK_STORE:
- mode = MODE_UNHOOK_STORE;
- username = validateInput(optarg);
- break;
- case OPT_REMOVE_STORE:
- mode = MODE_REMOVE_STORE;
- storeguid = validateInput(optarg);
- break;
- case OPT_COPYTO_PUBLIC:
- bCopyToPublic = true;
- break;
- case OPT_SYNC_USERS:
- mode = MODE_SYNC_USERS;
- break;
- case OPT_DETAILS:
- mode = MODE_DETAILS;
- username = validateInput(optarg);
- break;
- case OPT_DETAILS_TYPE:
- detailstype = validateInput(optarg);
- break;
- // Make values from Mb to bytes which the server wants
- case OPT_USER_QUOTA_HARD:
- quotahard = atoll(optarg) *1024*1024;
- break;
- case OPT_USER_QUOTA_SOFT:
- quotasoft = atoll(optarg) *1024*1024;
- break;
- case OPT_USER_QUOTA_WARN:
- quotawarn = atoll(optarg) *1024*1024;
- break;
- case OPT_USER_QUOTA_OVERRIDE:
- quota = parse_yesno(optarg);
- break;
- case OPT_USER_DEFAULT_QUOTA_HARD:
- ud_quotahard = atoll(optarg) * 1024 * 1024;
- break;
- case OPT_USER_DEFAULT_QUOTA_SOFT:
- ud_quotasoft = atoll(optarg) * 1024 * 1024;
- break;
- case OPT_USER_DEFAULT_QUOTA_WARN:
- ud_quotawarn = atoll(optarg) * 1024 * 1024;
- break;
- case OPT_USER_DEFAULT_QUOTA_OVERRIDE:
- ud_quota = parse_yesno(optarg);
- break;
- case OPT_LANG:
- // Use alternate language
- lang = validateInput(optarg);
- break;
- case OPT_MR_ACCEPT:
- mr_accept = parse_yesno(optarg);
- break;
- case OPT_MR_DECLINE_CONFLICT:
- mr_decline_conflict = parse_yesno(optarg);
- break;
- case OPT_MR_DECLINE_RECURRING:
- mr_decline_recurring = parse_yesno(optarg);
- break;
- case OPT_LIST_SENDAS:
- mode = MODE_LIST_SENDAS;
- username = validateInput(optarg);
- break;
- case OPT_ADD_SENDAS:
- sendas_user = validateInput(optarg);
- sendas_action = 1;
- break;
- case OPT_DEL_SENDAS:
- sendas_user = validateInput(optarg);
- sendas_action = 0;
- break;
- case OPT_UPDATE_GROUP:
- mode = MODE_UPDATE_GROUP;
- groupname = validateInput(optarg);
- break;
- case OPT_CREATE_COMPANY:
- mode = MODE_CREATE_COMPANY;
- companyname = validateInput(optarg);
- break;
- case OPT_UPDATE_COMPANY:
- mode = MODE_UPDATE_COMPANY;
- companyname = validateInput(optarg);
- break;
- case OPT_DELETE_COMPANY:
- mode = MODE_DELETE_COMPANY;
- companyname = validateInput(optarg);
- break;
- case OPT_LIST_COMPANY:
- mode = MODE_LIST_COMPANY;
- break;
- case OPT_ADD_VIEW:
- mode = MODE_ADD_VIEW;
- set_companyname = validateInput(optarg);
- break;
- case OPT_DEL_VIEW:
- mode = MODE_DEL_VIEW;
- set_companyname = validateInput(optarg);
- break;
- case OPT_LIST_VIEW:
- mode = MODE_LIST_VIEW;
- break;
- case OPT_ADD_ADMIN:
- mode = MODE_ADD_ADMIN;
- username = validateInput(optarg);
- break;
- case OPT_DEL_ADMIN:
- mode = MODE_DEL_ADMIN;
- username = validateInput(optarg);
- break;
- case OPT_LIST_ADMIN:
- mode = MODE_LIST_ADMIN;
- break;
- case OPT_SYSTEM_ADMIN:
- mode = MODE_SYSTEM_ADMIN;
- username = validateInput(optarg);
- break;
- case OPT_ADD_UQUOTA_RECIPIENT:
- mode = MODE_ADD_USERQUOTA_RECIPIENT;
- username = validateInput(optarg);
- break;
- case OPT_DEL_UQUOTA_RECIPIENT:
- mode = MODE_DEL_USERQUOTA_RECIPIENT;
- username = validateInput(optarg);
- break;
- case OPT_LIST_UQUOTA_RECIPIENT:
- mode = MODE_LIST_USERQUOTA_RECIPIENT;
- break;
- case OPT_ADD_CQUOTA_RECIPIENT:
- mode = MODE_ADD_COMPANYQUOTA_RECIPIENT;
- username = validateInput(optarg);
- break;
- case OPT_DEL_CQUOTA_RECIPIENT:
- mode = MODE_DEL_COMPANYQUOTA_RECIPIENT;
- username = validateInput(optarg);
- break;
- case OPT_LIST_CQUOTA_RECIPIENT:
- mode = MODE_LIST_COMPANYQUOTA_RECIPIENT;
- break;
- case OPT_PURGE_SOFTDELETE:
- ulDays = atoui(optarg);
- mode = MODE_PURGE_SOFTDELETE;
- break;
- case OPT_CLEAR_CACHE:
- mode = MODE_CLEAR_CACHE;
- if (optarg)
- ulCachePurgeMode = strtol(optarg, NULL, 0);
- break;
- case OPT_PURGE_DEFERRED:
- mode = MODE_PURGE_DEFERRED;
- break;
- case OPT_LIST_ORPHANS:
- mode = MODE_LIST_ORPHANS;
- break;
- case OPT_CONFIG:
- szConfig = validateInput(optarg);
- bExplicitConfig = true;
- break;
- case OPT_UTF8: {
- // set early, so other arguments are parsed in this charset.
- std::string locale;
- if (!forceUTF8Locale(false, &locale)) {
- cerr << "Your system does not have the '" << locale << "' locale installed." << endl;
- cerr << "Please install this locale before creating new users." << endl;
- return 1;
- }
- break;
- }
- case OPT_FORCE_RESYNC:
- mode = MODE_FORCE_RESYNC;
- break;
- case OPT_USER_COUNT:
- mode = MODE_USER_COUNT;
- break;
- case OPT_ENABLE_FEATURE:
- case OPT_DISABLE_FEATURE:
- if (feature) {
- cerr << "Only one feature can be enabled/disabled at a time" << endl;
- break;
- }
- if (!isFeature(optarg)) {
- cerr << optarg << " is not a valid kopano feature" << endl;
- break;
- }
- feature = optarg;
- bFeature = (c == OPT_ENABLE_FEATURE);
- break;
- case OPT_VERSION:
- case 'V':
- cout << "Product version:\t" << PROJECT_VERSION_PROFADMIN_STR << endl
- << "File version:\t\t" << PROJECT_SVN_REV_STR << endl;
- return EXIT_SUCCESS;
- case OPT_SELECT_NODE:
- node = validateInput(optarg);
- break;
- case OPT_RESET_FOLDER_COUNT:
- mode = MODE_RESET_FOLDER_COUNT;
- username = validateInput(optarg);
- break;
- default:
- break;
- };
- if (validateInput.Failed()) {
- cerr << "Invalid input '" << optarg << "' found." << endl;
- // no need to return, later input checking will print an error too
- }
- }
- // check empty input
- if (username && username[0] == 0x00) {
- cerr << "Username (-u) cannot be empty" << endl;
- return 1;
- }
- if (username && strcasecmp(username, "SYSTEM")==0) {
- cerr << "Username (-u) cannot be SYSTEM" << endl;
- return 1;
- }
- if (password && password[0] == 0x00) {
- cerr << "Password (-p) cannot be empty" << endl;
- return 1;
- }
- if (companyname && companyname[0] == 0x00){
- cerr << "Companyname (-I) cannot be empty" << endl;
- return 1;
- }
- if (groupname && groupname[0] == 0x00) {
- cerr << "Groupname cannot be empty" << endl;
- return 1;
- }
- if (fullname && fullname[0] == 0x00) {
- cerr << "Fullname (-f) cannot be empty" << endl;
- return 1;
- }
- if (emailadr && emailadr[0] == 0x00) {
- cerr << "Email address (-e) cannot be empty" << endl;
- return 1;
- }
- // --force-resync takes all left over arguments as usernames
- if (mode == MODE_FORCE_RESYNC) {
- assert(optind <= argc);
- std::copy(argv + optind, argv + argc, std::back_inserter(lstUsernames));
- optind = argc;
- }
- // check parameters
- if (optind < argc) {
- cerr << "Too many options given." << endl;
- return 1;
- }
- if (mode == MODE_INVALID) {
- cerr << "No correct command (e.g. -c for create user) given." << endl;
- return 1;
- }
- if (mode == MODE_HELP) {
- print_help(argv[0]);
- cout << endl << "Please read kopano-admin(8) for detailed information. Enter `man kopano-admin` to view it." << endl << endl;
- return 0;
- }
- // For the following modes we need a company name.
- if (!companyname &&
- (mode == MODE_ADD_VIEW || mode == MODE_DEL_VIEW || mode == MODE_LIST_VIEW ||
- mode == MODE_ADD_ADMIN || mode == MODE_DEL_ADMIN || mode == MODE_LIST_ADMIN ||
- mode == MODE_SYSTEM_ADMIN)){
- cerr << "Missing companyname to perform action" << endl;
- return 1;
- }
- if (mode == MODE_DETAILS && username == NULL) {
- cerr << "Missing information to show user details." << endl;
- return 1;
- }
- if (mode == MODE_CREATE_USER) {
- bool has_username = username != NULL;
- bool has_password = !(password == NULL && passprompt == 0 && isnonactive < 1);
- bool has_emailaddr = emailadr != NULL;
- bool has_fullname = fullname != NULL;
- if (!has_username || !has_password || !has_emailaddr || !has_fullname) {
- cerr << "Missing information to create user:";
- if (!has_username)
- cerr << " username (-u)";
- if (!has_password)
- cerr << " password (-p)";
- if (!has_emailaddr)
- cerr << " email address (-e)";
- if (!has_fullname)
- cerr << " full name (-f)";
- cerr << endl;
- return EXIT_FAILURE;
- }
- }
- if (mode == MODE_CREATE_USER && quota == 1 && (quotahard == -1 || quotawarn == -1 || quotasoft == -1)) {
- cerr << "Not all user specific quota levels are given." << endl;
- cerr << "Missing information to create user:";
- missing_quota(quotahard, quotawarn, quotasoft);
- cerr << endl;
- return 1;
- }
- if (mode == MODE_CREATE_COMPANY &&
- ((quota == 1 && quotawarn == -1) ||
- (ud_quota == 1 && (ud_quotahard == -1 || ud_quotasoft == -1 || ud_quotawarn == -1)))) {
- cerr << "Not all company specific quota levels are given." << endl;
- cerr << "Missing information to create company:";
- if (quota == 1 && quotawarn == -1)
- cerr << " warn quota (--qw)";
- if (ud_quota == 1)
- missing_quota(ud_quotahard, ud_quotawarn, ud_quotasoft);
- cerr << endl;
- return 1;
- }
- if (mode == MODE_CREATE_STORE && username == NULL) {
- cerr << "Missing username (-u) to be able to create store." << endl;
- return 1;
- }
- if (mode == MODE_DELETE_STORE) {
- cerr << "Delete store action is not available anymore. Use --remove-store to remove a store from the database." << endl;
- return 1;
- }
- if (mode == MODE_HOOK_STORE && (storeguid == NULL || (username == NULL && bCopyToPublic == false) ) ) {
- cerr << "Missing information to hook store:";
- if (storeguid == NULL)
- cerr << " store GUID (--hook-store)";
- if (username == NULL && bCopyToPublic == false)
- cerr << " username (-u)";
- cerr << endl;
- return 1;
- }
- if (mode == MODE_UNHOOK_STORE && username == NULL) {
- cerr << "Missing username (-u) to unhook store for." << endl;
- return 1;
- }
- if (mode == MODE_REMOVE_STORE && storeguid == NULL) {
- cerr << "Missing guid (--remove-store) to remove store for." << endl;
- return 1;
- }
- if (mode == MODE_UPDATE_USER && password == NULL && passprompt == 0 &&
- emailadr == NULL && fullname == NULL && new_username == NULL && isadmin == -1 &&
- quota == -1 && quotahard == -1 && quotasoft == -1 && quotawarn == -1 &&
- mr_accept == -1 && mr_decline_conflict == -1 && mr_decline_recurring == -1 &&
- sendas_user == NULL && isnonactive == -1 && feature == NULL) {
- cerr << "Missing information to update user (e.g. password, quota, see --help)." << endl;
- return 1;
- }
- if (mode == MODE_DELETE_USER && username == NULL) {
- cerr << "Missing username (-u) to delete." << endl;
- return 1;
- }
- if (mode == MODE_CREATE_GROUP && groupname == NULL) {
- cerr << "Missing name of group (-g) to create." << endl;
- return 1;
- }
- if (mode == MODE_UPDATE_GROUP && (groupname == NULL || (emailadr == NULL && sendas_user == NULL) ) ) {
- cerr << "Missing information to update group:";
- if (!groupname)
- cerr << " group name";
- if (!emailadr && !sendas_user)
- cerr << " either e-mail address (-e) or \"send-as user\" (--add-sendas)";
- cerr << endl;
- return 1;
- }
- if (mode == MODE_DELETE_GROUP && groupname == NULL) {
- cerr << "Missing name of group (-G) to delete." << endl;
- return 1;
- }
- if (mode == MODE_ADDUSER_GROUP && (groupname == NULL || username == NULL)) {
- cerr << "Missing information to add user to group:";
- if (!groupname)
- cerr << " group name (-i)";
- if (!username)
- cerr << " user name";
- cerr << endl;
- return 1;
- }
- if (mode == MODE_DELETEUSER_GROUP && (groupname == NULL || username == NULL)) {
- cerr << "Missing information to remove user from group:";
- if (!groupname)
- cerr << " group name (-i)";
- if (!username)
- cerr << " user name";
- cerr << endl;
- return 1;
- }
- if (mode == MODE_CREATE_COMPANY && companyname == NULL) {
- cerr << "Missing name of company to create." << endl;
- return 1;
- }
- if (mode == MODE_UPDATE_COMPANY &&
- ((quota == 1 && quotawarn == -1) ||
- (ud_quota == 1 && (ud_quotahard == -1 || ud_quotasoft == -1 || ud_quotawarn == -1)))) {
- cerr << "Missing information to update company:";
- if (quota == 1 && quotawarn == -1)
- cerr << " warn quota (--qw)";
- if (ud_quota == 1)
- missing_quota(ud_quotahard, ud_quotawarn, ud_quotasoft);
- cerr << endl;
- return 1;
- }
- if (mode == MODE_DELETE_COMPANY && companyname == NULL) {
- cerr << "Missing name of company to delete." << endl;
- return 1;
- }
- if (mode == MODE_ADD_VIEW && set_companyname == NULL) {
- cerr << "Missing company name to add remote view privilege to." << endl;
- return 1;
- }
- if (mode == MODE_DEL_VIEW && set_companyname == NULL) {
- cerr << "Missing company name to delete remote view privilege to." << endl;
- return 1;
- }
- if (mode == MODE_ADD_ADMIN && username == NULL) {
- cerr << "Missing username to add remote administrator to." << endl;
- return 1;
- }
- if (mode == MODE_DEL_ADMIN && username == NULL) {
- cerr << "Missing username to delete remote administration privilege for." << endl;
- return 1;
- }
- if (mode == MODE_SYSTEM_ADMIN && username == NULL) {
- cerr << "Missing username to set system administrator privilege for." << endl;
- return 1;
- }
- if ((mode == MODE_ADD_USERQUOTA_RECIPIENT || mode == MODE_DEL_USERQUOTA_RECIPIENT ||
- mode == MODE_ADD_COMPANYQUOTA_RECIPIENT || mode == MODE_DEL_COMPANYQUOTA_RECIPIENT) &&
- username == NULL) {
- cerr << "Missing username to edit quota recipients for." << endl;
- return 1;
- }
- if (mode == MODE_RESET_FOLDER_COUNT && username == NULL) {
- cerr << "Missing username to reset folder counts for." << endl;
- return 1;
- }
- if (lang && mode != MODE_CREATE_STORE) {
- cerr << "You can only use the --lang option in combination with --create-store. If you" << endl;
- cerr << "wish to create a store in this language, you must edit the 00createstore" << endl;
- cerr << "script, which is probably in /etc/kopano/userscripts/createuser.d and" << endl;
- cerr << "specify the --lang option there." << endl;
- return 1;
- }
- // check warnings
- if (new_username != NULL && mode != MODE_UPDATE_USER) {
- cerr << "WARNING: new username \"" << new_username << "\" will be ignored (only used for -U)." << endl;
- }
- if ((quota == 0 && (quotawarn >= 0 || quotasoft >= 0 || quotahard >= 0)) ||
- (ud_quota == 0 && (ud_quotawarn >= 0 || ud_quotasoft >= 0 || ud_quotahard >= 0))) {
- cerr << "Disabling quota override, but quota levels are provided." << endl;
- cerr << "By disabling quota overrides the existing values will be reset," << endl;
- cerr << "and these new values will be ignored." << endl;
- }
- if ((quota == -1 && (quotawarn >= 0 || quotasoft >= 0 || quotahard >= 0)) ||
- (ud_quota == -1 && (ud_quotawarn >= 0 || ud_quotasoft >= 0 || ud_quotahard >= 0))) {
- cerr << "Quota levels are provided, but not quota level override" << endl;
- cerr << "Without an explicit quota override value the quota levels will be ignored." << endl;
- }
- // confirmations
- if (mode == MODE_FORCE_RESYNC && lstUsernames.empty()) {
- string response;
- cout << "You requested a forced resync without arguments, are you sure you want" << endl;
- cout << "force a resync of all offline profiles for all users? [y/N]: ";
- cin >> response;
- if (response.empty() || strcasecmp(response.c_str(), "n") == 0 || strcasecmp(response.c_str(), "no") == 0)
- return 0;
- if (strcasecmp(response.c_str(), "y") != 0 && strcasecmp(response.c_str(), "yes") != 0) {
- cout << "Invalid response." << endl;
- return 1;
- }
- }
- if (szConfig) {
- bool bHaveConfig = lpsConfig->LoadSettings(szConfig);
- /* Special case on complaining errors in config file:
- * - explicit config file given, but was not found
- * - default config file loaded, but contains errors
- * This makes that we do not complain for errors when an "invalid" config was given but did load ok.
- * This is a trick that people can use to give the spooler or dagent a config for its client SSL settings,
- * which are the only settings used by the admin program.
- */
- if ((!bHaveConfig && bExplicitConfig) || (bHaveConfig && !bExplicitConfig && lpsConfig->HasErrors())) {
- cerr << "Error while reading configuration file " << szConfig << endl;
- // create fatal logger without a timestamp to stderr
- lpLogger.reset(new ECLogger_File(EC_LOGLEVEL_FATAL, 0, "-", false), false);
- ec_log_set(lpLogger);
- LogConfigErrors(lpsConfig.get());
- return 1;
- }
- }
- if(lang) {
- char* locale = setlocale(LC_MESSAGES, lang);
- if (!locale) {
- cerr << "Your system does not have the '" << lang << "' locale installed." << endl;
- cerr << "Please install this locale before creating new users." << endl;
- // do not create store in wrong language, give admin a chance to make store in right locale
- return 1;
- }
- }
- if (loglevel > EC_LOGLEVEL_DEBUG)
- loglevel = EC_LOGLEVEL_ALWAYS;
- if (loglevel > EC_LOGLEVEL_NONE)
- lpLogger.reset(new ECLogger_File(loglevel, 0, "-", false), false);
- else
- lpLogger.reset(new ECLogger_Null, false);
- ec_log_set(lpLogger);
- //Init mapi
- hr = mapiinit.Initialize();
- if (hr != hrSuccess) {
- cerr << "Unable to initialize" << endl;
- goto exit;
- }
- /*
- * server path sequence is:
- * 1. -h option from command line
- * 2. KOPANO_SOCKET environment variable
- * 3. config setting option
- */
- if (!path) {
- path = lpsConfig->GetSetting("server_socket");
- // environment variable may override setting variable
- path = GetServerUnixSocket(path);
- }
- hr = HrOpenECAdminSession(&~lpSession, "admin", PROJECT_SVN_REV_STR,
- path, EC_PROFILE_FLAGS_NO_NOTIFICATIONS,
- lpsConfig->GetSetting("sslkey_file", "", NULL),
- lpsConfig->GetSetting("sslkey_pass", "", NULL));
- if(hr != hrSuccess) {
- cerr << "Unable to open Admin session: " <<
- GetMAPIErrorMessage(hr) << " (" <<
- stringify(hr, true) << ")" << endl;
- switch (hr) {
- case MAPI_E_NETWORK_ERROR:
- cerr << "The server is not running, or not accessible";
- if (path != NULL && *path != '\0')
- cerr << " through \"" << path << "\"";
- cerr << "." << endl;
- break;
- case MAPI_E_LOGON_FAILED:
- case MAPI_E_NO_ACCESS:
- cerr << "Access was denied on " << path << "." << endl;
- break;
- default:
- break;
- };
- goto exit;
- }
- hr = HrOpenDefaultStore(lpSession, &~lpMsgStore);
- if(hr != hrSuccess) {
- cerr << "Unable to open Admin store: " <<
- GetMAPIErrorMessage(hr) << " (" <<
- stringify(hr,true) << ")" << endl;
- goto exit;
- }
- if (node != NULL && *node != '\0') {
- MsgStorePtr ptrRemoteStore;
- hr = HrGetRemoteAdminStore(lpSession, lpMsgStore, (LPTSTR)node, 0, &~ptrRemoteStore);
- if (hr != hrSuccess) {
- cerr << "Unable to connect to node '" << node << "':" <<
- GetMAPIErrorMessage(hr) << " (" <<
- stringify(hr, true) << ")" << endl;
- switch (hr) {
- case MAPI_E_NETWORK_ERROR:
- cerr << "The server is not running, or not accessible." << endl;
- break;
- case MAPI_E_LOGON_FAILED:
- case MAPI_E_NO_ACCESS:
- cerr << "Access was denied." << endl;
- break;
- case MAPI_E_NOT_FOUND:
- cerr << "Node '" << node << "' is unknown." << endl;
- break;
- default:
- break;
- }
- goto exit;
- }
- lpMsgStore.reset(ptrRemoteStore.release(), false);
- }
- hr = HrGetOneProp(lpMsgStore, PR_EC_OBJECT, &~lpPropValue);
- if(hr != hrSuccess || !lpPropValue || !lpPropValue->Value.lpszA) {
- cerr << "Admin object not found." << endl;
- goto exit;
- }
- lpECMsgStore.reset(reinterpret_cast<IECUnknown *>(lpPropValue->Value.lpszA));
- hr = lpECMsgStore->QueryInterface(IID_IECServiceAdmin, &~lpServiceAdmin);
- if(hr != hrSuccess) {
- cerr << "Admin object query error." << endl;
- goto exit;
- }
- switch (mode) {
- case MODE_CREATE_PUBLIC:
- if (companyname == nullptr) {
- cbCompanyId = g_cbEveryoneEid;
- hr = MAPIAllocateBuffer(g_cbEveryoneEid, &~lpCompanyId);
- if (hr != hrSuccess)
- goto exit;
- memcpy(lpCompanyId, g_lpEveryoneEid, g_cbEveryoneEid);
- break;
- }
- /* fallthrough */
- case MODE_UPDATE_COMPANY:
- case MODE_DELETE_COMPANY:
- case MODE_ADD_VIEW:
- case MODE_DEL_VIEW:
- case MODE_LIST_VIEW:
- case MODE_ADD_ADMIN:
- case MODE_DEL_ADMIN:
- case MODE_LIST_ADMIN:
- case MODE_SYSTEM_ADMIN:
- case MODE_ADD_USERQUOTA_RECIPIENT:
- case MODE_DEL_USERQUOTA_RECIPIENT:
- case MODE_LIST_USERQUOTA_RECIPIENT:
- case MODE_ADD_COMPANYQUOTA_RECIPIENT:
- case MODE_DEL_COMPANYQUOTA_RECIPIENT:
- case MODE_LIST_COMPANYQUOTA_RECIPIENT:
- hr = lpServiceAdmin->ResolveCompanyName((LPTSTR)companyname, 0, &cbCompanyId, &~lpCompanyId);
- if (hr != hrSuccess) {
- fprintf(stderr, "Failed to resolve company: %s\n", getMapiCodeString(hr, companyname).c_str());
- goto exit;
- }
- break;
- default:
- break;
- }
- switch (mode) {
- case MODE_CREATE_STORE:
- case MODE_DELETE_USER:
- case MODE_UPDATE_USER:
- case MODE_ADDUSER_GROUP:
- case MODE_DELETEUSER_GROUP:
- case MODE_ADD_ADMIN:
- case MODE_DEL_ADMIN:
- case MODE_SYSTEM_ADMIN:
- case MODE_ADD_USERQUOTA_RECIPIENT:
- case MODE_DEL_USERQUOTA_RECIPIENT:
- case MODE_ADD_COMPANYQUOTA_RECIPIENT:
- case MODE_DEL_COMPANYQUOTA_RECIPIENT:
- if (username == nullptr)
- break;
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- if (hr != hrSuccess) {
- fprintf(stderr, "Failed to resolve user: %s\n", getMapiCodeString(hr, username).c_str());
- goto exit;
- }
- break;
- default:
- break;
- }
- // fully logged on, action!
- switch(mode) {
- case MODE_LIST_USERS:
- hr = ForEachCompany(lpServiceAdmin, companyname, ListUsers);
- if (hr != hrSuccess)
- goto exit;
- break;
- case MODE_LIST_ORPHANS:
- hr = list_orphans(lpServiceAdmin);
- if (hr != hrSuccess)
- goto exit;
- break;
- case MODE_DETAILS:
- if (detailstype == NULL || strcasecmp(detailstype, "user") == 0)
- ulClass = ACTIVE_USER;
- else if (strcasecmp(detailstype, "group") == 0)
- ulClass = DISTLIST_GROUP;
- else if (strcasecmp(detailstype, "company") == 0)
- ulClass = CONTAINER_COMPANY;
- else if (strcasecmp(detailstype, "archive") != 0) {
- hr = MAPI_E_INVALID_TYPE;
- cerr << "Unknown userobject type \"" << detailstype << "\"" << endl;
- goto exit;
- }
- if (detailstype && strcasecmp(detailstype, "archive") == 0)
- hr = print_archive_details(lpSession, lpECMsgStore, username);
- else
- hr = print_details(lpSession, lpECMsgStore, ulClass, username);
- if (hr != hrSuccess)
- goto exit;
- break;
- case MODE_CREATE_PUBLIC:
- /* The public store is created for a particular company, to do this correctly we will
- * pass the company id as the group id for the store. */
- hr = lpServiceAdmin->CreateStore(ECSTORE_TYPE_PUBLIC, cbCompanyId, lpCompanyId, &cbStoreId, &~lpStoreId, &cbRootId, &~lpRootId);
- if(hr != hrSuccess) {
- cerr << "Unable to create store, " << getMapiCodeString(hr, "public") << endl;
- goto exit;
- }
- cout << "Public created." << endl;
- break;
- case MODE_CREATE_USER:
- memset(&sECUser, 0, sizeof(sECUser));
- sECUser.sUserId.cb = g_cbDefaultEid;
- sECUser.sUserId.lpb = g_lpDefaultEid;
- sECUser.lpszUsername = (LPTSTR)username;
- if (passprompt && isnonactive != 1) {
- sECUser.lpszPassword = (LPTSTR)get_password();
- if (sECUser.lpszPassword == NULL) {
- cerr << "Passwords don't match" << endl;
- return 1;
- }
- if (sECUser.lpszPassword[0] == 0x00) {
- cerr << "Password cannot be empty" << endl;
- return 1;
- }
- }
- else if (isnonactive == 1)
- sECUser.lpszPassword = NULL;
- else
- sECUser.lpszPassword = (LPTSTR)password;
- sECUser.lpszMailAddress = (LPTSTR)emailadr;
- sECUser.lpszFullName = (LPTSTR)fullname;
- sECUser.ulIsAdmin = (isadmin != -1)?isadmin:0;
- // FIXME: create user, room or equipment!
- sECUser.ulObjClass = ACTIVE_USER;
- if (isnonactive == 1)
- sECUser.ulObjClass = NONACTIVE_USER; // NONACTIVE_ROOM, NONACTIVE_EQUIPMENT, (NONACTIVE_CONTACT?)
- hr = lpServiceAdmin->CreateUser(&sECUser, 0, &cbUserId, &~lpUserId);
- if (hr != hrSuccess) {
- cerr << "Unable to create user, " << getMapiCodeString(hr, username) << endl;
- cerr << "Check server.log for details." << endl;
- goto exit;
- }
- // set quota data
- if (quota != -1 || quotahard != -1 || quotasoft != -1 || quotawarn != -1) {
- hr = setQuota(lpServiceAdmin, cbUserId, lpUserId, quota, false, quotawarn, quotasoft, quotahard);
- if(hr != hrSuccess)
- goto exit;
- }
- cout << "User created." << endl;
- break;
- case MODE_CREATE_STORE:
- hr = lpServiceAdmin->CreateStore(ECSTORE_TYPE_PRIVATE, cbUserId, lpUserId, &cbStoreId, &~lpStoreId, &cbRootId, &~lpRootId);
- if (hr != hrSuccess) {
- cerr << "Unable to create store, " << getMapiCodeString(hr, "store") << endl;
- goto exit;
- }
- cout << "User store '" << username << "' created." << endl;
- break;
- case MODE_DELETE_USER:
- hr = lpServiceAdmin->DeleteUser(cbUserId, lpUserId);
- if (hr != hrSuccess) {
- cerr << "Unable to delete user, " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- cout << "User deleted." << endl;
- break;
- case MODE_DELETE_STORE:
- // happy compiler
- break;
- case MODE_HOOK_STORE:
- hr = Util::hex2bin(storeguid, sizeof(GUID)*2, &cbGUID, (&~lpGUID).as<unsigned char>());
- if (hr != hrSuccess) {
- cerr << "Incorrect store guid '" << storeguid << "'" << endl;
- goto exit;
- }
- if (bCopyToPublic == true) {
- // Find store entryid
- hr = GetOrphanStoreInfo(lpServiceAdmin, lpGUID, path, strUsername, strCompanyName, &cbEntryID, &~lpEntryID);
- if (hr != hrSuccess) {
- cerr << "Unable to get the store information. store guid '" << storeguid << "'" << endl;
- goto exit;
- }
- // Open store the orphan store
- hr = lpSession->OpenMsgStore(0, cbEntryID, lpEntryID, nullptr, MAPI_BEST_ACCESS, &~lpUserStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open the orphan store, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- // Open the root container for copy the folders
- hr = lpUserStore->OpenEntry(0, nullptr, nullptr, MAPI_BEST_ACCESS, &ulObjType, &~lpRootFolder);
- if (hr != hrSuccess) {
- cerr << "Unable to open root folder of the orphan store, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- hr = HrGetOneProp(lpUserStore, PR_IPM_SUBTREE_ENTRYID, &~lpPropValue);
- if (hr != hrSuccess) {
- cerr << "Unable to open the orphans store sub-entry, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- // Open the public store
- hr = GetPublicStore(lpSession, lpUserStore, strCompanyName, &~lpPublicStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open the public store, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- // open/create folders admin/stores
- hr = OpenDeletedStoresFolder(lpPublicStore, &~lpDeletedStoresFolder);
- if (hr != hrSuccess) {
- cerr << "Unable to open the folder \"deleted stores\", " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- // Copy everything to the public
- strStorename = L"Deleted User - ";
- if (strUsername.empty())
- strStorename += convert_to<wstring>(storeguid);
- else
- strStorename += strUsername;
- cerr << "Start to copy the orphan store to the public store to the folder '" << convert_to<string>(strStorename) << "'" << endl;
- strStorenameTMP = strStorename;
- nFolderId = 1;
- while (true)
- {
- hr = lpRootFolder->CopyFolder(lpPropValue->Value.bin.cb, (LPENTRYID)lpPropValue->Value.bin.lpb,
- NULL, lpDeletedStoresFolder, (LPTSTR)strStorenameTMP.c_str(), 0, NULL, COPY_SUBFOLDERS | MAPI_UNICODE);
- if (hr == MAPI_E_COLLISION) {
- if (nFolderId >= 1000) {
- cerr << "Unable to copy the store to the public, maximum folder collisions exceeded" << endl;
- goto exit;
- }
- strStorenameTMP = strStorename + std::to_wstring(nFolderId);
- ++nFolderId;
- cerr << "Folder already exist, retrying with foldername '" << convert_to<string>(strStorenameTMP) << "'" << endl;
- } else if (FAILED(hr)) {
- cerr << "Unable to copy the store to the public," << getMapiCodeString(hr) << endl;
- goto exit;
- } else if (hr != hrSuccess) {
- cerr << "Warning, the copy succeeded, but not all entries were copied (" << getMapiCodeString(hr) << ")" << endl;
- break;
- } else {
- cerr << "Copy succeeded" << endl;
- break;
- }
- }
- } else {
- ULONG ulStoreType;
- if (detailstype == NULL)
- detailstype = "user";
- if (strcmp(detailstype, "user") == 0) {
- ulStoreType = ECSTORE_TYPE_PRIVATE;
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else if (strcmp(detailstype, "archive") == 0) {
- ulStoreType = ECSTORE_TYPE_ARCHIVE;
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else if (strcmp(detailstype, "group") == 0) {
- ulStoreType = ECSTORE_TYPE_PUBLIC;
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else if (strcmp(detailstype, "company") == 0) {
- ulStoreType = ECSTORE_TYPE_PUBLIC;
- hr = lpServiceAdmin->ResolveCompanyName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else {
- cerr << "Unknown store type: '" << detailstype << "'." << endl;
- goto exit;
- }
- if (hr != hrSuccess) {
- cerr << "Unable to find " << detailstype << ", " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- if (strcmp(detailstype, "user") == 0) {
- // check if this user should exist on the connected server. depending on --force, print a warning or an error
- hr = lpServiceAdmin->GetUser(cbUserId, lpUserId, 0, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Unable to load details, " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- // homeserver on single server installations is empty
- if (lpECUser->lpszServername != NULL && *reinterpret_cast<LPSTR>(lpECUser->lpszServername) != '\0') {
- // note, this has to be mapi allocated because GetServerDetails does a More allocation on this base pointer
- if (MAPIAllocateBuffer(sizeof(ECSVRNAMELIST), &~lpsServer) != hrSuccess ||
- MAPIAllocateMore(sizeof(LPTSTR), lpsServer, (void**)&lpsServer->lpszaServer) != hrSuccess) {
- hr = MAPI_E_NOT_ENOUGH_MEMORY;
- cerr << "Unable to allocate memory for server details" << endl;
- goto exit;
- }
- lpsServer->cServers = 1;
- lpsServer->lpszaServer[0] = lpECUser->lpszServername;
- hr = lpServiceAdmin->GetServerDetails(lpsServer, 0, &~lpServerDetails);
- if (hr != hrSuccess) {
- cerr << "Unable to load server details, " << getMapiCodeString(hr, (char*)lpECUser->lpszServername) << endl;
- goto exit;
- }
- if ((lpServerDetails->lpsaServer[0].ulFlags & EC_SDFLAG_IS_PEER) == 0)
- // since we don't know which server we're connected to, don't print a server name.
- cerr << "WARNING: Hooking store on non-homeserver of " << username << endl;
- }
- }
- // the server won't let you hook public stores to users and vice-versa.
- hr = lpServiceAdmin->HookStore(ulStoreType, cbUserId, lpUserId, lpGUID);
- if (hr != hrSuccess) {
- cerr << "Unable to hook store, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "Store hooked." << endl;
- }
- break;
- case MODE_UNHOOK_STORE: {
- ULONG ulStoreType;
- if (detailstype == NULL)
- detailstype = "user";
- if (strcmp(detailstype, "user") == 0) {
- ulStoreType = ECSTORE_TYPE_PRIVATE;
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else if (strcmp(detailstype, "archive") == 0) {
- ulStoreType = ECSTORE_TYPE_ARCHIVE;
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else if (strcmp(detailstype, "group") == 0) {
- ulStoreType = ECSTORE_TYPE_PUBLIC;
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else if (strcmp(detailstype, "company") == 0) {
- ulStoreType = ECSTORE_TYPE_PUBLIC;
- strCompanyName = convert_to<wstring>(username);
- hr = lpServiceAdmin->ResolveCompanyName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else {
- cerr << "Unknown store type: '" << detailstype << "'." << endl;
- goto exit;
- }
- if (hr != hrSuccess) {
- cerr << "Unable to find " << detailstype << ", " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- if (ulStoreType != ECSTORE_TYPE_PUBLIC) {
- if (ulStoreType == ECSTORE_TYPE_ARCHIVE) {
- hr = lpServiceAdmin->GetArchiveStoreEntryID(reinterpret_cast<LPTSTR>(username), NULL, 0, &cbStoreId, &~lpStoreId);
- if (hr != hrSuccess) {
- cout << "Unable to unhook store, unable to retrieve store entryid, " << getMapiCodeString(hr, "entryid") << endl;
- goto exit;
- }
- } else {
- hr = lpMsgStore->QueryInterface(IID_IExchangeManageStore, &~lpIEMS);
- if (hr != hrSuccess)
- goto exit;
- // do not redirect to another server, unhook works on the server it's connected to
- hr = lpIEMS->CreateStoreEntryID(NULL, reinterpret_cast<LPTSTR>(username), OPENSTORE_OVERRIDE_HOME_MDB, &cbStoreId, &~lpStoreId);
- if (hr != hrSuccess) {
- if (hr == MAPI_E_NOT_FOUND)
- cout << "Unable to unhook store. User '" << username << "' has no store attached." << endl;
- else
- cout << "Unable to unhook store. Can not create store entryid, " << getMapiCodeString(hr, "store") << endl;
- goto exit;
- }
- }
- hr = UnWrapStoreEntryID(cbStoreId, lpStoreId, &cbUnWrappedEntry, &~lpUnWrappedEntry);
- if (hr != hrSuccess) {
- cout << "Unable to unhook store. Unable to unwrap the store entryid, " << getMapiCodeString(hr, "entryid") << endl;
- goto exit;
- }
- } else {
- // ns__resolveUserStore (CreateStoreEntryID) does not work with normal (non-company) public store
- hr = GetPublicStore(lpSession, lpMsgStore, strCompanyName, &~lpPublicStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open public store, " << getMapiCodeString(hr, "public") << endl;
- goto exit;
- }
- hr = HrGetOneProp(lpPublicStore, PR_STORE_ENTRYID, &~lpPropValue);
- if (hr != hrSuccess) {
- cerr << "Unable to get public store entryid, " << getMapiCodeString(hr, "store") << endl;
- goto exit;
- }
- hr = UnWrapStoreEntryID(lpPropValue->Value.bin.cb, reinterpret_cast<ENTRYID *>(lpPropValue->Value.bin.lpb), &cbUnWrappedEntry, &~lpUnWrappedEntry);
- if (hr != hrSuccess) {
- cout << "Unable to unhook store. Unable to unwrap the store entryid, " << getMapiCodeString(hr, "entryid") << endl;
- goto exit;
- }
- }
- hr = lpServiceAdmin->UnhookStore(ulStoreType, cbUserId, lpUserId);
- if (hr != hrSuccess) {
- cerr << "Unable to unhook store, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "Store unhooked. Store guid is " << bin2hex(sizeof(GUID), (unsigned char*)lpUnWrappedEntry->ab) << endl;
- break;
- }
- case MODE_REMOVE_STORE:
- hr = Util::hex2bin(storeguid, sizeof(GUID)*2, &cbGUID, (&~lpGUID).as<unsigned char>());
- if (hr != hrSuccess) {
- cerr << "Incorrect store guid '" << storeguid << "'" << endl;
- goto exit;
- }
- hr = lpServiceAdmin->RemoveStore(lpGUID);
- if (hr != hrSuccess) {
- cerr << "Unable to remove store, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "Store removed." << endl;
- break;
- case MODE_UPDATE_USER:
- if (new_username) {
- memory_ptr<ENTRYID> userid;
- ULONG usize;
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(new_username), 0, &usize, &~userid);
- if (hr == hrSuccess) {
- cerr << "User with name '" << new_username << "' is already present." << endl;
- hr = MAPI_E_COLLISION;
- goto exit;
- }
- }
- // get old features. we need these, because not setting them would mean: remove
- hr = lpServiceAdmin->GetUser(cbUserId, lpUserId, 0, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Unable to get user details, " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- // lpECUser memory will be kept alive to let the SetUser() call work
- for (ULONG i = 0; i < lpECUser->sMVPropmap.cEntries; ++i) {
- if (lpECUser->sMVPropmap.lpEntries[i].ulPropId == PR_EC_ENABLED_FEATURES_A)
- sEnabled.insert((char**)lpECUser->sMVPropmap.lpEntries[i].lpszValues,
- (char**)lpECUser->sMVPropmap.lpEntries[i].lpszValues + lpECUser->sMVPropmap.lpEntries[i].cValues);
- else if (lpECUser->sMVPropmap.lpEntries[i].ulPropId == PR_EC_DISABLED_FEATURES_A)
- sDisabled.insert((char**)lpECUser->sMVPropmap.lpEntries[i].lpszValues,
- (char**)lpECUser->sMVPropmap.lpEntries[i].lpszValues + lpECUser->sMVPropmap.lpEntries[i].cValues);
- }
- if (feature) {
- if (bFeature) {
- sEnabled.insert(feature);
- sDisabled.erase(feature);
- } else {
- sEnabled.erase(feature);
- sDisabled.insert(feature);
- }
- }
- if (new_username || password || passprompt || emailadr || fullname || isadmin != -1 || isnonactive != -1 || feature) {
- memset(&sECUser, 0, sizeof(sECUser));
- // if the user did not select an active/inactive state on the command-line,
- // then check what the status was before admin started; that status
- // will then be re-used
- if (isnonactive == -1)
- isnonactive = lpECUser -> ulObjClass == NONACTIVE_USER;
- // copy static info
- sECUser.sUserId.cb = cbUserId;
- sECUser.sUserId.lpb = reinterpret_cast<unsigned char *>(lpUserId.get());
- // possibly set new values
- sECUser.lpszUsername = (LPTSTR)(new_username ? new_username : username);
- if (passprompt) {
- sECUser.lpszPassword = (LPTSTR)get_password();
- if (sECUser.lpszPassword == NULL)
- {
- cerr << "Passwords don't match" << endl;
- return 1;
- }
- if (sECUser.lpszPassword[0] == 0)
- {
- cerr << "Password cannot be empty" << endl;
- return 1;
- }
- } else
- sECUser.lpszPassword = (LPTSTR)password;
- sECUser.lpszMailAddress = (LPTSTR)emailadr;
- sECUser.lpszFullName = (LPTSTR)fullname;
- sECUser.ulIsAdmin = isadmin;
- sECUser.ulObjClass = ACTIVE_USER;
- if (isnonactive == 1)
- sECUser.ulObjClass = NONACTIVE_USER;
- // sEnabled to sECUser.sMVPropmap ergens
- sECUser.sMVPropmap.cEntries = 2; // @note: if we have more mv props than the feature lists, adjust this value!
- // mapi allocate more on lpECUser, so this will be freed automatically at exit.
- hr = MAPIAllocateMore(sizeof(MVPROPMAPENTRY) * sECUser.sMVPropmap.cEntries, lpECUser, (void**)&sECUser.sMVPropmap.lpEntries);
- if (hr != hrSuccess) {
- cerr << "Memory error" << endl;
- goto exit;
- }
- if (fillMVPropmap(sECUser, PR_EC_ENABLED_FEATURES_A, 0, sEnabled, lpECUser) != hrSuccess ||
- fillMVPropmap(sECUser, PR_EC_DISABLED_FEATURES_A, 1, sDisabled, lpECUser) != hrSuccess)
- goto exit;
- hr = lpServiceAdmin->SetUser(&sECUser, 0);
- if (hr != hrSuccess) {
- cerr << "Unable to update user information, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- }
- // update quota data
- if (quota != -1 || quotahard != -1 || quotasoft != -1 || quotawarn != -1) {
- hr = setQuota(lpServiceAdmin, cbUserId, lpUserId, quota, false, quotawarn, quotasoft, quotahard, true);
- if (hr != hrSuccess)
- goto exit;
- }
- if (mr_accept != -1 || mr_decline_conflict != -1 || mr_decline_recurring != -1)
- {
- hr = lpECMsgStore->QueryInterface(IID_IExchangeManageStore, &~lpIEMS);
- if (hr != hrSuccess) {
- cerr << "Unable to get admin interface." << endl;
- goto exit;
- }
- hr = lpIEMS->CreateStoreEntryID((LPTSTR)"", (LPTSTR)username, 0, &cbEntryID, &~lpEntryID);
- if (hr != hrSuccess) {
- cerr << "Unable to get user store entry id. User has possibly has not store." << endl;
- goto exit;
- }
- hr = lpSession->OpenMsgStore(0, cbEntryID, lpEntryID, &IID_IMsgStore, MDB_WRITE, &~lpUserStore);
- if (hr != hrSuccess) {
- cerr << "Unable to open user store." << endl;
- goto exit;
- }
- hr = GetAutoAcceptSettings(lpUserStore, &bAutoAccept, &bDeclineConflict, &bDeclineRecurring);
- if (hr != hrSuccess)
- // ignore and assume 'false' for all values
- hr = hrSuccess;
- if (mr_accept != -1)
- bAutoAccept = mr_accept;
- if (mr_decline_conflict != -1)
- bDeclineConflict = mr_decline_conflict;
- if (mr_decline_recurring != -1)
- bDeclineRecurring = mr_decline_recurring;
- hr = SetAutoAcceptSettings(lpUserStore, bAutoAccept, bDeclineConflict, bDeclineRecurring);
- if (hr != hrSuccess) {
- cerr << "Unable to set auto-accept settings." << endl;
- goto exit;
- }
- }
- if (sendas_user) {
- hr = lpServiceAdmin->ResolveUserName((LPTSTR)sendas_user, 0, &cbSenderId, &~lpSenderId);
- if (hr != hrSuccess) {
- cerr << "Unable to update user, sendas user not available, " << getMapiCodeString(hr, sendas_user) << endl;
- goto exit;
- }
- if (sendas_action == 0)
- // delete sendas user
- hr = lpServiceAdmin->DelSendAsUser(cbUserId, lpUserId, cbSenderId, lpSenderId);
- else if (sendas_action == 1)
- // add sendas user
- hr = lpServiceAdmin->AddSendAsUser(cbUserId, lpUserId, cbSenderId, lpSenderId);
- switch (hr) {
- case MAPI_E_NOT_FOUND:
- // on del, not in list
- if (sendas_action == 0)
- cerr << "Unable to update user, sender " << sendas_user << " not found in sendas list" << endl;
- else
- cerr << "Unable to update user, sender " << sendas_user << " not allowed in sendas list" << endl;
- goto exit;
- case MAPI_E_COLLISION:
- // on add, already added ... too bad an insert collision does not return this error :|
- cerr << "Unable to update user, sender " << sendas_user << " already in sendas list" << endl;
- goto exit;
- default:
- if (hr != hrSuccess) {
- cerr << "Unable to update user, unable to update sendas list, " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- break;
- };
- }
- cout << "User information updated." << endl;
- break;
- case MODE_CREATE_COMPANY:
- memset(&sECCompany, 0, sizeof(sECCompany));
- sECCompany.lpszCompanyname = (LPTSTR)companyname;
- hr = lpServiceAdmin->CreateCompany(&sECCompany, 0, &cbCompanyId, &~lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Unable to create company, " << getMapiCodeString(hr, companyname) << endl;
- goto exit;
- }
- if (quota != -1) {
- // this is the company public quota, and only contains a warning, nothing more.
- hr = setQuota(lpServiceAdmin, cbCompanyId, lpCompanyId, quota, false, quotawarn, 0, 0, false, true);
- if(hr != hrSuccess)
- goto exit;
- }
- if (ud_quota != -1) {
- hr = setQuota(lpServiceAdmin, cbCompanyId, lpCompanyId, ud_quota, true, ud_quotawarn, ud_quotasoft, ud_quotahard);
- if (hr != hrSuccess)
- goto exit;
- }
- cout << "Company created" << endl;
- break;
- case MODE_UPDATE_COMPANY:
- if (quota != -1 || quotahard != -1 || quotasoft != -1 || quotawarn != -1) {
- hr = setQuota(lpServiceAdmin, cbCompanyId, lpCompanyId, quota, false, quotawarn, quotasoft, quotahard, true, true);
- if (hr != hrSuccess)
- goto exit;
- }
- if (ud_quota != -1 || ud_quotahard != -1 || ud_quotasoft != -1 || ud_quotawarn != -1) {
- hr = setQuota(lpServiceAdmin, cbCompanyId, lpCompanyId, ud_quota, true, ud_quotawarn, ud_quotasoft, ud_quotahard);
- if (hr != hrSuccess)
- goto exit;
- }
- break;
- case MODE_DELETE_COMPANY:
- hr = lpServiceAdmin->DeleteCompany(cbCompanyId, lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Unable to delete company, " << getMapiCodeString(hr, companyname) << endl;
- goto exit;
- }
- cout << "Company deleted" << endl;
- break;
- case MODE_LIST_COMPANY:
- hr = lpServiceAdmin->GetCompanyList(0, &cCompanies, &lpECCompanies);
- if(hr != hrSuccess) {
- cerr << "Unable to list companies, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "Company list ("<< cCompanies <<"):" << endl;
- ct.Resize(cCompanies, 2);
- ct.SetHeader(0, "Companyname");
- ct.SetHeader(1, "System administrator");
- for (unsigned int i = 0; i < cCompanies; ++i) {
- ct.AddColumn(0, (LPSTR)lpECCompanies[i].lpszCompanyname);
- hr = lpServiceAdmin->GetUser(lpECCompanies[i].sAdministrator.cb, reinterpret_cast<ENTRYID *>(lpECCompanies[i].sAdministrator.lpb), 0, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Unable to get administrator details, " << getMapiCodeString(hr, "administrator") << endl;
- goto exit;
- }
- ct.AddColumn(1, (LPSTR)lpECUser->lpszUsername);
- }
- ct.PrintTable();
- break;
- case MODE_CREATE_GROUP:
- memset(&sECGroup, 0, sizeof(sECGroup));
- sECGroup.lpszGroupname = (LPTSTR)groupname;
- sECGroup.lpszFullname = (LPTSTR)groupname;
- sECGroup.lpszFullEmail = (LPTSTR)emailadr;
- hr = lpServiceAdmin->CreateGroup(&sECGroup, 0, &cbGroupId, &~lpGroupId);
- if (hr != hrSuccess) {
- cerr << "Unable to create group, " << getMapiCodeString(hr, groupname) << endl;
- goto exit;
- }
- cout << "Group created." << endl;
- break;
- case MODE_UPDATE_GROUP:
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(groupname), 0, &cbGroupId, &~lpGroupId);
- if (hr != hrSuccess) {
- cerr << "Unable to update group, " << getMapiCodeString(hr, groupname) << endl;
- goto exit;
- }
- if (emailadr) {
- memset(&sECGroup, 0, sizeof(sECGroup));
- // copy static info
- sECGroup.sGroupId.cb = cbGroupId;
- sECGroup.sGroupId.lpb = reinterpret_cast<unsigned char *>(lpGroupId.get());
- // possibly set new values
- sECGroup.lpszFullEmail = (LPTSTR)emailadr;
- sECGroup.lpszGroupname = (LPTSTR)groupname;
- sECGroup.lpszFullname = (LPTSTR)groupname;
- hr = lpServiceAdmin->SetGroup(&sECGroup, 0);
- if (hr != hrSuccess) {
- cerr << "Unable to update group information, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- }
- if (sendas_user) {
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(sendas_user), 0, &cbSenderId, &~lpSenderId);
- if (hr != hrSuccess) {
- cerr << "Unable to update group, sendas user not available, " << getMapiCodeString(hr, sendas_user) << endl;
- goto exit;
- }
- if (sendas_action == 0)
- // delete sendas user
- hr = lpServiceAdmin->DelSendAsUser(cbGroupId, lpGroupId, cbSenderId, lpSenderId);
- else if (sendas_action == 1)
- // add sendas user
- hr = lpServiceAdmin->AddSendAsUser(cbGroupId, lpGroupId, cbSenderId, lpSenderId);
- switch (hr) {
- case MAPI_E_NOT_FOUND:
- // on del, not in list
- if (sendas_action == 0)
- cerr << "Unable to update group, sender " << sendas_user << " not found in sendas list" << endl;
- else
- cerr << "Unable to update group, sender " << sendas_user << " not allowed in sendas list" << endl;
- goto exit;
- case MAPI_E_COLLISION:
- // on add, already added ... too bad an insert collision does not return this error :|
- cerr << "Unable to update group, sender " << sendas_user << " already in sendas list" << endl;
- goto exit;
- default:
- if (hr != hrSuccess) {
- cerr << "Unable to update group, unable to update sendas list, " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- break;
- };
- }
- cout << "Group information updated." << endl;
- break;
- case MODE_DELETE_GROUP:
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(groupname), 0, &cbGroupId, &~lpGroupId);
- if (hr != hrSuccess) {
- cerr << "Unable to delete group, " << getMapiCodeString(hr, groupname) << endl;
- goto exit;
- }
- hr = lpServiceAdmin->DeleteGroup(cbGroupId, lpGroupId);
- if (hr != hrSuccess) {
- cerr << "Unable to delete group." << endl;
- goto exit;
- }
- cout << "Group deleted." << endl;
- break;
- case MODE_LIST_GROUP:
- hr = ForEachCompany(lpServiceAdmin, companyname, ListGroups);
- if (hr != hrSuccess)
- goto exit;
- break;
- case MODE_ADDUSER_GROUP:
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(groupname), 0, &cbGroupId, &~lpGroupId);
- if (hr != hrSuccess) {
- cerr << "Unable to add user to group, " << getMapiCodeString(hr, groupname) << endl;
- goto exit;
- }
- hr = lpServiceAdmin->AddGroupUser(cbGroupId, lpGroupId, cbUserId, lpUserId);
- if (hr != hrSuccess) {
- cerr << "Unable to add user to group." << endl;
- goto exit;
- }
- cout << "User added to group." << endl;
- break;
- case MODE_DELETEUSER_GROUP:
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(groupname), 0, &cbGroupId, &~lpGroupId);
- if (hr != hrSuccess) {
- cerr << "Unable to remove user from group, " << getMapiCodeString(hr, groupname) << endl;
- goto exit;
- }
- hr = lpServiceAdmin->DeleteGroupUser(cbGroupId, lpGroupId, cbUserId, lpUserId);
- if (hr != hrSuccess) {
- cerr << "Unable to remove user from group." << endl;
- goto exit;
- }
- cout << "User removed from group." << endl;
- break;
- case MODE_SYNC_USERS:
- hr = SyncUsers(lpServiceAdmin);
- if (hr != hrSuccess)
- goto exit;
- cout << "Users and groups synchronized." << endl;
- break;
- case MODE_ADD_VIEW:
- hr = lpServiceAdmin->ResolveCompanyName(reinterpret_cast<LPTSTR>(set_companyname), 0, &cbSetCompanyId, &~lpSetCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to resolve company name, " << getMapiCodeString(hr, set_companyname) << endl;
- goto exit;
- }
- hr = lpServiceAdmin->AddCompanyToRemoteViewList(cbSetCompanyId, lpSetCompanyId, cbCompanyId, lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to add company to remote-view list" << endl;
- goto exit;
- }
- cout << "Company " << set_companyname << " added to the remote-view list of " << companyname << endl;
- break;
- case MODE_DEL_VIEW:
- hr = lpServiceAdmin->ResolveCompanyName(reinterpret_cast<LPTSTR>(set_companyname), 0, &cbSetCompanyId, &~lpSetCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to resolve company name, " << getMapiCodeString(hr, set_companyname) << endl;
- goto exit;
- }
- hr = lpServiceAdmin->DelCompanyFromRemoteViewList(cbSetCompanyId, lpSetCompanyId, cbCompanyId, lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to remove company from remote-view list, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "Company " << set_companyname << " removed from the remote-view list of " << companyname << endl;
- break;
- case MODE_LIST_VIEW:
- hr = lpServiceAdmin->GetRemoteViewList(cbCompanyId, lpCompanyId, 0, &cCompanies, &lpECCompanies);
- if (hr != hrSuccess) {
- cerr << "Unable to display remote-view list, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "remote-view list ("<< cCompanies <<"):" << endl;
- cout << "\t" << "companyname" << "" << endl;
- cout << "\t-------------------------------------" << endl;
- print_companies(cCompanies, lpECCompanies, true);
- break;
- case MODE_ADD_ADMIN:
- hr = lpServiceAdmin->AddUserToRemoteAdminList(cbUserId, lpUserId, cbCompanyId, lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to add user to remote-admin list, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "User " << username << " added to the remote-admin list of " << companyname << endl;
- break;
- case MODE_DEL_ADMIN:
- hr = lpServiceAdmin->DelUserFromRemoteAdminList(cbUserId, lpUserId, cbCompanyId, lpCompanyId);
- if (hr != hrSuccess) {
- cerr << "Failed to delete user from remote-admin list, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "User " << username << " removed from the remote-admin list of " << companyname << endl;
- break;
- case MODE_LIST_ADMIN:
- hr = lpServiceAdmin->GetRemoteAdminList(cbCompanyId, lpCompanyId, 0, &cUsers, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Unable to display remote-admin list, " << getMapiCodeString(hr) << endl;
- goto exit;
- }
- cout << "remote-admin list ("<< cUsers <<"):" << endl;
- print_users(cUsers, lpECUser);
- break;
- case MODE_SYSTEM_ADMIN:
- memset(&sECCompany, 0, sizeof(sECCompany));
- sECCompany.sAdministrator.cb = cbUserId;
- sECCompany.sAdministrator.lpb = reinterpret_cast<unsigned char *>(lpUserId.get());
- sECCompany.lpszCompanyname = (LPTSTR)companyname;
- sECCompany.sCompanyId.cb = cbCompanyId;
- sECCompany.sCompanyId.lpb = reinterpret_cast<unsigned char *>(lpCompanyId.get());
- hr = lpServiceAdmin->SetCompany(&sECCompany, 0);
- if (hr != hrSuccess) {
- cerr << "Failed to set company system administrator" << endl;
- goto exit;
- }
- cout << "User " << username << " is set as admin of company " << companyname << endl;
- break;
- case MODE_ADD_USERQUOTA_RECIPIENT:
- hr = lpServiceAdmin->AddQuotaRecipient(cbCompanyId, lpCompanyId, cbUserId, lpUserId, ACTIVE_USER);
- if (hr != hrSuccess) {
- cerr << "Failed to add recipient to quota list." << endl;
- goto exit;
- }
- cout << "User " << username << " added to user quota recipients list for company " << companyname << endl;
- break;
- case MODE_DEL_USERQUOTA_RECIPIENT:
- hr = lpServiceAdmin->DeleteQuotaRecipient(cbCompanyId, lpCompanyId, cbUserId, lpUserId, ACTIVE_USER);
- if (hr != hrSuccess) {
- cerr << "Failed to remove company from quota list." << endl;
- goto exit;
- }
- cout << "User " << username << " removed from user quota recipients list for company " << companyname << endl;
- break;
- case MODE_LIST_USERQUOTA_RECIPIENT:
- /* HACK: request a user from the specified company, and request the recipients for that user. */
- hr = lpServiceAdmin->GetUserList(cbCompanyId, lpCompanyId, 0, &cUsers, &~lpECUser);
- if (hr != hrSuccess || cUsers <= 1) /* First user is always SYSTEM */ {
- cerr << "Failed to get quota recipient list" << endl;
- goto exit;
- }
- cbUserId = lpECUser[1].sUserId.cb;
- hr = MAPIAllocateBuffer(cbUserId, &~lpUserId);
- if (hr != hrSuccess)
- goto exit;
- memcpy(lpUserId, lpECUser[1].sUserId.lpb, cbUserId);
- hr = lpServiceAdmin->GetQuotaRecipients(cbUserId, lpUserId, 0, &cUsers, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Failed to get quota recipient list" << endl;
- goto exit;
- }
- cout << "Recipient list ("<< cUsers-1 <<"):" << endl;
- /* Skip the dummy entry we used to obtain the list */
- print_users(cUsers - 1, &lpECUser[1]);
- break;
- case MODE_ADD_COMPANYQUOTA_RECIPIENT:
- hr = lpServiceAdmin->AddQuotaRecipient(cbCompanyId, lpCompanyId, cbUserId, lpUserId, CONTAINER_COMPANY);
- if (hr != hrSuccess) {
- cerr << "Failed to add recipient to quota list." << endl;
- goto exit;
- }
- cout << "User " << username << " added to company quota recipients list for company " << companyname << endl;
- break;
- case MODE_DEL_COMPANYQUOTA_RECIPIENT:
- hr = lpServiceAdmin->DeleteQuotaRecipient(cbCompanyId, lpCompanyId, cbUserId, lpUserId, CONTAINER_COMPANY);
- if (hr != hrSuccess) {
- cerr << "Failed to delete recipient to quota list." << endl;
- goto exit;
- }
- cout << "User " << username << " removed from company quota recipients list for company " << companyname << endl;
- break;
- case MODE_LIST_COMPANYQUOTA_RECIPIENT:
- hr = lpServiceAdmin->GetQuotaRecipients(cbCompanyId, lpCompanyId, 0, &cUsers, &~lpECUser);
- if (hr != hrSuccess) {
- cerr << "Failed to get quota recipient list." << endl;
- goto exit;
- }
- cout << "Recipient list ("<< cUsers-1 <<"):" << endl;
- /* Skipt first entry, that is the company itself which will not get the mail */
- print_users(cUsers - 1, &lpECUser[1]);
- break;
- case MODE_LIST_SENDAS:
- if (detailstype == NULL || strcasecmp(detailstype, "user") == 0) {
- hr = lpServiceAdmin->ResolveUserName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- detailstype = "user";
- } else if (strcasecmp(detailstype, "group") == 0) {
- hr = lpServiceAdmin->ResolveGroupName(reinterpret_cast<LPTSTR>(username), 0, &cbUserId, &~lpUserId);
- } else {
- hr = MAPI_E_INVALID_TYPE;
- cerr << "Unknown object type \"" << detailstype << "\"" << endl;
- goto exit;
- }
- if (hr != hrSuccess) {
- cerr << "Failed to resolve "<< detailstype <<" name, " << getMapiCodeString(hr, username) << endl;
- goto exit;
- }
- hr = lpServiceAdmin->GetSendAsList(cbUserId, lpUserId, 0, &cSenders, &~lpSenders);
- if (hr != hrSuccess) {
- cerr << "Failed to retrieve send-as list for " << detailstype << " " << username << endl;
- goto exit;
- }
- cout << "Send-as list ("<< cSenders <<") for " << detailstype << " " << username << ":" << endl;
- print_users(cSenders, lpSenders);
- break;
- case MODE_PURGE_SOFTDELETE:
- hr = lpServiceAdmin->PurgeSoftDelete(ulDays);
- if (hr != hrSuccess) {
- cerr << "Softdelete purge failed" << endl;
- goto exit;
- }
- cout << "Softdelete purge done." << endl;
- break;
- case MODE_CLEAR_CACHE:
- hr = lpServiceAdmin->PurgeCache(ulCachePurgeMode);
- if (hr != hrSuccess) {
- cerr << "Cache clear failed" << endl;
- goto exit;
- }
- if (ulCachePurgeMode != PURGE_CACHE_ALL)
- cout << "Cache cleared with flags " << ulCachePurgeMode << endl;
- else
- cout << "Cache cleared." << endl;
- break;
- case MODE_PURGE_DEFERRED:
- while(1) {
- ULONG ulRemaining;
- hr = lpServiceAdmin->PurgeDeferredUpdates(&ulRemaining);
- if(hr == MAPI_E_NOT_FOUND)
- break;
- if(hr != hrSuccess) {
- cerr << "Purge failed." << endl;
- break;
- }
- cerr << "Remaining deferred records: " << ulRemaining << " \r";
- }
- cerr << endl;
- cerr << "Done." << endl;
- break;
- case MODE_FORCE_RESYNC:
- if (lstUsernames.empty())
- hr = ForceResyncAll(lpSession, lpMsgStore);
- else
- hr = ForceResync(lpSession, lpMsgStore, lstUsernames);
- if (hr != hrSuccess) {
- cerr << "Failed to force resync." << endl;
- goto exit;
- }
- cerr << "Successfully forced resync." << endl;
- break;
- case MODE_USER_COUNT:
- hr = DisplayUserCount(lpMsgStore);
- if (hr != hrSuccess) {
- cerr << "Failed to get user statistics." << endl;
- goto exit;
- }
- break;
- case MODE_RESET_FOLDER_COUNT:
- hr = ResetFolderCount(lpSession, lpMsgStore, username);
- if (FAILED(hr)) {
- cerr << "Failed to reset folder counters." << endl;
- goto exit;
- } else if (hr != hrSuccess) {
- cerr << "Some folder counters could not be reset." << endl;
- } else {
- cerr << "Successfully reset folder counters." << endl;
- }
- case MODE_INVALID:
- case MODE_HELP:
- // happy compiler
- break;
- };
- exit:
- SSL_library_cleanup();
- if (forcedExitCode > 0)
- return forcedExitCode;
- if (hr == hrSuccess)
- return 0;
- cerr << "Using the -v option (possibly multiple times) may "
- << "give more hints." << endl;
- return 1;
- }
|