123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984 |
- /**
- * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
- * SPDX-License-Identifier: Apache-2.0.
- */
- #include <aws/mqtt/private/v5/mqtt5_options_storage.h>
- #include <aws/common/clock.h>
- #include <aws/common/encoding.h>
- #include <aws/common/string.h>
- #include <aws/io/channel_bootstrap.h>
- #include <aws/io/event_loop.h>
- #include <aws/io/stream.h>
- #include <aws/mqtt/private/v5/mqtt5_client_impl.h>
- #include <aws/mqtt/private/v5/mqtt5_utils.h>
- #include <aws/mqtt/v5/mqtt5_client.h>
- #include <inttypes.h>
- /*********************************************************************************************************************
- * Property set
- ********************************************************************************************************************/
- int aws_mqtt5_user_property_set_init(
- struct aws_mqtt5_user_property_set *property_set,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*property_set);
- if (aws_array_list_init_dynamic(&property_set->properties, allocator, 0, sizeof(struct aws_mqtt5_user_property))) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_user_property_set_init_with_storage(
- struct aws_mqtt5_user_property_set *property_set,
- struct aws_allocator *allocator,
- struct aws_byte_buf *storage,
- size_t property_count,
- const struct aws_mqtt5_user_property *properties) {
- AWS_ZERO_STRUCT(*property_set);
- if (aws_array_list_init_dynamic(
- &property_set->properties, allocator, property_count, sizeof(struct aws_mqtt5_user_property))) {
- goto error;
- }
- for (size_t i = 0; i < property_count; ++i) {
- const struct aws_mqtt5_user_property *property = &properties[i];
- struct aws_mqtt5_user_property property_clone = *property;
- if (aws_byte_buf_append_and_update(storage, &property_clone.name)) {
- goto error;
- }
- if (aws_byte_buf_append_and_update(storage, &property_clone.value)) {
- goto error;
- }
- if (aws_array_list_push_back(&property_set->properties, &property_clone)) {
- goto error;
- }
- }
- return AWS_OP_SUCCESS;
- error:
- aws_mqtt5_user_property_set_clean_up(property_set);
- return AWS_OP_ERR;
- }
- void aws_mqtt5_user_property_set_clean_up(struct aws_mqtt5_user_property_set *property_set) {
- aws_array_list_clean_up(&property_set->properties);
- }
- size_t aws_mqtt5_user_property_set_size(const struct aws_mqtt5_user_property_set *property_set) {
- return aws_array_list_length(&property_set->properties);
- }
- int aws_mqtt5_user_property_set_get_property(
- const struct aws_mqtt5_user_property_set *property_set,
- size_t index,
- struct aws_mqtt5_user_property *property_out) {
- return aws_array_list_get_at(&property_set->properties, property_out, index);
- }
- int aws_mqtt5_user_property_set_add_stored_property(
- struct aws_mqtt5_user_property_set *property_set,
- struct aws_mqtt5_user_property *property) {
- return aws_array_list_push_back(&property_set->properties, property);
- }
- static void s_aws_mqtt5_user_property_set_log(
- struct aws_logger *log_handle,
- const struct aws_mqtt5_user_property *properties,
- size_t property_count,
- void *log_context,
- enum aws_log_level level,
- const char *log_prefix) {
- if (property_count == 0) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s with %zu user properties:",
- log_context,
- log_prefix,
- property_count);
- for (size_t i = 0; i < property_count; ++i) {
- const struct aws_mqtt5_user_property *property = &properties[i];
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s user property %zu with name \"" PRInSTR "\", value \"" PRInSTR "\"",
- log_context,
- log_prefix,
- i,
- AWS_BYTE_CURSOR_PRI(property->name),
- AWS_BYTE_CURSOR_PRI(property->value));
- }
- }
- static size_t s_aws_mqtt5_user_property_set_compute_storage_size(
- const struct aws_mqtt5_user_property *properties,
- size_t property_count) {
- size_t storage_size = 0;
- for (size_t i = 0; i < property_count; ++i) {
- const struct aws_mqtt5_user_property *property = &properties[i];
- storage_size += property->name.len;
- storage_size += property->value.len;
- }
- return storage_size;
- }
- static int s_aws_mqtt5_user_property_set_validate(
- const struct aws_mqtt5_user_property *properties,
- size_t property_count,
- const char *log_prefix,
- void *log_context) {
- if (properties == NULL) {
- if (property_count == 0) {
- return AWS_OP_SUCCESS;
- } else {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s - Invalid user property configuration, null properties, non-zero property count",
- log_context,
- log_prefix);
- return aws_raise_error(AWS_ERROR_MQTT5_USER_PROPERTY_VALIDATION);
- }
- }
- if (property_count > AWS_MQTT5_CLIENT_MAXIMUM_USER_PROPERTIES) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s - user property limit (%d) exceeded (%zu)",
- log_context,
- log_prefix,
- (int)AWS_MQTT5_CLIENT_MAXIMUM_USER_PROPERTIES,
- property_count);
- return aws_raise_error(AWS_ERROR_MQTT5_USER_PROPERTY_VALIDATION);
- }
- for (size_t i = 0; i < property_count; ++i) {
- const struct aws_mqtt5_user_property *property = &properties[i];
- if (property->name.len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s - user property #%zu name too long (%zu)",
- log_context,
- log_prefix,
- i,
- property->name.len);
- return aws_raise_error(AWS_ERROR_MQTT5_USER_PROPERTY_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(property->name)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL, "id=%p: %s - user property #%zu name not valid UTF8", log_context, log_prefix, i);
- return aws_raise_error(AWS_ERROR_MQTT5_USER_PROPERTY_VALIDATION);
- }
- if (property->value.len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s - user property #%zu value too long (%zu)",
- log_context,
- log_prefix,
- i,
- property->value.len);
- return aws_raise_error(AWS_ERROR_MQTT5_USER_PROPERTY_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(property->value)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: %s - user property #%zu value not valid UTF8",
- log_context,
- log_prefix,
- i);
- return aws_raise_error(AWS_ERROR_MQTT5_USER_PROPERTY_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- /*********************************************************************************************************************
- * Operation base
- ********************************************************************************************************************/
- struct aws_mqtt5_operation *aws_mqtt5_operation_acquire(struct aws_mqtt5_operation *operation) {
- if (operation == NULL) {
- return NULL;
- }
- aws_ref_count_acquire(&operation->ref_count);
- return operation;
- }
- struct aws_mqtt5_operation *aws_mqtt5_operation_release(struct aws_mqtt5_operation *operation) {
- if (operation != NULL) {
- aws_ref_count_release(&operation->ref_count);
- }
- return NULL;
- }
- void aws_mqtt5_operation_complete(
- struct aws_mqtt5_operation *operation,
- int error_code,
- enum aws_mqtt5_packet_type packet_type,
- const void *associated_view) {
- AWS_FATAL_ASSERT(operation->vtable != NULL);
- if (operation->vtable->aws_mqtt5_operation_completion_fn != NULL) {
- (*operation->vtable->aws_mqtt5_operation_completion_fn)(operation, error_code, packet_type, associated_view);
- }
- }
- void aws_mqtt5_operation_set_packet_id(struct aws_mqtt5_operation *operation, aws_mqtt5_packet_id_t packet_id) {
- AWS_FATAL_ASSERT(operation->vtable != NULL);
- if (operation->vtable->aws_mqtt5_operation_set_packet_id_fn != NULL) {
- (*operation->vtable->aws_mqtt5_operation_set_packet_id_fn)(operation, packet_id);
- }
- }
- aws_mqtt5_packet_id_t aws_mqtt5_operation_get_packet_id(const struct aws_mqtt5_operation *operation) {
- AWS_FATAL_ASSERT(operation->vtable != NULL);
- if (operation->vtable->aws_mqtt5_operation_get_packet_id_address_fn != NULL) {
- aws_mqtt5_packet_id_t *packet_id_ptr =
- (*operation->vtable->aws_mqtt5_operation_get_packet_id_address_fn)(operation);
- if (packet_id_ptr != NULL) {
- return *packet_id_ptr;
- }
- }
- return 0;
- }
- aws_mqtt5_packet_id_t *aws_mqtt5_operation_get_packet_id_address(const struct aws_mqtt5_operation *operation) {
- AWS_FATAL_ASSERT(operation->vtable != NULL);
- if (operation->vtable->aws_mqtt5_operation_get_packet_id_address_fn != NULL) {
- return (*operation->vtable->aws_mqtt5_operation_get_packet_id_address_fn)(operation);
- }
- return NULL;
- }
- int aws_mqtt5_operation_validate_vs_connection_settings(
- const struct aws_mqtt5_operation *operation,
- const struct aws_mqtt5_client *client) {
- AWS_FATAL_ASSERT(operation->vtable != NULL);
- AWS_FATAL_ASSERT(client->loop == NULL || aws_event_loop_thread_is_callers_thread(client->loop));
- /* If we have valid negotiated settings, check against them as well */
- if (aws_mqtt5_client_are_negotiated_settings_valid(client)) {
- const struct aws_mqtt5_negotiated_settings *settings = &client->negotiated_settings;
- size_t packet_size_in_bytes = 0;
- if (aws_mqtt5_packet_view_get_encoded_size(
- operation->packet_type, operation->packet_view, &packet_size_in_bytes)) {
- int error_code = aws_last_error();
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_CLIENT,
- "id=%p: error %d (%s) computing %s packet size",
- (void *)client,
- error_code,
- aws_error_debug_str(error_code),
- aws_mqtt5_packet_type_to_c_string(operation->packet_type));
- return aws_raise_error(AWS_ERROR_MQTT5_PACKET_VALIDATION);
- }
- if (packet_size_in_bytes > settings->maximum_packet_size_to_server) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_CLIENT,
- "id=%p: encoded %s packet size (%zu) exceeds server's maximum "
- "packet size (%" PRIu32 ")",
- (void *)client,
- aws_mqtt5_packet_type_to_c_string(operation->packet_type),
- packet_size_in_bytes,
- settings->maximum_packet_size_to_server);
- return aws_raise_error(AWS_ERROR_MQTT5_PACKET_VALIDATION);
- }
- }
- if (operation->vtable->aws_mqtt5_operation_validate_vs_connection_settings_fn != NULL) {
- return (*operation->vtable->aws_mqtt5_operation_validate_vs_connection_settings_fn)(
- operation->packet_view, client);
- }
- return AWS_OP_SUCCESS;
- }
- static struct aws_mqtt5_operation_vtable s_empty_operation_vtable = {
- .aws_mqtt5_operation_completion_fn = NULL,
- .aws_mqtt5_operation_set_packet_id_fn = NULL,
- .aws_mqtt5_operation_get_packet_id_address_fn = NULL,
- .aws_mqtt5_operation_validate_vs_connection_settings_fn = NULL,
- };
- /*********************************************************************************************************************
- * Connect
- ********************************************************************************************************************/
- int aws_mqtt5_packet_connect_view_validate(const struct aws_mqtt5_packet_connect_view *connect_options) {
- if (connect_options == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "Null CONNECT options");
- return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
- }
- if (connect_options->client_id.len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL, "id=%p: aws_mqtt5_packet_connect_view - client id too long", (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(connect_options->client_id)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - client id not valid UTF-8",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- if (connect_options->username != NULL) {
- if (connect_options->username->len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - username too long",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(*connect_options->username)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - username not valid UTF-8",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (connect_options->password != NULL) {
- if (connect_options->password->len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - password too long",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (connect_options->receive_maximum != NULL) {
- if (*connect_options->receive_maximum == 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - receive maximum property of CONNECT packet may not be zero.",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (connect_options->maximum_packet_size_bytes != NULL) {
- if (*connect_options->maximum_packet_size_bytes == 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - maximum packet size property of CONNECT packet may not be "
- "zero.",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (connect_options->will != NULL) {
- const struct aws_mqtt5_packet_publish_view *will_options = connect_options->will;
- if (aws_mqtt5_packet_publish_view_validate(will_options)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - CONNECT packet Will message failed validation",
- (void *)connect_options);
- return AWS_OP_ERR;
- }
- if (will_options->payload.len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - will payload larger than %d",
- (void *)connect_options,
- (int)UINT16_MAX);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (connect_options->request_problem_information != NULL) {
- if (*connect_options->request_problem_information > 1) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - CONNECT packet request problem information has invalid value",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (connect_options->request_response_information != NULL) {
- if (*connect_options->request_response_information > 1) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - CONNECT packet request response information has invalid value",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- }
- }
- if (s_aws_mqtt5_user_property_set_validate(
- connect_options->user_properties,
- connect_options->user_property_count,
- "aws_mqtt5_packet_connect_view",
- (void *)connect_options)) {
- return AWS_OP_ERR;
- }
- if (connect_options->authentication_method != NULL || connect_options->authentication_data != NULL) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view - CONNECT packet has unsupported authentication fields set.",
- (void *)connect_options);
- return aws_raise_error(AWS_ERROR_MQTT5_CONNECT_OPTIONS_VALIDATION);
- // TODO: UTF-8 validation for authentication_method once supported.
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_connect_view_log(
- const struct aws_mqtt5_packet_connect_view *connect_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view keep alive interval set to %" PRIu16,
- (void *)connect_view,
- connect_view->keep_alive_interval_seconds);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view client id set to \"" PRInSTR "\"",
- (void *)connect_view,
- AWS_BYTE_CURSOR_PRI(connect_view->client_id));
- if (connect_view->username != NULL) {
- /* Intentionally do not log username since it too can contain sensitive information */
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view username set",
- (void *)connect_view);
- }
- if (connect_view->password != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view password set",
- (void *)connect_view);
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view clean start set to %d",
- (void *)connect_view,
- (int)(connect_view->clean_start ? 1 : 0));
- if (connect_view->session_expiry_interval_seconds != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view session expiry interval set to %" PRIu32,
- (void *)connect_view,
- *connect_view->session_expiry_interval_seconds);
- }
- if (connect_view->request_response_information != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view request response information set to %d",
- (void *)connect_view,
- (int)*connect_view->request_response_information);
- }
- if (connect_view->request_problem_information) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view request problem information set to %d",
- (void *)connect_view,
- (int)*connect_view->request_problem_information);
- }
- if (connect_view->receive_maximum != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view receive maximum set to %" PRIu16,
- (void *)connect_view,
- *connect_view->receive_maximum);
- }
- if (connect_view->topic_alias_maximum != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view topic alias maximum set to %" PRIu16,
- (void *)connect_view,
- *connect_view->topic_alias_maximum);
- }
- if (connect_view->maximum_packet_size_bytes != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view maximum packet size set to %" PRIu32,
- (void *)connect_view,
- *connect_view->maximum_packet_size_bytes);
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view set will to (%p)",
- (void *)connect_view,
- (void *)connect_view->will);
- if (connect_view->will != NULL) {
- aws_mqtt5_packet_publish_view_log(connect_view->will, level);
- }
- if (connect_view->will_delay_interval_seconds != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view will delay interval set to %" PRIu32,
- (void *)connect_view,
- *connect_view->will_delay_interval_seconds);
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- connect_view->user_properties,
- connect_view->user_property_count,
- (void *)connect_view,
- level,
- "aws_mqtt5_packet_connect_view");
- if (connect_view->authentication_method != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view authentication method set",
- (void *)connect_view);
- }
- if (connect_view->password != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connect_view authentication data set",
- (void *)connect_view);
- }
- }
- void aws_mqtt5_packet_connect_storage_clean_up(struct aws_mqtt5_packet_connect_storage *storage) {
- if (storage == NULL) {
- return;
- }
- if (storage->will != NULL) {
- aws_mqtt5_packet_publish_storage_clean_up(storage->will);
- aws_mem_release(storage->allocator, storage->will);
- }
- aws_mqtt5_user_property_set_clean_up(&storage->user_properties);
- aws_byte_buf_clean_up_secure(&storage->storage);
- }
- static size_t s_aws_mqtt5_packet_connect_compute_storage_size(const struct aws_mqtt5_packet_connect_view *view) {
- if (view == NULL) {
- return 0;
- }
- size_t storage_size = 0;
- storage_size += view->client_id.len;
- if (view->username != NULL) {
- storage_size += view->username->len;
- }
- if (view->password != NULL) {
- storage_size += view->password->len;
- }
- storage_size +=
- s_aws_mqtt5_user_property_set_compute_storage_size(view->user_properties, view->user_property_count);
- if (view->authentication_method != NULL) {
- storage_size += view->authentication_method->len;
- }
- if (view->authentication_data != NULL) {
- storage_size += view->authentication_data->len;
- }
- return storage_size;
- }
- int aws_mqtt5_packet_connect_storage_init(
- struct aws_mqtt5_packet_connect_storage *storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_connect_view *view) {
- AWS_ZERO_STRUCT(*storage);
- struct aws_mqtt5_packet_connect_view *storage_view = &storage->storage_view;
- size_t storage_capacity = s_aws_mqtt5_packet_connect_compute_storage_size(view);
- if (aws_byte_buf_init(&storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- storage->allocator = allocator;
- storage_view->keep_alive_interval_seconds = view->keep_alive_interval_seconds;
- storage_view->client_id = view->client_id;
- if (aws_byte_buf_append_and_update(&storage->storage, &storage_view->client_id)) {
- return AWS_OP_ERR;
- }
- if (view->username != NULL) {
- storage->username = *view->username;
- if (aws_byte_buf_append_and_update(&storage->storage, &storage->username)) {
- return AWS_OP_ERR;
- }
- storage_view->username = &storage->username;
- }
- if (view->password != NULL) {
- storage->password = *view->password;
- if (aws_byte_buf_append_and_update(&storage->storage, &storage->password)) {
- return AWS_OP_ERR;
- }
- storage_view->password = &storage->password;
- }
- storage_view->clean_start = view->clean_start;
- if (view->session_expiry_interval_seconds != NULL) {
- storage->session_expiry_interval_seconds = *view->session_expiry_interval_seconds;
- storage_view->session_expiry_interval_seconds = &storage->session_expiry_interval_seconds;
- }
- if (view->request_response_information != NULL) {
- storage->request_response_information = *view->request_response_information;
- storage_view->request_response_information = &storage->request_response_information;
- }
- if (view->request_problem_information != NULL) {
- storage->request_problem_information = *view->request_problem_information;
- storage_view->request_problem_information = &storage->request_problem_information;
- }
- if (view->receive_maximum != NULL) {
- storage->receive_maximum = *view->receive_maximum;
- storage_view->receive_maximum = &storage->receive_maximum;
- }
- if (view->topic_alias_maximum != NULL) {
- storage->topic_alias_maximum = *view->topic_alias_maximum;
- storage_view->topic_alias_maximum = &storage->topic_alias_maximum;
- }
- if (view->maximum_packet_size_bytes != NULL) {
- storage->maximum_packet_size_bytes = *view->maximum_packet_size_bytes;
- storage_view->maximum_packet_size_bytes = &storage->maximum_packet_size_bytes;
- }
- if (view->will != NULL) {
- storage->will = aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_packet_publish_storage));
- if (storage->will == NULL) {
- return AWS_OP_ERR;
- }
- if (aws_mqtt5_packet_publish_storage_init(storage->will, allocator, view->will)) {
- return AWS_OP_ERR;
- }
- storage_view->will = &storage->will->storage_view;
- }
- if (view->will_delay_interval_seconds != 0) {
- storage->will_delay_interval_seconds = *view->will_delay_interval_seconds;
- storage_view->will_delay_interval_seconds = &storage->will_delay_interval_seconds;
- }
- if (aws_mqtt5_user_property_set_init_with_storage(
- &storage->user_properties,
- allocator,
- &storage->storage,
- view->user_property_count,
- view->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&storage->user_properties);
- storage_view->user_properties = storage->user_properties.properties.data;
- if (view->authentication_method != NULL) {
- storage->authentication_method = *view->authentication_method;
- if (aws_byte_buf_append_and_update(&storage->storage, &storage->authentication_method)) {
- return AWS_OP_ERR;
- }
- storage_view->authentication_method = &storage->authentication_method;
- }
- if (view->authentication_data != NULL) {
- storage->authentication_data = *view->authentication_data;
- if (aws_byte_buf_append_and_update(&storage->storage, &storage->authentication_data)) {
- return AWS_OP_ERR;
- }
- storage_view->authentication_data = &storage->authentication_data;
- }
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_connect_storage_init_from_external_storage(
- struct aws_mqtt5_packet_connect_storage *connect_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*connect_storage);
- if (aws_mqtt5_user_property_set_init(&connect_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- static void s_destroy_operation_connect(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_connect *connect_op = object;
- aws_mqtt5_packet_connect_storage_clean_up(&connect_op->options_storage);
- aws_mem_release(connect_op->allocator, connect_op);
- }
- struct aws_mqtt5_operation_connect *aws_mqtt5_operation_connect_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_connect_view *connect_options) {
- AWS_PRECONDITION(allocator != NULL);
- AWS_PRECONDITION(connect_options != NULL);
- if (aws_mqtt5_packet_connect_view_validate(connect_options)) {
- return NULL;
- }
- struct aws_mqtt5_operation_connect *connect_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_connect));
- if (connect_op == NULL) {
- return NULL;
- }
- connect_op->allocator = allocator;
- connect_op->base.vtable = &s_empty_operation_vtable;
- connect_op->base.packet_type = AWS_MQTT5_PT_CONNECT;
- aws_ref_count_init(&connect_op->base.ref_count, connect_op, s_destroy_operation_connect);
- connect_op->base.impl = connect_op;
- if (aws_mqtt5_packet_connect_storage_init(&connect_op->options_storage, allocator, connect_options)) {
- goto error;
- }
- connect_op->base.packet_view = &connect_op->options_storage.storage_view;
- return connect_op;
- error:
- aws_mqtt5_operation_release(&connect_op->base);
- return NULL;
- }
- /*********************************************************************************************************************
- * Connack
- ********************************************************************************************************************/
- static size_t s_aws_mqtt5_packet_connack_compute_storage_size(const struct aws_mqtt5_packet_connack_view *view) {
- if (view == NULL) {
- return 0;
- }
- size_t storage_size = 0;
- if (view->assigned_client_identifier != NULL) {
- storage_size += view->assigned_client_identifier->len;
- }
- if (view->reason_string != NULL) {
- storage_size += view->reason_string->len;
- }
- if (view->response_information != NULL) {
- storage_size += view->response_information->len;
- }
- if (view->server_reference != NULL) {
- storage_size += view->server_reference->len;
- }
- if (view->authentication_method != NULL) {
- storage_size += view->authentication_method->len;
- }
- if (view->authentication_data != NULL) {
- storage_size += view->authentication_data->len;
- }
- storage_size +=
- s_aws_mqtt5_user_property_set_compute_storage_size(view->user_properties, view->user_property_count);
- return storage_size;
- }
- int aws_mqtt5_packet_connack_storage_init(
- struct aws_mqtt5_packet_connack_storage *connack_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_connack_view *connack_view) {
- AWS_ZERO_STRUCT(*connack_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_connack_compute_storage_size(connack_view);
- if (aws_byte_buf_init(&connack_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_connack_view *stored_view = &connack_storage->storage_view;
- connack_storage->allocator = allocator;
- stored_view->session_present = connack_view->session_present;
- stored_view->reason_code = connack_view->reason_code;
- if (connack_view->session_expiry_interval != NULL) {
- connack_storage->session_expiry_interval = *connack_view->session_expiry_interval;
- stored_view->session_expiry_interval = &connack_storage->session_expiry_interval;
- }
- if (connack_view->receive_maximum != NULL) {
- connack_storage->receive_maximum = *connack_view->receive_maximum;
- stored_view->receive_maximum = &connack_storage->receive_maximum;
- }
- if (connack_view->maximum_qos != NULL) {
- connack_storage->maximum_qos = *connack_view->maximum_qos;
- stored_view->maximum_qos = &connack_storage->maximum_qos;
- }
- if (connack_view->retain_available != NULL) {
- connack_storage->retain_available = *connack_view->retain_available;
- stored_view->retain_available = &connack_storage->retain_available;
- }
- if (connack_view->maximum_packet_size != NULL) {
- connack_storage->maximum_packet_size = *connack_view->maximum_packet_size;
- stored_view->maximum_packet_size = &connack_storage->maximum_packet_size;
- }
- if (connack_view->assigned_client_identifier != NULL) {
- connack_storage->assigned_client_identifier = *connack_view->assigned_client_identifier;
- if (aws_byte_buf_append_and_update(&connack_storage->storage, &connack_storage->assigned_client_identifier)) {
- return AWS_OP_ERR;
- }
- stored_view->assigned_client_identifier = &connack_storage->assigned_client_identifier;
- }
- if (connack_view->topic_alias_maximum != NULL) {
- connack_storage->topic_alias_maximum = *connack_view->topic_alias_maximum;
- stored_view->topic_alias_maximum = &connack_storage->topic_alias_maximum;
- }
- if (connack_view->reason_string != NULL) {
- connack_storage->reason_string = *connack_view->reason_string;
- if (aws_byte_buf_append_and_update(&connack_storage->storage, &connack_storage->reason_string)) {
- return AWS_OP_ERR;
- }
- stored_view->reason_string = &connack_storage->reason_string;
- }
- if (connack_view->wildcard_subscriptions_available != NULL) {
- connack_storage->wildcard_subscriptions_available = *connack_view->wildcard_subscriptions_available;
- stored_view->wildcard_subscriptions_available = &connack_storage->wildcard_subscriptions_available;
- }
- if (connack_view->subscription_identifiers_available != NULL) {
- connack_storage->subscription_identifiers_available = *connack_view->subscription_identifiers_available;
- stored_view->subscription_identifiers_available = &connack_storage->subscription_identifiers_available;
- }
- if (connack_view->shared_subscriptions_available != NULL) {
- connack_storage->shared_subscriptions_available = *connack_view->shared_subscriptions_available;
- stored_view->shared_subscriptions_available = &connack_storage->shared_subscriptions_available;
- }
- if (connack_view->server_keep_alive != NULL) {
- connack_storage->server_keep_alive = *connack_view->server_keep_alive;
- stored_view->server_keep_alive = &connack_storage->server_keep_alive;
- }
- if (connack_view->response_information != NULL) {
- connack_storage->response_information = *connack_view->response_information;
- if (aws_byte_buf_append_and_update(&connack_storage->storage, &connack_storage->response_information)) {
- return AWS_OP_ERR;
- }
- stored_view->response_information = &connack_storage->response_information;
- }
- if (connack_view->server_reference != NULL) {
- connack_storage->server_reference = *connack_view->server_reference;
- if (aws_byte_buf_append_and_update(&connack_storage->storage, &connack_storage->server_reference)) {
- return AWS_OP_ERR;
- }
- stored_view->server_reference = &connack_storage->server_reference;
- }
- if (connack_view->authentication_method != NULL) {
- connack_storage->authentication_method = *connack_view->authentication_method;
- if (aws_byte_buf_append_and_update(&connack_storage->storage, &connack_storage->authentication_method)) {
- return AWS_OP_ERR;
- }
- stored_view->authentication_method = &connack_storage->authentication_method;
- }
- if (connack_view->authentication_data != NULL) {
- connack_storage->authentication_data = *connack_view->authentication_data;
- if (aws_byte_buf_append_and_update(&connack_storage->storage, &connack_storage->authentication_data)) {
- return AWS_OP_ERR;
- }
- stored_view->authentication_data = &connack_storage->authentication_data;
- }
- if (aws_mqtt5_user_property_set_init_with_storage(
- &connack_storage->user_properties,
- allocator,
- &connack_storage->storage,
- connack_view->user_property_count,
- connack_view->user_properties)) {
- return AWS_OP_ERR;
- }
- stored_view->user_property_count = aws_mqtt5_user_property_set_size(&connack_storage->user_properties);
- stored_view->user_properties = connack_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_connack_storage_init_from_external_storage(
- struct aws_mqtt5_packet_connack_storage *connack_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*connack_storage);
- if (aws_mqtt5_user_property_set_init(&connack_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_connack_storage_clean_up(struct aws_mqtt5_packet_connack_storage *connack_storage) {
- if (connack_storage == NULL) {
- return;
- }
- aws_mqtt5_user_property_set_clean_up(&connack_storage->user_properties);
- aws_byte_buf_clean_up(&connack_storage->storage);
- }
- void aws_mqtt5_packet_connack_view_log(
- const struct aws_mqtt5_packet_connack_view *connack_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view reason code set to %d (%s)",
- (void *)connack_view,
- (int)connack_view->reason_code,
- aws_mqtt5_connect_reason_code_to_c_string(connack_view->reason_code));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view session present set to %d",
- (void *)connack_view,
- (int)connack_view->session_present);
- if (connack_view->session_expiry_interval != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view session expiry interval set to %" PRIu32,
- (void *)connack_view,
- *connack_view->session_expiry_interval);
- }
- if (connack_view->receive_maximum != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view receive maximum set to %" PRIu16,
- (void *)connack_view,
- *connack_view->receive_maximum);
- }
- if (connack_view->maximum_qos != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view maximum qos set to %d",
- (void *)connack_view,
- (int)(*connack_view->maximum_qos));
- }
- if (connack_view->retain_available != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view retain available set to %d",
- (void *)connack_view,
- (int)(*connack_view->retain_available));
- }
- if (connack_view->maximum_packet_size != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view maximum packet size set to %" PRIu32,
- (void *)connack_view,
- *connack_view->maximum_packet_size);
- }
- if (connack_view->assigned_client_identifier != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view assigned client identifier set to \"" PRInSTR "\"",
- (void *)connack_view,
- AWS_BYTE_CURSOR_PRI(*connack_view->assigned_client_identifier));
- }
- if (connack_view->topic_alias_maximum != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view topic alias maximum set to %" PRIu16,
- (void *)connack_view,
- *connack_view->topic_alias_maximum);
- }
- if (connack_view->reason_string != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view reason string set to \"" PRInSTR "\"",
- (void *)connack_view,
- AWS_BYTE_CURSOR_PRI(*connack_view->reason_string));
- }
- if (connack_view->wildcard_subscriptions_available != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view wildcard subscriptions available set to %d",
- (void *)connack_view,
- (int)(*connack_view->wildcard_subscriptions_available));
- }
- if (connack_view->subscription_identifiers_available != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view subscription identifiers available set to %d",
- (void *)connack_view,
- (int)(*connack_view->subscription_identifiers_available));
- }
- if (connack_view->shared_subscriptions_available != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view shared subscriptions available set to %d",
- (void *)connack_view,
- (int)(*connack_view->shared_subscriptions_available));
- }
- if (connack_view->server_keep_alive != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view server keep alive set to %" PRIu16,
- (void *)connack_view,
- *connack_view->server_keep_alive);
- }
- if (connack_view->response_information != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view response information set to \"" PRInSTR "\"",
- (void *)connack_view,
- AWS_BYTE_CURSOR_PRI(*connack_view->response_information));
- }
- if (connack_view->server_reference != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view server reference set to \"" PRInSTR "\"",
- (void *)connack_view,
- AWS_BYTE_CURSOR_PRI(*connack_view->server_reference));
- }
- if (connack_view->authentication_method != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view authentication method set",
- (void *)connack_view);
- }
- if (connack_view->authentication_data != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_connack_view authentication data set",
- (void *)connack_view);
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- connack_view->user_properties,
- connack_view->user_property_count,
- (void *)connack_view,
- level,
- "aws_mqtt5_packet_connack_view");
- }
- /*********************************************************************************************************************
- * Disconnect
- ********************************************************************************************************************/
- int aws_mqtt5_packet_disconnect_view_validate(const struct aws_mqtt5_packet_disconnect_view *disconnect_view) {
- if (disconnect_view == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "null DISCONNECT packet options");
- return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
- }
- bool is_valid_reason_code = true;
- aws_mqtt5_disconnect_reason_code_to_c_string(disconnect_view->reason_code, &is_valid_reason_code);
- if (!is_valid_reason_code) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view - invalid DISCONNECT reason code:%d",
- (void *)disconnect_view,
- (int)disconnect_view->reason_code);
- return aws_raise_error(AWS_ERROR_MQTT5_DISCONNECT_OPTIONS_VALIDATION);
- }
- if (disconnect_view->reason_string != NULL) {
- if (disconnect_view->reason_string->len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view - reason string too long",
- (void *)disconnect_view);
- return aws_raise_error(AWS_ERROR_MQTT5_DISCONNECT_OPTIONS_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(*disconnect_view->reason_string)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view - reason string not valid UTF-8",
- (void *)disconnect_view);
- return aws_raise_error(AWS_ERROR_MQTT5_DISCONNECT_OPTIONS_VALIDATION);
- }
- }
- if (disconnect_view->server_reference != NULL) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view - sending a server reference with a client-sourced DISCONNECT is "
- "not allowed",
- (void *)disconnect_view);
- return aws_raise_error(AWS_ERROR_MQTT5_DISCONNECT_OPTIONS_VALIDATION);
- }
- if (s_aws_mqtt5_user_property_set_validate(
- disconnect_view->user_properties,
- disconnect_view->user_property_count,
- "aws_mqtt5_packet_disconnect_view",
- (void *)disconnect_view)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- static int s_aws_mqtt5_packet_disconnect_view_validate_vs_connection_settings(
- const void *packet_view,
- const struct aws_mqtt5_client *client) {
- const struct aws_mqtt5_packet_disconnect_view *disconnect_view = packet_view;
- if (disconnect_view->session_expiry_interval_seconds != NULL) {
- /*
- * By spec (https://docs.oasis-open.org/mqtt/mqtt/v5.0/os/mqtt-v5.0-os.html#_Toc3901211), you
- * cannot set a non-zero value here if you sent a 0-value or no value in the CONNECT (presumably allows
- * the server to skip tracking session state, and we can't undo that now)
- */
- const uint32_t *session_expiry_ptr = client->config->connect.storage_view.session_expiry_interval_seconds;
- if (*disconnect_view->session_expiry_interval_seconds > 0 &&
- (session_expiry_ptr == NULL || *session_expiry_ptr == 0)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view - cannot specify a positive session expiry after "
- "committing "
- "to 0-valued session expiry in CONNECT",
- (void *)disconnect_view);
- return aws_raise_error(AWS_ERROR_MQTT5_DISCONNECT_OPTIONS_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_disconnect_view_log(
- const struct aws_mqtt5_packet_disconnect_view *disconnect_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view reason code set to %d (%s)",
- (void *)disconnect_view,
- (int)disconnect_view->reason_code,
- aws_mqtt5_disconnect_reason_code_to_c_string(disconnect_view->reason_code, NULL));
- if (disconnect_view->session_expiry_interval_seconds != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view session expiry interval set to %" PRIu32,
- (void *)disconnect_view,
- *disconnect_view->session_expiry_interval_seconds);
- }
- if (disconnect_view->reason_string != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view reason string set to \"" PRInSTR "\"",
- (void *)disconnect_view,
- AWS_BYTE_CURSOR_PRI(*disconnect_view->reason_string));
- }
- if (disconnect_view->server_reference != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_disconnect_view server reference set to \"" PRInSTR "\"",
- (void *)disconnect_view,
- AWS_BYTE_CURSOR_PRI(*disconnect_view->server_reference));
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- disconnect_view->user_properties,
- disconnect_view->user_property_count,
- (void *)disconnect_view,
- level,
- "aws_mqtt5_packet_disconnect_view");
- }
- void aws_mqtt5_packet_disconnect_storage_clean_up(struct aws_mqtt5_packet_disconnect_storage *disconnect_storage) {
- if (disconnect_storage == NULL) {
- return;
- }
- aws_mqtt5_user_property_set_clean_up(&disconnect_storage->user_properties);
- aws_byte_buf_clean_up(&disconnect_storage->storage);
- }
- static size_t s_aws_mqtt5_packet_disconnect_compute_storage_size(
- const struct aws_mqtt5_packet_disconnect_view *disconnect_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- disconnect_view->user_properties, disconnect_view->user_property_count);
- if (disconnect_view->reason_string != NULL) {
- storage_size += disconnect_view->reason_string->len;
- }
- if (disconnect_view->server_reference != NULL) {
- storage_size += disconnect_view->server_reference->len;
- }
- return storage_size;
- }
- int aws_mqtt5_packet_disconnect_storage_init(
- struct aws_mqtt5_packet_disconnect_storage *disconnect_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_disconnect_view *disconnect_options) {
- AWS_ZERO_STRUCT(*disconnect_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_disconnect_compute_storage_size(disconnect_options);
- if (aws_byte_buf_init(&disconnect_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_disconnect_view *storage_view = &disconnect_storage->storage_view;
- storage_view->reason_code = disconnect_options->reason_code;
- if (disconnect_options->session_expiry_interval_seconds != NULL) {
- disconnect_storage->session_expiry_interval_seconds = *disconnect_options->session_expiry_interval_seconds;
- storage_view->session_expiry_interval_seconds = &disconnect_storage->session_expiry_interval_seconds;
- }
- if (disconnect_options->reason_string != NULL) {
- disconnect_storage->reason_string = *disconnect_options->reason_string;
- if (aws_byte_buf_append_and_update(&disconnect_storage->storage, &disconnect_storage->reason_string)) {
- return AWS_OP_ERR;
- }
- storage_view->reason_string = &disconnect_storage->reason_string;
- }
- if (disconnect_options->server_reference != NULL) {
- disconnect_storage->server_reference = *disconnect_options->server_reference;
- if (aws_byte_buf_append_and_update(&disconnect_storage->storage, &disconnect_storage->server_reference)) {
- return AWS_OP_ERR;
- }
- storage_view->server_reference = &disconnect_storage->server_reference;
- }
- if (aws_mqtt5_user_property_set_init_with_storage(
- &disconnect_storage->user_properties,
- allocator,
- &disconnect_storage->storage,
- disconnect_options->user_property_count,
- disconnect_options->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&disconnect_storage->user_properties);
- storage_view->user_properties = disconnect_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_disconnect_storage_init_from_external_storage(
- struct aws_mqtt5_packet_disconnect_storage *disconnect_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*disconnect_storage);
- if (aws_mqtt5_user_property_set_init(&disconnect_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- static void s_destroy_operation_disconnect(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_disconnect *disconnect_op = object;
- aws_mqtt5_packet_disconnect_storage_clean_up(&disconnect_op->options_storage);
- aws_mem_release(disconnect_op->allocator, disconnect_op);
- }
- static void s_aws_mqtt5_disconnect_operation_completion(
- struct aws_mqtt5_operation *operation,
- int error_code,
- enum aws_mqtt5_packet_type packet_type,
- const void *completion_view) {
- (void)completion_view;
- (void)packet_type;
- struct aws_mqtt5_operation_disconnect *disconnect_op = operation->impl;
- if (disconnect_op->internal_completion_options.completion_callback != NULL) {
- (*disconnect_op->internal_completion_options.completion_callback)(
- error_code, disconnect_op->internal_completion_options.completion_user_data);
- }
- if (disconnect_op->external_completion_options.completion_callback != NULL) {
- (*disconnect_op->external_completion_options.completion_callback)(
- error_code, disconnect_op->external_completion_options.completion_user_data);
- }
- }
- static struct aws_mqtt5_operation_vtable s_disconnect_operation_vtable = {
- .aws_mqtt5_operation_completion_fn = s_aws_mqtt5_disconnect_operation_completion,
- .aws_mqtt5_operation_set_packet_id_fn = NULL,
- .aws_mqtt5_operation_get_packet_id_address_fn = NULL,
- .aws_mqtt5_operation_validate_vs_connection_settings_fn =
- s_aws_mqtt5_packet_disconnect_view_validate_vs_connection_settings,
- };
- struct aws_mqtt5_operation_disconnect *aws_mqtt5_operation_disconnect_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_disconnect_view *disconnect_options,
- const struct aws_mqtt5_disconnect_completion_options *external_completion_options,
- const struct aws_mqtt5_disconnect_completion_options *internal_completion_options) {
- AWS_PRECONDITION(allocator != NULL);
- if (aws_mqtt5_packet_disconnect_view_validate(disconnect_options)) {
- return NULL;
- }
- struct aws_mqtt5_operation_disconnect *disconnect_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_disconnect));
- if (disconnect_op == NULL) {
- return NULL;
- }
- disconnect_op->allocator = allocator;
- disconnect_op->base.vtable = &s_disconnect_operation_vtable;
- disconnect_op->base.packet_type = AWS_MQTT5_PT_DISCONNECT;
- aws_ref_count_init(&disconnect_op->base.ref_count, disconnect_op, s_destroy_operation_disconnect);
- disconnect_op->base.impl = disconnect_op;
- if (aws_mqtt5_packet_disconnect_storage_init(&disconnect_op->options_storage, allocator, disconnect_options)) {
- goto error;
- }
- disconnect_op->base.packet_view = &disconnect_op->options_storage.storage_view;
- if (external_completion_options != NULL) {
- disconnect_op->external_completion_options = *external_completion_options;
- }
- if (internal_completion_options != NULL) {
- disconnect_op->internal_completion_options = *internal_completion_options;
- }
- return disconnect_op;
- error:
- aws_mqtt5_operation_release(&disconnect_op->base);
- return NULL;
- }
- /*********************************************************************************************************************
- * Publish
- ********************************************************************************************************************/
- int aws_mqtt5_packet_publish_view_validate(const struct aws_mqtt5_packet_publish_view *publish_view) {
- if (publish_view == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "null PUBLISH packet options");
- return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
- }
- if (publish_view->qos < AWS_MQTT5_QOS_AT_MOST_ONCE || publish_view->qos > AWS_MQTT5_QOS_EXACTLY_ONCE) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - unsupported QoS value in PUBLISH packet options: %d",
- (void *)publish_view,
- (int)publish_view->qos);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (publish_view->qos == AWS_MQTT5_QOS_AT_MOST_ONCE) {
- if (publish_view->duplicate) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - duplicate flag must be set to 0 for QoS 0 messages",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (publish_view->packet_id != 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - Packet ID must not be set for QoS 0 messages",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- /* 0-length topic is never valid, even with user-controlled outbound aliasing */
- if (publish_view->topic.len == 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL, "id=%p: aws_mqtt5_packet_publish_view - missing topic", (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- } else if (aws_mqtt5_validate_utf8_text(publish_view->topic)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL, "id=%p: aws_mqtt5_packet_publish_view - topic not valid UTF-8", (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- } else if (!aws_mqtt_is_valid_topic(&publish_view->topic)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - invalid topic: \"" PRInSTR "\"",
- (void *)publish_view,
- AWS_BYTE_CURSOR_PRI(publish_view->topic));
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (publish_view->topic_alias != NULL) {
- if (*publish_view->topic_alias == 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - topic alias may not be zero",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- if (publish_view->payload_format != NULL) {
- if (*publish_view->payload_format < AWS_MQTT5_PFI_BYTES || *publish_view->payload_format > AWS_MQTT5_PFI_UTF8) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - invalid payload format value: %d",
- (void *)publish_view,
- (int)*publish_view->payload_format);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- // Make sure the payload data is UTF-8 if the payload_format set to UTF8
- if (*publish_view->payload_format == AWS_MQTT5_PFI_UTF8) {
- if (aws_mqtt5_validate_utf8_text(publish_view->payload)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - payload value is not valid UTF-8 while payload format "
- "set to UTF-8",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- }
- if (publish_view->response_topic != NULL) {
- if (publish_view->response_topic->len >= UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - response topic too long",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(*publish_view->response_topic)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - response topic not valid UTF-8",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (!aws_mqtt_is_valid_topic(publish_view->response_topic)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - response topic must be a valid mqtt topic",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- if (publish_view->correlation_data != NULL) {
- if (publish_view->correlation_data->len >= UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - correlation data too long",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- /*
- * validate is done from a client perspective and clients should never generate subscription identifier in a
- * publish message
- */
- if (publish_view->subscription_identifier_count != 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL, "Client-initiated PUBLISH packets may not contain subscription identifiers");
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (publish_view->content_type != NULL) {
- if (publish_view->content_type->len >= UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - content type too long",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (aws_mqtt5_validate_utf8_text(*publish_view->content_type)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - content type not valid UTF-8",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- if (s_aws_mqtt5_user_property_set_validate(
- publish_view->user_properties,
- publish_view->user_property_count,
- "aws_mqtt5_packet_publish_view",
- (void *)publish_view)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_publish_view_validate_vs_iot_core(const struct aws_mqtt5_packet_publish_view *publish_view) {
- if (!aws_mqtt_is_valid_topic_for_iot_core(publish_view->topic)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - topic not valid for AWS Iot Core limits: \"" PRInSTR "\"",
- (void *)publish_view,
- AWS_BYTE_CURSOR_PRI(publish_view->topic));
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- static int s_aws_mqtt5_packet_publish_view_validate_vs_connection_settings(
- const void *packet_view,
- const struct aws_mqtt5_client *client) {
- const struct aws_mqtt5_packet_publish_view *publish_view = packet_view;
- /* If we have valid negotiated settings, check against them as well */
- if (aws_mqtt5_client_are_negotiated_settings_valid(client)) {
- const struct aws_mqtt5_negotiated_settings *settings = &client->negotiated_settings;
- if (publish_view->qos > settings->maximum_qos) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - QoS value %d exceeds negotiated maximum qos %d",
- (void *)publish_view,
- (int)publish_view->qos,
- (int)settings->maximum_qos);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (publish_view->topic_alias != NULL) {
- const struct aws_mqtt5_client_options_storage *client_options = client->config;
- if (client_options->topic_aliasing_options.outbound_topic_alias_behavior != AWS_MQTT5_COTABT_USER) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - topic alias set but outbound topic alias behavior has not "
- "been set to user controlled",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- if (*publish_view->topic_alias > settings->topic_alias_maximum_to_server) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - outbound topic alias (%d) exceeds server's topic alias "
- "maximum "
- "(%d)",
- (void *)publish_view,
- (int)(*publish_view->topic_alias),
- (int)settings->topic_alias_maximum_to_server);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- if (publish_view->retain && settings->retain_available == false) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - server does not support Retain",
- (void *)publish_view);
- return aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_publish_view_log(
- const struct aws_mqtt5_packet_publish_view *publish_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view packet id set to %d",
- (void *)publish_view,
- (int)publish_view->packet_id);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view payload set containing %zu bytes",
- (void *)publish_view,
- publish_view->payload.len);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view qos set to %d",
- (void *)publish_view,
- (int)publish_view->qos);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view retain set to %d",
- (void *)publish_view,
- (int)publish_view->retain);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view topic set to \"" PRInSTR "\"",
- (void *)publish_view,
- AWS_BYTE_CURSOR_PRI(publish_view->topic));
- if (publish_view->payload_format != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view payload format indicator set to %d (%s)",
- (void *)publish_view,
- (int)*publish_view->payload_format,
- aws_mqtt5_payload_format_indicator_to_c_string(*publish_view->payload_format));
- }
- if (publish_view->message_expiry_interval_seconds != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view message expiry interval set to %" PRIu32,
- (void *)publish_view,
- *publish_view->message_expiry_interval_seconds);
- }
- if (publish_view->topic_alias != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view topic alias set to %" PRIu16,
- (void *)publish_view,
- *publish_view->topic_alias);
- }
- if (publish_view->response_topic != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view response topic set to \"" PRInSTR "\"",
- (void *)publish_view,
- AWS_BYTE_CURSOR_PRI(*publish_view->response_topic));
- }
- if (publish_view->correlation_data != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view - set correlation data",
- (void *)publish_view);
- }
- if (publish_view->content_type != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view content type set to \"" PRInSTR "\"",
- (void *)publish_view,
- AWS_BYTE_CURSOR_PRI(*publish_view->content_type));
- }
- for (size_t i = 0; i < publish_view->subscription_identifier_count; ++i) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view subscription identifier %d: %" PRIu32,
- (void *)publish_view,
- (int)i,
- publish_view->subscription_identifiers[i]);
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- publish_view->user_properties,
- publish_view->user_property_count,
- (void *)publish_view,
- level,
- "aws_mqtt5_packet_publish_view");
- }
- static size_t s_aws_mqtt5_packet_publish_compute_storage_size(
- const struct aws_mqtt5_packet_publish_view *publish_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- publish_view->user_properties, publish_view->user_property_count);
- storage_size += publish_view->topic.len;
- storage_size += publish_view->payload.len;
- if (publish_view->response_topic != NULL) {
- storage_size += publish_view->response_topic->len;
- }
- if (publish_view->correlation_data != NULL) {
- storage_size += publish_view->correlation_data->len;
- }
- if (publish_view->content_type != NULL) {
- storage_size += publish_view->content_type->len;
- }
- return storage_size;
- }
- int aws_mqtt5_packet_publish_storage_init(
- struct aws_mqtt5_packet_publish_storage *publish_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_publish_view *publish_options) {
- AWS_ZERO_STRUCT(*publish_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_publish_compute_storage_size(publish_options);
- if (aws_byte_buf_init(&publish_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_init_dynamic(&publish_storage->subscription_identifiers, allocator, 0, sizeof(uint32_t))) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_publish_view *storage_view = &publish_storage->storage_view;
- storage_view->packet_id = publish_options->packet_id;
- storage_view->payload = publish_options->payload;
- if (aws_byte_buf_append_and_update(&publish_storage->storage, &storage_view->payload)) {
- return AWS_OP_ERR;
- }
- storage_view->qos = publish_options->qos;
- storage_view->retain = publish_options->retain;
- storage_view->duplicate = publish_options->duplicate;
- storage_view->topic = publish_options->topic;
- if (aws_byte_buf_append_and_update(&publish_storage->storage, &storage_view->topic)) {
- return AWS_OP_ERR;
- }
- if (publish_options->payload_format != NULL) {
- publish_storage->payload_format = *publish_options->payload_format;
- storage_view->payload_format = &publish_storage->payload_format;
- }
- if (publish_options->message_expiry_interval_seconds != NULL) {
- publish_storage->message_expiry_interval_seconds = *publish_options->message_expiry_interval_seconds;
- storage_view->message_expiry_interval_seconds = &publish_storage->message_expiry_interval_seconds;
- }
- if (publish_options->topic_alias != NULL) {
- publish_storage->topic_alias = *publish_options->topic_alias;
- storage_view->topic_alias = &publish_storage->topic_alias;
- }
- if (publish_options->response_topic != NULL) {
- publish_storage->response_topic = *publish_options->response_topic;
- if (aws_byte_buf_append_and_update(&publish_storage->storage, &publish_storage->response_topic)) {
- return AWS_OP_ERR;
- }
- storage_view->response_topic = &publish_storage->response_topic;
- }
- if (publish_options->correlation_data != NULL) {
- publish_storage->correlation_data = *publish_options->correlation_data;
- if (aws_byte_buf_append_and_update(&publish_storage->storage, &publish_storage->correlation_data)) {
- return AWS_OP_ERR;
- }
- storage_view->correlation_data = &publish_storage->correlation_data;
- }
- for (size_t i = 0; i < publish_options->subscription_identifier_count; ++i) {
- aws_array_list_push_back(
- &publish_storage->subscription_identifiers, &publish_options->subscription_identifiers[i]);
- }
- storage_view->subscription_identifier_count = aws_array_list_length(&publish_storage->subscription_identifiers);
- storage_view->subscription_identifiers = publish_storage->subscription_identifiers.data;
- if (publish_options->content_type != NULL) {
- publish_storage->content_type = *publish_options->content_type;
- if (aws_byte_buf_append_and_update(&publish_storage->storage, &publish_storage->content_type)) {
- return AWS_OP_ERR;
- }
- storage_view->content_type = &publish_storage->content_type;
- }
- if (aws_mqtt5_user_property_set_init_with_storage(
- &publish_storage->user_properties,
- allocator,
- &publish_storage->storage,
- publish_options->user_property_count,
- publish_options->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&publish_storage->user_properties);
- storage_view->user_properties = publish_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_publish_storage_init_from_external_storage(
- struct aws_mqtt5_packet_publish_storage *publish_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*publish_storage);
- if (aws_mqtt5_user_property_set_init(&publish_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_init_dynamic(&publish_storage->subscription_identifiers, allocator, 0, sizeof(uint32_t))) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_publish_storage_clean_up(struct aws_mqtt5_packet_publish_storage *publish_storage) {
- aws_mqtt5_user_property_set_clean_up(&publish_storage->user_properties);
- aws_array_list_clean_up(&publish_storage->subscription_identifiers);
- aws_byte_buf_clean_up(&publish_storage->storage);
- }
- static void s_aws_mqtt5_operation_publish_complete(
- struct aws_mqtt5_operation *operation,
- int error_code,
- enum aws_mqtt5_packet_type packet_type,
- const void *completion_view) {
- struct aws_mqtt5_operation_publish *publish_op = operation->impl;
- if (publish_op->completion_options.completion_callback != NULL) {
- (*publish_op->completion_options.completion_callback)(
- packet_type, completion_view, error_code, publish_op->completion_options.completion_user_data);
- }
- }
- static void s_aws_mqtt5_operation_publish_set_packet_id(
- struct aws_mqtt5_operation *operation,
- aws_mqtt5_packet_id_t packet_id) {
- struct aws_mqtt5_operation_publish *publish_op = operation->impl;
- publish_op->options_storage.storage_view.packet_id = packet_id;
- }
- static aws_mqtt5_packet_id_t *s_aws_mqtt5_operation_publish_get_packet_id_address(
- const struct aws_mqtt5_operation *operation) {
- struct aws_mqtt5_operation_publish *publish_op = operation->impl;
- return &publish_op->options_storage.storage_view.packet_id;
- }
- static struct aws_mqtt5_operation_vtable s_publish_operation_vtable = {
- .aws_mqtt5_operation_completion_fn = s_aws_mqtt5_operation_publish_complete,
- .aws_mqtt5_operation_set_packet_id_fn = s_aws_mqtt5_operation_publish_set_packet_id,
- .aws_mqtt5_operation_get_packet_id_address_fn = s_aws_mqtt5_operation_publish_get_packet_id_address,
- .aws_mqtt5_operation_validate_vs_connection_settings_fn =
- s_aws_mqtt5_packet_publish_view_validate_vs_connection_settings,
- };
- static void s_destroy_operation_publish(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_publish *publish_op = object;
- aws_mqtt5_packet_publish_storage_clean_up(&publish_op->options_storage);
- aws_mem_release(publish_op->allocator, publish_op);
- }
- struct aws_mqtt5_operation_publish *aws_mqtt5_operation_publish_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_client *client,
- const struct aws_mqtt5_packet_publish_view *publish_options,
- const struct aws_mqtt5_publish_completion_options *completion_options) {
- AWS_PRECONDITION(allocator != NULL);
- AWS_PRECONDITION(publish_options != NULL);
- if (aws_mqtt5_packet_publish_view_validate(publish_options)) {
- return NULL;
- }
- if (publish_options->packet_id != 0) {
- AWS_LOGF_DEBUG(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_publish_view packet id must be zero",
- (void *)publish_options);
- aws_raise_error(AWS_ERROR_MQTT5_PUBLISH_OPTIONS_VALIDATION);
- return NULL;
- }
- if (client != NULL && client->config->extended_validation_and_flow_control_options != AWS_MQTT5_EVAFCO_NONE) {
- if (aws_mqtt5_packet_publish_view_validate_vs_iot_core(publish_options)) {
- return NULL;
- }
- }
- struct aws_mqtt5_operation_publish *publish_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_publish));
- if (publish_op == NULL) {
- return NULL;
- }
- publish_op->allocator = allocator;
- publish_op->base.vtable = &s_publish_operation_vtable;
- publish_op->base.packet_type = AWS_MQTT5_PT_PUBLISH;
- aws_ref_count_init(&publish_op->base.ref_count, publish_op, s_destroy_operation_publish);
- publish_op->base.impl = publish_op;
- if (aws_mqtt5_packet_publish_storage_init(&publish_op->options_storage, allocator, publish_options)) {
- goto error;
- }
- publish_op->base.packet_view = &publish_op->options_storage.storage_view;
- if (completion_options != NULL) {
- publish_op->completion_options = *completion_options;
- }
- return publish_op;
- error:
- aws_mqtt5_operation_release(&publish_op->base);
- return NULL;
- }
- /*********************************************************************************************************************
- * Puback
- ********************************************************************************************************************/
- static size_t s_aws_mqtt5_packet_puback_compute_storage_size(const struct aws_mqtt5_packet_puback_view *puback_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- puback_view->user_properties, puback_view->user_property_count);
- if (puback_view->reason_string != NULL) {
- storage_size += puback_view->reason_string->len;
- }
- return storage_size;
- }
- AWS_MQTT_API int aws_mqtt5_packet_puback_storage_init(
- struct aws_mqtt5_packet_puback_storage *puback_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_puback_view *puback_view) {
- AWS_ZERO_STRUCT(*puback_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_puback_compute_storage_size(puback_view);
- if (aws_byte_buf_init(&puback_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_puback_view *storage_view = &puback_storage->storage_view;
- storage_view->packet_id = puback_view->packet_id;
- storage_view->reason_code = puback_view->reason_code;
- if (puback_view->reason_string != NULL) {
- puback_storage->reason_string = *puback_view->reason_string;
- if (aws_byte_buf_append_and_update(&puback_storage->storage, &puback_storage->reason_string)) {
- return AWS_OP_ERR;
- }
- storage_view->reason_string = &puback_storage->reason_string;
- }
- if (aws_mqtt5_user_property_set_init_with_storage(
- &puback_storage->user_properties,
- allocator,
- &puback_storage->storage,
- puback_view->user_property_count,
- puback_view->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&puback_storage->user_properties);
- storage_view->user_properties = puback_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_puback_storage_init_from_external_storage(
- struct aws_mqtt5_packet_puback_storage *puback_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*puback_storage);
- if (aws_mqtt5_user_property_set_init(&puback_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_puback_storage_clean_up(struct aws_mqtt5_packet_puback_storage *puback_storage) {
- if (puback_storage == NULL) {
- return;
- }
- aws_mqtt5_user_property_set_clean_up(&puback_storage->user_properties);
- aws_byte_buf_clean_up(&puback_storage->storage);
- }
- void aws_mqtt5_packet_puback_view_log(
- const struct aws_mqtt5_packet_puback_view *puback_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_puback_view packet id set to %d",
- (void *)puback_view,
- (int)puback_view->packet_id);
- enum aws_mqtt5_puback_reason_code reason_code = puback_view->reason_code;
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: puback %d reason code: %s",
- (void *)puback_view,
- (int)reason_code,
- aws_mqtt5_puback_reason_code_to_c_string(reason_code));
- if (puback_view->reason_string != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_puback_view reason string set to \"" PRInSTR "\"",
- (void *)puback_view,
- AWS_BYTE_CURSOR_PRI(*puback_view->reason_string));
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- puback_view->user_properties,
- puback_view->user_property_count,
- (void *)puback_view,
- level,
- "aws_mqtt5_packet_puback_view");
- }
- static void s_destroy_operation_puback(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_puback *puback_op = object;
- aws_mqtt5_packet_puback_storage_clean_up(&puback_op->options_storage);
- aws_mem_release(puback_op->allocator, puback_op);
- }
- struct aws_mqtt5_operation_puback *aws_mqtt5_operation_puback_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_puback_view *puback_options) {
- AWS_PRECONDITION(allocator != NULL);
- AWS_PRECONDITION(puback_options != NULL);
- struct aws_mqtt5_operation_puback *puback_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_puback));
- if (puback_op == NULL) {
- return NULL;
- }
- puback_op->allocator = allocator;
- puback_op->base.vtable = &s_empty_operation_vtable;
- puback_op->base.packet_type = AWS_MQTT5_PT_PUBACK;
- aws_ref_count_init(&puback_op->base.ref_count, puback_op, s_destroy_operation_puback);
- puback_op->base.impl = puback_op;
- if (aws_mqtt5_packet_puback_storage_init(&puback_op->options_storage, allocator, puback_options)) {
- goto error;
- }
- puback_op->base.packet_view = &puback_op->options_storage.storage_view;
- return puback_op;
- error:
- aws_mqtt5_operation_release(&puback_op->base);
- return NULL;
- }
- /*********************************************************************************************************************
- * Unsubscribe
- ********************************************************************************************************************/
- int aws_mqtt5_packet_unsubscribe_view_validate(const struct aws_mqtt5_packet_unsubscribe_view *unsubscribe_view) {
- if (unsubscribe_view == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "null UNSUBSCRIBE packet options");
- return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
- }
- if (unsubscribe_view->topic_filter_count == 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view - must contain at least one topic",
- (void *)unsubscribe_view);
- return aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (unsubscribe_view->topic_filter_count > AWS_MQTT5_CLIENT_MAXIMUM_TOPIC_FILTERS_PER_UNSUBSCRIBE) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view - contains too many topics (%zu)",
- (void *)unsubscribe_view,
- unsubscribe_view->topic_filter_count);
- return aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- }
- for (size_t i = 0; i < unsubscribe_view->topic_filter_count; ++i) {
- const struct aws_byte_cursor *topic_filter = &unsubscribe_view->topic_filters[i];
- if (aws_mqtt5_validate_utf8_text(*topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view - topic filter not valid UTF-8: \"" PRInSTR "\"",
- (void *)unsubscribe_view,
- AWS_BYTE_CURSOR_PRI(*topic_filter));
- return aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (!aws_mqtt_is_valid_topic_filter(topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view - invalid topic filter: \"" PRInSTR "\"",
- (void *)unsubscribe_view,
- AWS_BYTE_CURSOR_PRI(*topic_filter));
- return aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- }
- }
- if (s_aws_mqtt5_user_property_set_validate(
- unsubscribe_view->user_properties,
- unsubscribe_view->user_property_count,
- "aws_mqtt5_packet_unsubscribe_view",
- (void *)unsubscribe_view)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- AWS_MQTT_API int aws_mqtt5_packet_unsubscribe_view_validate_vs_iot_core(
- const struct aws_mqtt5_packet_unsubscribe_view *unsubscribe_view) {
- for (size_t i = 0; i < unsubscribe_view->topic_filter_count; ++i) {
- const struct aws_byte_cursor *topic_filter = &unsubscribe_view->topic_filters[i];
- if (!aws_mqtt_is_valid_topic_filter_for_iot_core(*topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view - topic filter not valid for AWS Iot Core limits: \"" PRInSTR
- "\"",
- (void *)unsubscribe_view,
- AWS_BYTE_CURSOR_PRI(*topic_filter));
- return aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_unsubscribe_view_log(
- const struct aws_mqtt5_packet_unsubscribe_view *unsubscribe_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- size_t topic_count = unsubscribe_view->topic_filter_count;
- for (size_t i = 0; i < topic_count; ++i) {
- const struct aws_byte_cursor *topic_cursor = &unsubscribe_view->topic_filters[i];
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view topic #%zu: \"" PRInSTR "\"",
- (void *)unsubscribe_view,
- i,
- AWS_BYTE_CURSOR_PRI(*topic_cursor));
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- unsubscribe_view->user_properties,
- unsubscribe_view->user_property_count,
- (void *)unsubscribe_view,
- level,
- "aws_mqtt5_packet_unsubscribe_view");
- }
- void aws_mqtt5_packet_unsubscribe_storage_clean_up(struct aws_mqtt5_packet_unsubscribe_storage *unsubscribe_storage) {
- if (unsubscribe_storage == NULL) {
- return;
- }
- aws_array_list_clean_up(&unsubscribe_storage->topic_filters);
- aws_mqtt5_user_property_set_clean_up(&unsubscribe_storage->user_properties);
- aws_byte_buf_clean_up(&unsubscribe_storage->storage);
- }
- static int s_aws_mqtt5_packet_unsubscribe_build_topic_list(
- struct aws_mqtt5_packet_unsubscribe_storage *unsubscribe_storage,
- struct aws_allocator *allocator,
- size_t topic_count,
- const struct aws_byte_cursor *topics) {
- if (aws_array_list_init_dynamic(
- &unsubscribe_storage->topic_filters, allocator, topic_count, sizeof(struct aws_byte_cursor))) {
- return AWS_OP_ERR;
- }
- for (size_t i = 0; i < topic_count; ++i) {
- const struct aws_byte_cursor *topic_cursor_ptr = &topics[i];
- struct aws_byte_cursor topic_cursor = *topic_cursor_ptr;
- if (aws_byte_buf_append_and_update(&unsubscribe_storage->storage, &topic_cursor)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_push_back(&unsubscribe_storage->topic_filters, &topic_cursor)) {
- return AWS_OP_ERR;
- }
- }
- return AWS_OP_SUCCESS;
- }
- static size_t s_aws_mqtt5_packet_unsubscribe_compute_storage_size(
- const struct aws_mqtt5_packet_unsubscribe_view *unsubscribe_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- unsubscribe_view->user_properties, unsubscribe_view->user_property_count);
- for (size_t i = 0; i < unsubscribe_view->topic_filter_count; ++i) {
- const struct aws_byte_cursor *topic = &unsubscribe_view->topic_filters[i];
- storage_size += topic->len;
- }
- return storage_size;
- }
- int aws_mqtt5_packet_unsubscribe_storage_init(
- struct aws_mqtt5_packet_unsubscribe_storage *unsubscribe_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_unsubscribe_view *unsubscribe_options) {
- AWS_ZERO_STRUCT(*unsubscribe_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_unsubscribe_compute_storage_size(unsubscribe_options);
- if (aws_byte_buf_init(&unsubscribe_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_unsubscribe_view *storage_view = &unsubscribe_storage->storage_view;
- if (s_aws_mqtt5_packet_unsubscribe_build_topic_list(
- unsubscribe_storage,
- allocator,
- unsubscribe_options->topic_filter_count,
- unsubscribe_options->topic_filters)) {
- return AWS_OP_ERR;
- }
- storage_view->topic_filter_count = aws_array_list_length(&unsubscribe_storage->topic_filters);
- storage_view->topic_filters = unsubscribe_storage->topic_filters.data;
- if (aws_mqtt5_user_property_set_init_with_storage(
- &unsubscribe_storage->user_properties,
- allocator,
- &unsubscribe_storage->storage,
- unsubscribe_options->user_property_count,
- unsubscribe_options->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&unsubscribe_storage->user_properties);
- storage_view->user_properties = unsubscribe_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_unsubscribe_storage_init_from_external_storage(
- struct aws_mqtt5_packet_unsubscribe_storage *unsubscribe_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*unsubscribe_storage);
- if (aws_mqtt5_user_property_set_init(&unsubscribe_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_init_dynamic(
- &unsubscribe_storage->topic_filters, allocator, 0, sizeof(struct aws_byte_cursor))) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- static void s_aws_mqtt5_operation_unsubscribe_complete(
- struct aws_mqtt5_operation *operation,
- int error_code,
- enum aws_mqtt5_packet_type packet_type,
- const void *completion_view) {
- struct aws_mqtt5_operation_unsubscribe *unsubscribe_op = operation->impl;
- (void)packet_type;
- if (unsubscribe_op->completion_options.completion_callback != NULL) {
- (*unsubscribe_op->completion_options.completion_callback)(
- completion_view, error_code, unsubscribe_op->completion_options.completion_user_data);
- }
- }
- static void s_aws_mqtt5_operation_unsubscribe_set_packet_id(
- struct aws_mqtt5_operation *operation,
- aws_mqtt5_packet_id_t packet_id) {
- struct aws_mqtt5_operation_unsubscribe *unsubscribe_op = operation->impl;
- unsubscribe_op->options_storage.storage_view.packet_id = packet_id;
- }
- static aws_mqtt5_packet_id_t *s_aws_mqtt5_operation_unsubscribe_get_packet_id_address(
- const struct aws_mqtt5_operation *operation) {
- struct aws_mqtt5_operation_unsubscribe *unsubscribe_op = operation->impl;
- return &unsubscribe_op->options_storage.storage_view.packet_id;
- }
- static struct aws_mqtt5_operation_vtable s_unsubscribe_operation_vtable = {
- .aws_mqtt5_operation_completion_fn = s_aws_mqtt5_operation_unsubscribe_complete,
- .aws_mqtt5_operation_set_packet_id_fn = s_aws_mqtt5_operation_unsubscribe_set_packet_id,
- .aws_mqtt5_operation_get_packet_id_address_fn = s_aws_mqtt5_operation_unsubscribe_get_packet_id_address,
- .aws_mqtt5_operation_validate_vs_connection_settings_fn = NULL,
- };
- static void s_destroy_operation_unsubscribe(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_unsubscribe *unsubscribe_op = object;
- aws_mqtt5_packet_unsubscribe_storage_clean_up(&unsubscribe_op->options_storage);
- aws_mem_release(unsubscribe_op->allocator, unsubscribe_op);
- }
- struct aws_mqtt5_operation_unsubscribe *aws_mqtt5_operation_unsubscribe_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_client *client,
- const struct aws_mqtt5_packet_unsubscribe_view *unsubscribe_options,
- const struct aws_mqtt5_unsubscribe_completion_options *completion_options) {
- AWS_PRECONDITION(allocator != NULL);
- AWS_PRECONDITION(unsubscribe_options != NULL);
- if (aws_mqtt5_packet_unsubscribe_view_validate(unsubscribe_options)) {
- return NULL;
- }
- if (unsubscribe_options->packet_id != 0) {
- AWS_LOGF_DEBUG(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsubscribe_view packet id must be zero",
- (void *)unsubscribe_options);
- aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- return NULL;
- }
- if (client != NULL && client->config->extended_validation_and_flow_control_options != AWS_MQTT5_EVAFCO_NONE) {
- if (aws_mqtt5_packet_unsubscribe_view_validate_vs_iot_core(unsubscribe_options)) {
- return NULL;
- }
- }
- struct aws_mqtt5_operation_unsubscribe *unsubscribe_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_unsubscribe));
- if (unsubscribe_op == NULL) {
- return NULL;
- }
- unsubscribe_op->allocator = allocator;
- unsubscribe_op->base.vtable = &s_unsubscribe_operation_vtable;
- unsubscribe_op->base.packet_type = AWS_MQTT5_PT_UNSUBSCRIBE;
- aws_ref_count_init(&unsubscribe_op->base.ref_count, unsubscribe_op, s_destroy_operation_unsubscribe);
- unsubscribe_op->base.impl = unsubscribe_op;
- if (aws_mqtt5_packet_unsubscribe_storage_init(&unsubscribe_op->options_storage, allocator, unsubscribe_options)) {
- goto error;
- }
- unsubscribe_op->base.packet_view = &unsubscribe_op->options_storage.storage_view;
- if (completion_options != NULL) {
- unsubscribe_op->completion_options = *completion_options;
- }
- return unsubscribe_op;
- error:
- aws_mqtt5_operation_release(&unsubscribe_op->base);
- return NULL;
- }
- /*********************************************************************************************************************
- * Subscribe
- ********************************************************************************************************************/
- static int s_aws_mqtt5_validate_subscription(
- const struct aws_mqtt5_subscription_view *subscription,
- void *log_context) {
- if (aws_mqtt5_validate_utf8_text(subscription->topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - topic filter \"" PRInSTR "\" not valid UTF-8 in subscription",
- log_context,
- AWS_BYTE_CURSOR_PRI(subscription->topic_filter));
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (!aws_mqtt_is_valid_topic_filter(&subscription->topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - invalid topic filter \"" PRInSTR "\" in subscription",
- log_context,
- AWS_BYTE_CURSOR_PRI(subscription->topic_filter));
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (subscription->topic_filter.len > UINT16_MAX) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - subscription contains too-long topic filter",
- log_context);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (subscription->qos < AWS_MQTT5_QOS_AT_MOST_ONCE || subscription->qos > AWS_MQTT5_QOS_AT_LEAST_ONCE) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - unsupported QoS value: %d",
- log_context,
- (int)subscription->qos);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (subscription->retain_handling_type < AWS_MQTT5_RHT_SEND_ON_SUBSCRIBE ||
- subscription->retain_handling_type > AWS_MQTT5_RHT_DONT_SEND) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - unsupported retain handling value: %d",
- log_context,
- (int)subscription->retain_handling_type);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- /* mqtt5 forbids no_local to be set to 1 if the topic filter represents a shared subscription */
- if (subscription->no_local) {
- if (aws_mqtt_is_topic_filter_shared_subscription(subscription->topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - no_local cannot be 1 if the topic filter is a shared"
- "subscription",
- log_context);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_subscribe_view_validate(const struct aws_mqtt5_packet_subscribe_view *subscribe_view) {
- if (subscribe_view == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "null SUBSCRIBE packet options");
- return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
- }
- if (subscribe_view->subscription_count == 0) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - must contain at least one subscription",
- (void *)subscribe_view);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- if (subscribe_view->subscription_count > AWS_MQTT5_CLIENT_MAXIMUM_SUBSCRIPTIONS_PER_SUBSCRIBE) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - too many subscriptions",
- (void *)subscribe_view);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- for (size_t i = 0; i < subscribe_view->subscription_count; ++i) {
- const struct aws_mqtt5_subscription_view *subscription = &subscribe_view->subscriptions[i];
- if (s_aws_mqtt5_validate_subscription(subscription, (void *)subscribe_view)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - invalid subscription",
- (void *)subscribe_view);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- }
- if (subscribe_view->subscription_identifier != NULL) {
- if (*subscribe_view->subscription_identifier > AWS_MQTT5_MAXIMUM_VARIABLE_LENGTH_INTEGER) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - subscription identifier (%" PRIu32 ") too large",
- (void *)subscribe_view,
- *subscribe_view->subscription_identifier);
- return aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- }
- }
- if (s_aws_mqtt5_user_property_set_validate(
- subscribe_view->user_properties,
- subscribe_view->user_property_count,
- "aws_mqtt5_packet_subscribe_view",
- (void *)subscribe_view)) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- AWS_MQTT_API int aws_mqtt5_packet_subscribe_view_validate_vs_iot_core(
- const struct aws_mqtt5_packet_subscribe_view *subscribe_view) {
- if (subscribe_view->subscription_count > AWS_IOT_CORE_MAXIMUM_SUSBCRIPTIONS_PER_SUBSCRIBE) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - number of subscriptions (%zu) exceeds default AWS IoT Core limit "
- "(%d)",
- (void *)subscribe_view,
- subscribe_view->subscription_count,
- (int)AWS_IOT_CORE_MAXIMUM_SUSBCRIPTIONS_PER_SUBSCRIBE);
- return AWS_OP_ERR;
- }
- for (size_t i = 0; i < subscribe_view->subscription_count; ++i) {
- const struct aws_mqtt5_subscription_view *subscription = &subscribe_view->subscriptions[i];
- const struct aws_byte_cursor *topic_filter = &subscription->topic_filter;
- if (!aws_mqtt_is_valid_topic_filter_for_iot_core(*topic_filter)) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view - topic filter not valid for AWS Iot Core limits: \"" PRInSTR
- "\"",
- (void *)subscribe_view,
- AWS_BYTE_CURSOR_PRI(*topic_filter));
- return aws_raise_error(AWS_ERROR_MQTT5_UNSUBSCRIBE_OPTIONS_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_subscribe_view_log(
- const struct aws_mqtt5_packet_subscribe_view *subscribe_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- size_t subscription_count = subscribe_view->subscription_count;
- for (size_t i = 0; i < subscription_count; ++i) {
- const struct aws_mqtt5_subscription_view *view = &subscribe_view->subscriptions[i];
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view subscription #%zu, topic filter \"" PRInSTR
- "\", qos %d, no local %d, retain as "
- "published %d, retain handling %d (%s)",
- (void *)subscribe_view,
- i,
- AWS_BYTE_CURSOR_PRI(view->topic_filter),
- (int)view->qos,
- (int)view->no_local,
- (int)view->retain_as_published,
- (int)view->retain_handling_type,
- aws_mqtt5_retain_handling_type_to_c_string(view->retain_handling_type));
- }
- if (subscribe_view->subscription_identifier != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view subscription identifier set to %" PRIu32,
- (void *)subscribe_view,
- *subscribe_view->subscription_identifier);
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- subscribe_view->user_properties,
- subscribe_view->user_property_count,
- (void *)subscribe_view,
- level,
- "aws_mqtt5_packet_subscribe_view");
- }
- void aws_mqtt5_packet_subscribe_storage_clean_up(struct aws_mqtt5_packet_subscribe_storage *subscribe_storage) {
- if (subscribe_storage == NULL) {
- return;
- }
- aws_array_list_clean_up(&subscribe_storage->subscriptions);
- aws_mqtt5_user_property_set_clean_up(&subscribe_storage->user_properties);
- aws_byte_buf_clean_up(&subscribe_storage->storage);
- }
- static int s_aws_mqtt5_packet_subscribe_storage_init_subscriptions(
- struct aws_mqtt5_packet_subscribe_storage *subscribe_storage,
- struct aws_allocator *allocator,
- size_t subscription_count,
- const struct aws_mqtt5_subscription_view *subscriptions) {
- if (aws_array_list_init_dynamic(
- &subscribe_storage->subscriptions,
- allocator,
- subscription_count,
- sizeof(struct aws_mqtt5_subscription_view))) {
- return AWS_OP_ERR;
- }
- for (size_t i = 0; i < subscription_count; ++i) {
- const struct aws_mqtt5_subscription_view *source = &subscriptions[i];
- struct aws_mqtt5_subscription_view copy = *source;
- if (aws_byte_buf_append_and_update(&subscribe_storage->storage, ©.topic_filter)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_push_back(&subscribe_storage->subscriptions, ©)) {
- return AWS_OP_ERR;
- }
- }
- return AWS_OP_SUCCESS;
- }
- static size_t s_aws_mqtt5_packet_subscribe_compute_storage_size(
- const struct aws_mqtt5_packet_subscribe_view *subscribe_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- subscribe_view->user_properties, subscribe_view->user_property_count);
- for (size_t i = 0; i < subscribe_view->subscription_count; ++i) {
- const struct aws_mqtt5_subscription_view *subscription = &subscribe_view->subscriptions[i];
- storage_size += subscription->topic_filter.len;
- }
- return storage_size;
- }
- int aws_mqtt5_packet_subscribe_storage_init(
- struct aws_mqtt5_packet_subscribe_storage *subscribe_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_subscribe_view *subscribe_options) {
- AWS_ZERO_STRUCT(*subscribe_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_subscribe_compute_storage_size(subscribe_options);
- if (aws_byte_buf_init(&subscribe_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_subscribe_view *storage_view = &subscribe_storage->storage_view;
- storage_view->packet_id = subscribe_options->packet_id;
- if (subscribe_options->subscription_identifier != NULL) {
- subscribe_storage->subscription_identifier = *subscribe_options->subscription_identifier;
- storage_view->subscription_identifier = &subscribe_storage->subscription_identifier;
- }
- if (s_aws_mqtt5_packet_subscribe_storage_init_subscriptions(
- subscribe_storage, allocator, subscribe_options->subscription_count, subscribe_options->subscriptions)) {
- return AWS_OP_ERR;
- }
- storage_view->subscription_count = aws_array_list_length(&subscribe_storage->subscriptions);
- storage_view->subscriptions = subscribe_storage->subscriptions.data;
- if (aws_mqtt5_user_property_set_init_with_storage(
- &subscribe_storage->user_properties,
- allocator,
- &subscribe_storage->storage,
- subscribe_options->user_property_count,
- subscribe_options->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&subscribe_storage->user_properties);
- storage_view->user_properties = subscribe_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_subscribe_storage_init_from_external_storage(
- struct aws_mqtt5_packet_subscribe_storage *subscribe_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*subscribe_storage);
- if (aws_mqtt5_user_property_set_init(&subscribe_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_init_dynamic(
- &subscribe_storage->subscriptions, allocator, 0, sizeof(struct aws_mqtt5_subscription_view))) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- static void s_aws_mqtt5_operation_subscribe_complete(
- struct aws_mqtt5_operation *operation,
- int error_code,
- enum aws_mqtt5_packet_type packet_type,
- const void *completion_view) {
- (void)packet_type;
- struct aws_mqtt5_operation_subscribe *subscribe_op = operation->impl;
- if (subscribe_op->completion_options.completion_callback != NULL) {
- (*subscribe_op->completion_options.completion_callback)(
- completion_view, error_code, subscribe_op->completion_options.completion_user_data);
- }
- }
- static void s_aws_mqtt5_operation_subscribe_set_packet_id(
- struct aws_mqtt5_operation *operation,
- aws_mqtt5_packet_id_t packet_id) {
- struct aws_mqtt5_operation_subscribe *subscribe_op = operation->impl;
- subscribe_op->options_storage.storage_view.packet_id = packet_id;
- }
- static aws_mqtt5_packet_id_t *s_aws_mqtt5_operation_subscribe_get_packet_id_address(
- const struct aws_mqtt5_operation *operation) {
- struct aws_mqtt5_operation_subscribe *subscribe_op = operation->impl;
- return &subscribe_op->options_storage.storage_view.packet_id;
- }
- static struct aws_mqtt5_operation_vtable s_subscribe_operation_vtable = {
- .aws_mqtt5_operation_completion_fn = s_aws_mqtt5_operation_subscribe_complete,
- .aws_mqtt5_operation_set_packet_id_fn = s_aws_mqtt5_operation_subscribe_set_packet_id,
- .aws_mqtt5_operation_get_packet_id_address_fn = s_aws_mqtt5_operation_subscribe_get_packet_id_address,
- .aws_mqtt5_operation_validate_vs_connection_settings_fn = NULL,
- };
- static void s_destroy_operation_subscribe(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_subscribe *subscribe_op = object;
- aws_mqtt5_packet_subscribe_storage_clean_up(&subscribe_op->options_storage);
- aws_mem_release(subscribe_op->allocator, subscribe_op);
- }
- struct aws_mqtt5_operation_subscribe *aws_mqtt5_operation_subscribe_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_client *client,
- const struct aws_mqtt5_packet_subscribe_view *subscribe_options,
- const struct aws_mqtt5_subscribe_completion_options *completion_options) {
- AWS_PRECONDITION(allocator != NULL);
- AWS_PRECONDITION(subscribe_options != NULL);
- if (aws_mqtt5_packet_subscribe_view_validate(subscribe_options)) {
- return NULL;
- }
- if (subscribe_options->packet_id != 0) {
- AWS_LOGF_DEBUG(
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_subscribe_view packet id must be zero",
- (void *)subscribe_options);
- aws_raise_error(AWS_ERROR_MQTT5_SUBSCRIBE_OPTIONS_VALIDATION);
- return NULL;
- }
- if (client != NULL && client->config->extended_validation_and_flow_control_options != AWS_MQTT5_EVAFCO_NONE) {
- if (aws_mqtt5_packet_subscribe_view_validate_vs_iot_core(subscribe_options)) {
- return NULL;
- }
- }
- struct aws_mqtt5_operation_subscribe *subscribe_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_subscribe));
- if (subscribe_op == NULL) {
- return NULL;
- }
- subscribe_op->allocator = allocator;
- subscribe_op->base.vtable = &s_subscribe_operation_vtable;
- subscribe_op->base.packet_type = AWS_MQTT5_PT_SUBSCRIBE;
- aws_ref_count_init(&subscribe_op->base.ref_count, subscribe_op, s_destroy_operation_subscribe);
- subscribe_op->base.impl = subscribe_op;
- if (aws_mqtt5_packet_subscribe_storage_init(&subscribe_op->options_storage, allocator, subscribe_options)) {
- goto error;
- }
- subscribe_op->base.packet_view = &subscribe_op->options_storage.storage_view;
- if (completion_options != NULL) {
- subscribe_op->completion_options = *completion_options;
- }
- return subscribe_op;
- error:
- aws_mqtt5_operation_release(&subscribe_op->base);
- return NULL;
- }
- /*********************************************************************************************************************
- * Suback
- ********************************************************************************************************************/
- static size_t s_aws_mqtt5_packet_suback_compute_storage_size(const struct aws_mqtt5_packet_suback_view *suback_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- suback_view->user_properties, suback_view->user_property_count);
- if (suback_view->reason_string != NULL) {
- storage_size += suback_view->reason_string->len;
- }
- return storage_size;
- }
- static int s_aws_mqtt5_packet_suback_storage_init_reason_codes(
- struct aws_mqtt5_packet_suback_storage *suback_storage,
- struct aws_allocator *allocator,
- size_t reason_code_count,
- const enum aws_mqtt5_suback_reason_code *reason_codes) {
- if (aws_array_list_init_dynamic(
- &suback_storage->reason_codes, allocator, reason_code_count, sizeof(enum aws_mqtt5_suback_reason_code))) {
- return AWS_OP_ERR;
- }
- for (size_t i = 0; i < reason_code_count; ++i) {
- aws_array_list_push_back(&suback_storage->reason_codes, &reason_codes[i]);
- }
- return AWS_OP_SUCCESS;
- }
- AWS_MQTT_API int aws_mqtt5_packet_suback_storage_init(
- struct aws_mqtt5_packet_suback_storage *suback_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_suback_view *suback_view) {
- AWS_ZERO_STRUCT(*suback_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_suback_compute_storage_size(suback_view);
- if (aws_byte_buf_init(&suback_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_suback_view *storage_view = &suback_storage->storage_view;
- storage_view->packet_id = suback_view->packet_id;
- if (suback_view->reason_string != NULL) {
- suback_storage->reason_string = *suback_view->reason_string;
- if (aws_byte_buf_append_and_update(&suback_storage->storage, &suback_storage->reason_string)) {
- return AWS_OP_ERR;
- }
- storage_view->reason_string = &suback_storage->reason_string;
- }
- if (s_aws_mqtt5_packet_suback_storage_init_reason_codes(
- suback_storage, allocator, suback_view->reason_code_count, suback_view->reason_codes)) {
- return AWS_OP_ERR;
- }
- storage_view->reason_code_count = aws_array_list_length(&suback_storage->reason_codes);
- storage_view->reason_codes = suback_storage->reason_codes.data;
- if (aws_mqtt5_user_property_set_init_with_storage(
- &suback_storage->user_properties,
- allocator,
- &suback_storage->storage,
- suback_view->user_property_count,
- suback_view->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&suback_storage->user_properties);
- storage_view->user_properties = suback_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_suback_storage_init_from_external_storage(
- struct aws_mqtt5_packet_suback_storage *suback_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*suback_storage);
- if (aws_mqtt5_user_property_set_init(&suback_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_init_dynamic(
- &suback_storage->reason_codes, allocator, 0, sizeof(enum aws_mqtt5_suback_reason_code))) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_suback_storage_clean_up(struct aws_mqtt5_packet_suback_storage *suback_storage) {
- if (suback_storage == NULL) {
- return;
- }
- aws_mqtt5_user_property_set_clean_up(&suback_storage->user_properties);
- aws_array_list_clean_up(&suback_storage->reason_codes);
- aws_byte_buf_clean_up(&suback_storage->storage);
- }
- void aws_mqtt5_packet_suback_view_log(
- const struct aws_mqtt5_packet_suback_view *suback_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_suback_view packet id set to %d",
- (void *)suback_view,
- (int)suback_view->packet_id);
- for (size_t i = 0; i < suback_view->reason_code_count; ++i) {
- enum aws_mqtt5_suback_reason_code reason_code = suback_view->reason_codes[i];
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_suback_view topic #%zu, reason code %d (%s)",
- (void *)suback_view,
- i,
- (int)reason_code,
- aws_mqtt5_suback_reason_code_to_c_string(reason_code));
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- suback_view->user_properties,
- suback_view->user_property_count,
- (void *)suback_view,
- level,
- "aws_mqtt5_packet_suback_view");
- }
- /*********************************************************************************************************************
- * Unsuback
- ********************************************************************************************************************/
- static size_t s_aws_mqtt5_packet_unsuback_compute_storage_size(
- const struct aws_mqtt5_packet_unsuback_view *unsuback_view) {
- size_t storage_size = s_aws_mqtt5_user_property_set_compute_storage_size(
- unsuback_view->user_properties, unsuback_view->user_property_count);
- if (unsuback_view->reason_string != NULL) {
- storage_size += unsuback_view->reason_string->len;
- }
- return storage_size;
- }
- static int s_aws_mqtt5_packet_unsuback_storage_init_reason_codes(
- struct aws_mqtt5_packet_unsuback_storage *unsuback_storage,
- struct aws_allocator *allocator,
- size_t reason_code_count,
- const enum aws_mqtt5_unsuback_reason_code *reason_codes) {
- if (aws_array_list_init_dynamic(
- &unsuback_storage->reason_codes,
- allocator,
- reason_code_count,
- sizeof(enum aws_mqtt5_unsuback_reason_code))) {
- return AWS_OP_ERR;
- }
- for (size_t i = 0; i < reason_code_count; ++i) {
- aws_array_list_push_back(&unsuback_storage->reason_codes, &reason_codes[i]);
- }
- return AWS_OP_SUCCESS;
- }
- AWS_MQTT_API int aws_mqtt5_packet_unsuback_storage_init(
- struct aws_mqtt5_packet_unsuback_storage *unsuback_storage,
- struct aws_allocator *allocator,
- const struct aws_mqtt5_packet_unsuback_view *unsuback_view) {
- AWS_ZERO_STRUCT(*unsuback_storage);
- size_t storage_capacity = s_aws_mqtt5_packet_unsuback_compute_storage_size(unsuback_view);
- if (aws_byte_buf_init(&unsuback_storage->storage, allocator, storage_capacity)) {
- return AWS_OP_ERR;
- }
- struct aws_mqtt5_packet_unsuback_view *storage_view = &unsuback_storage->storage_view;
- storage_view->packet_id = unsuback_view->packet_id;
- if (unsuback_view->reason_string != NULL) {
- unsuback_storage->reason_string = *unsuback_view->reason_string;
- if (aws_byte_buf_append_and_update(&unsuback_storage->storage, &unsuback_storage->reason_string)) {
- return AWS_OP_ERR;
- }
- storage_view->reason_string = &unsuback_storage->reason_string;
- }
- if (s_aws_mqtt5_packet_unsuback_storage_init_reason_codes(
- unsuback_storage, allocator, unsuback_view->reason_code_count, unsuback_view->reason_codes)) {
- return AWS_OP_ERR;
- }
- storage_view->reason_code_count = aws_array_list_length(&unsuback_storage->reason_codes);
- storage_view->reason_codes = unsuback_storage->reason_codes.data;
- if (aws_mqtt5_user_property_set_init_with_storage(
- &unsuback_storage->user_properties,
- allocator,
- &unsuback_storage->storage,
- unsuback_view->user_property_count,
- unsuback_view->user_properties)) {
- return AWS_OP_ERR;
- }
- storage_view->user_property_count = aws_mqtt5_user_property_set_size(&unsuback_storage->user_properties);
- storage_view->user_properties = unsuback_storage->user_properties.properties.data;
- return AWS_OP_SUCCESS;
- }
- int aws_mqtt5_packet_unsuback_storage_init_from_external_storage(
- struct aws_mqtt5_packet_unsuback_storage *unsuback_storage,
- struct aws_allocator *allocator) {
- AWS_ZERO_STRUCT(*unsuback_storage);
- if (aws_mqtt5_user_property_set_init(&unsuback_storage->user_properties, allocator)) {
- return AWS_OP_ERR;
- }
- if (aws_array_list_init_dynamic(
- &unsuback_storage->reason_codes, allocator, 0, sizeof(enum aws_mqtt5_unsuback_reason_code))) {
- return AWS_OP_ERR;
- }
- return AWS_OP_SUCCESS;
- }
- void aws_mqtt5_packet_unsuback_storage_clean_up(struct aws_mqtt5_packet_unsuback_storage *unsuback_storage) {
- if (unsuback_storage == NULL) {
- return;
- }
- aws_mqtt5_user_property_set_clean_up(&unsuback_storage->user_properties);
- aws_array_list_clean_up(&unsuback_storage->reason_codes);
- aws_byte_buf_clean_up(&unsuback_storage->storage);
- }
- void aws_mqtt5_packet_unsuback_view_log(
- const struct aws_mqtt5_packet_unsuback_view *unsuback_view,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsuback_view packet id set to %d",
- (void *)unsuback_view,
- (int)unsuback_view->packet_id);
- for (size_t i = 0; i < unsuback_view->reason_code_count; ++i) {
- enum aws_mqtt5_unsuback_reason_code reason_code = unsuback_view->reason_codes[i];
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_packet_unsuback_view topic #%zu, reason code %d (%s)",
- (void *)unsuback_view,
- i,
- (int)reason_code,
- aws_mqtt5_unsuback_reason_code_to_c_string(reason_code));
- }
- s_aws_mqtt5_user_property_set_log(
- log_handle,
- unsuback_view->user_properties,
- unsuback_view->user_property_count,
- (void *)unsuback_view,
- level,
- "aws_mqtt5_packet_unsuback_view");
- }
- /*********************************************************************************************************************
- * PINGREQ
- ********************************************************************************************************************/
- static void s_destroy_operation_pingreq(void *object) {
- if (object == NULL) {
- return;
- }
- struct aws_mqtt5_operation_pingreq *pingreq_op = object;
- aws_mem_release(pingreq_op->allocator, pingreq_op);
- }
- struct aws_mqtt5_operation_pingreq *aws_mqtt5_operation_pingreq_new(struct aws_allocator *allocator) {
- struct aws_mqtt5_operation_pingreq *pingreq_op =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_operation_pingreq));
- if (pingreq_op == NULL) {
- return NULL;
- }
- pingreq_op->allocator = allocator;
- pingreq_op->base.vtable = &s_empty_operation_vtable;
- pingreq_op->base.packet_type = AWS_MQTT5_PT_PINGREQ;
- aws_ref_count_init(&pingreq_op->base.ref_count, pingreq_op, s_destroy_operation_pingreq);
- pingreq_op->base.impl = pingreq_op;
- return pingreq_op;
- }
- /*********************************************************************************************************************
- * Client storage options
- ********************************************************************************************************************/
- int aws_mqtt5_client_options_validate(const struct aws_mqtt5_client_options *options) {
- if (options == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "null mqtt5 client configuration options");
- return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT);
- }
- if (options->host_name.len == 0) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "host name not set in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- if (options->bootstrap == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "client bootstrap not set in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- /* forbid no-timeout until someone convinces me otherwise */
- if (options->socket_options != NULL) {
- if (options->socket_options->type == AWS_SOCKET_DGRAM || options->socket_options->connect_timeout_ms == 0) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "invalid socket options in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- }
- if (options->http_proxy_options != NULL) {
- if (options->http_proxy_options->host.len == 0) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "proxy host name not set in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- if (options->http_proxy_options->port == 0) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "proxy port not set in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- }
- /* can't think of why you'd ever want an MQTT client without lifecycle event notifications */
- if (options->lifecycle_event_handler == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "lifecycle event handler not set in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- if (options->publish_received_handler == NULL) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "publish received not set in mqtt5 client configuration");
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- if (aws_mqtt5_packet_connect_view_validate(options->connect_options)) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "invalid CONNECT options in mqtt5 client configuration");
- return AWS_OP_ERR;
- }
- /* The client will not behave properly if ping timeout is not significantly shorter than the keep alive interval */
- if (options->connect_options->keep_alive_interval_seconds > 0) {
- uint64_t keep_alive_ms = aws_timestamp_convert(
- options->connect_options->keep_alive_interval_seconds, AWS_TIMESTAMP_SECS, AWS_TIMESTAMP_MILLIS, NULL);
- uint64_t one_second_ms = aws_timestamp_convert(1, AWS_TIMESTAMP_SECS, AWS_TIMESTAMP_MILLIS, NULL);
- uint64_t ping_timeout_ms = options->ping_timeout_ms;
- if (ping_timeout_ms == 0) {
- ping_timeout_ms = AWS_MQTT5_CLIENT_DEFAULT_PING_TIMEOUT_MS;
- }
- if (ping_timeout_ms + one_second_ms > keep_alive_ms) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "keep alive interval is too small relative to ping timeout interval");
- return AWS_OP_ERR;
- }
- }
- if (options->extended_validation_and_flow_control_options != AWS_MQTT5_EVAFCO_NONE) {
- if (options->connect_options->client_id.len > AWS_IOT_CORE_MAXIMUM_CLIENT_ID_LENGTH) {
- AWS_LOGF_ERROR(
- AWS_LS_MQTT5_GENERAL,
- "AWS IoT Core limits client_id to be less than or equal to %d bytes in length",
- (int)AWS_IOT_CORE_MAXIMUM_CLIENT_ID_LENGTH);
- return aws_raise_error(AWS_ERROR_MQTT5_CLIENT_OPTIONS_VALIDATION);
- }
- }
- return AWS_OP_SUCCESS;
- }
- static void s_log_tls_connection_options(
- struct aws_logger *log_handle,
- const struct aws_mqtt5_client_options_storage *options_storage,
- const struct aws_tls_connection_options *tls_options,
- enum aws_log_level level,
- const char *log_text) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options set:",
- (void *)options_storage,
- log_text);
- if (tls_options->advertise_alpn_message && tls_options->alpn_list) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options alpn protocol list set to \"%s\"",
- (void *)options_storage,
- log_text,
- aws_string_c_str(tls_options->alpn_list));
- } else {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options alpn not used",
- (void *)options_storage,
- log_text);
- }
- if (tls_options->server_name) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options SNI value set to \"%s\"",
- (void *)options_storage,
- log_text,
- aws_string_c_str(tls_options->server_name));
- } else {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options SNI not used",
- (void *)options_storage,
- log_text);
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options tls context set to (%p)",
- (void *)options_storage,
- log_text,
- (void *)(tls_options->ctx));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage %s tls options handshake timeout set to %" PRIu32,
- (void *)options_storage,
- log_text,
- tls_options->timeout_ms);
- }
- static void s_log_topic_aliasing_options(
- struct aws_logger *log_handle,
- const struct aws_mqtt5_client_options_storage *options_storage,
- const struct aws_mqtt5_client_topic_alias_options *topic_aliasing_options,
- enum aws_log_level level) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage outbound topic aliasing behavior set to %d (%s)",
- (void *)options_storage,
- (int)topic_aliasing_options->outbound_topic_alias_behavior,
- aws_mqtt5_outbound_topic_alias_behavior_type_to_c_string(
- topic_aliasing_options->outbound_topic_alias_behavior));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage maximum outbound topic alias cache size set to %" PRIu16,
- (void *)options_storage,
- topic_aliasing_options->outbound_alias_cache_max_size);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage inbound topic aliasing behavior set to %d (%s)",
- (void *)options_storage,
- (int)topic_aliasing_options->inbound_topic_alias_behavior,
- aws_mqtt5_inbound_topic_alias_behavior_type_to_c_string(topic_aliasing_options->inbound_topic_alias_behavior));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage inbound topic alias cache size set to %" PRIu16,
- (void *)options_storage,
- topic_aliasing_options->inbound_alias_cache_size);
- }
- void aws_mqtt5_client_options_storage_log(
- const struct aws_mqtt5_client_options_storage *options_storage,
- enum aws_log_level level) {
- struct aws_logger *log_handle = aws_logger_get_conditional(AWS_LS_MQTT5_GENERAL, level);
- if (log_handle == NULL) {
- return;
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage host name set to %s",
- (void *)options_storage,
- aws_string_c_str(options_storage->host_name));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage port set to %" PRIu16,
- (void *)options_storage,
- options_storage->port);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage client bootstrap set to (%p)",
- (void *)options_storage,
- (void *)options_storage->bootstrap);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage socket options set to: type = %d, domain = %d, connect_timeout_ms = "
- "%" PRIu32,
- (void *)options_storage,
- (int)options_storage->socket_options.type,
- (int)options_storage->socket_options.domain,
- options_storage->socket_options.connect_timeout_ms);
- if (options_storage->socket_options.keepalive) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage socket keepalive options set to: keep_alive_interval_sec = "
- "%" PRIu16 ", "
- "keep_alive_timeout_sec = %" PRIu16 ", keep_alive_max_failed_probes = %" PRIu16,
- (void *)options_storage,
- options_storage->socket_options.keep_alive_interval_sec,
- options_storage->socket_options.keep_alive_timeout_sec,
- options_storage->socket_options.keep_alive_max_failed_probes);
- }
- if (options_storage->tls_options_ptr != NULL) {
- s_log_tls_connection_options(log_handle, options_storage, options_storage->tls_options_ptr, level, "");
- }
- if (options_storage->http_proxy_config != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage using http proxy:",
- (void *)options_storage);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage http proxy host name set to " PRInSTR,
- (void *)options_storage,
- AWS_BYTE_CURSOR_PRI(options_storage->http_proxy_options.host));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage http proxy port set to %" PRIu16,
- (void *)options_storage,
- options_storage->http_proxy_options.port);
- if (options_storage->http_proxy_options.tls_options != NULL) {
- s_log_tls_connection_options(
- log_handle, options_storage, options_storage->tls_options_ptr, level, "http proxy");
- }
- /* ToDo: add (and use) an API to proxy strategy that returns a debug string (Basic, Adaptive, etc...) */
- if (options_storage->http_proxy_options.proxy_strategy != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage http proxy strategy set to (%p)",
- (void *)options_storage,
- (void *)options_storage->http_proxy_options.proxy_strategy);
- }
- }
- if (options_storage->websocket_handshake_transform != NULL) {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage enabling websockets",
- (void *)options_storage);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage websocket handshake transform user data set to (%p)",
- (void *)options_storage,
- options_storage->websocket_handshake_transform_user_data);
- } else {
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: mqtt5_client_options_storage disabling websockets",
- (void *)options_storage);
- }
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage session behavior set to %d (%s)",
- (void *)options_storage,
- (int)options_storage->session_behavior,
- aws_mqtt5_client_session_behavior_type_to_c_string(options_storage->session_behavior));
- s_log_topic_aliasing_options(log_handle, options_storage, &options_storage->topic_aliasing_options, level);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage extended validation and flow control options set to %d (%s)",
- (void *)options_storage,
- (int)options_storage->extended_validation_and_flow_control_options,
- aws_mqtt5_extended_validation_and_flow_control_options_to_c_string(
- options_storage->extended_validation_and_flow_control_options));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage operation queue behavior set to %d (%s)",
- (void *)options_storage,
- (int)options_storage->offline_queue_behavior,
- aws_mqtt5_client_operation_queue_behavior_type_to_c_string(options_storage->offline_queue_behavior));
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage reconnect jitter mode set to %d",
- (void *)options_storage,
- (int)options_storage->retry_jitter_mode);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: mqtt5_client_options_storage reconnect delay min set to %" PRIu64 " ms, max set to %" PRIu64 " ms",
- (void *)options_storage,
- options_storage->min_reconnect_delay_ms,
- options_storage->max_reconnect_delay_ms);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage minimum necessary connection time in order to reset the reconnect "
- "delay "
- "set "
- "to %" PRIu64 " ms",
- (void *)options_storage,
- options_storage->min_connected_time_to_reset_reconnect_delay_ms);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage ping timeout interval set to %" PRIu32 " ms",
- (void *)options_storage,
- options_storage->ping_timeout_ms);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage connack timeout interval set to %" PRIu32 " ms",
- (void *)options_storage,
- options_storage->connack_timeout_ms);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage connect options:",
- (void *)options_storage);
- aws_mqtt5_packet_connect_view_log(&options_storage->connect.storage_view, level);
- AWS_LOGUF(
- log_handle,
- level,
- AWS_LS_MQTT5_GENERAL,
- "id=%p: aws_mqtt5_client_options_storage lifecycle event handler user data set to (%p)",
- (void *)options_storage,
- options_storage->lifecycle_event_handler_user_data);
- }
- void aws_mqtt5_client_options_storage_destroy(struct aws_mqtt5_client_options_storage *options_storage) {
- if (options_storage == NULL) {
- return;
- }
- aws_string_destroy(options_storage->host_name);
- aws_client_bootstrap_release(options_storage->bootstrap);
- aws_tls_connection_options_clean_up(&options_storage->tls_options);
- aws_http_proxy_config_destroy(options_storage->http_proxy_config);
- aws_mqtt5_packet_connect_storage_clean_up(&options_storage->connect);
- aws_mem_release(options_storage->allocator, options_storage);
- }
- static void s_apply_zero_valued_defaults_to_client_options_storage(
- struct aws_mqtt5_client_options_storage *options_storage) {
- if (options_storage->min_reconnect_delay_ms == 0) {
- options_storage->min_reconnect_delay_ms = AWS_MQTT5_CLIENT_DEFAULT_MIN_RECONNECT_DELAY_MS;
- }
- if (options_storage->max_reconnect_delay_ms == 0) {
- options_storage->max_reconnect_delay_ms = AWS_MQTT5_CLIENT_DEFAULT_MAX_RECONNECT_DELAY_MS;
- }
- if (options_storage->min_connected_time_to_reset_reconnect_delay_ms == 0) {
- options_storage->min_connected_time_to_reset_reconnect_delay_ms =
- AWS_MQTT5_CLIENT_DEFAULT_MIN_CONNECTED_TIME_TO_RESET_RECONNECT_DELAY_MS;
- }
- if (options_storage->ping_timeout_ms == 0) {
- options_storage->ping_timeout_ms = AWS_MQTT5_CLIENT_DEFAULT_PING_TIMEOUT_MS;
- }
- if (options_storage->connack_timeout_ms == 0) {
- options_storage->connack_timeout_ms = AWS_MQTT5_CLIENT_DEFAULT_CONNACK_TIMEOUT_MS;
- }
- if (options_storage->ack_timeout_seconds == 0) {
- options_storage->ack_timeout_seconds = AWS_MQTT5_CLIENT_DEFAULT_OPERATION_TIMEOUNT_SECONDS;
- }
- if (options_storage->topic_aliasing_options.inbound_alias_cache_size == 0) {
- options_storage->topic_aliasing_options.inbound_alias_cache_size =
- AWS_MQTT5_CLIENT_DEFAULT_INBOUND_TOPIC_ALIAS_CACHE_SIZE;
- }
- if (options_storage->topic_aliasing_options.outbound_alias_cache_max_size == 0) {
- options_storage->topic_aliasing_options.outbound_alias_cache_max_size =
- AWS_MQTT5_CLIENT_DEFAULT_OUTBOUND_TOPIC_ALIAS_CACHE_SIZE;
- }
- }
- struct aws_mqtt5_client_options_storage *aws_mqtt5_client_options_storage_new(
- struct aws_allocator *allocator,
- const struct aws_mqtt5_client_options *options) {
- AWS_PRECONDITION(allocator != NULL);
- AWS_PRECONDITION(options != NULL);
- if (aws_mqtt5_client_options_validate(options)) {
- return NULL;
- }
- struct aws_mqtt5_client_options_storage *options_storage =
- aws_mem_calloc(allocator, 1, sizeof(struct aws_mqtt5_client_options_storage));
- if (options_storage == NULL) {
- return NULL;
- }
- options_storage->allocator = allocator;
- options_storage->host_name = aws_string_new_from_cursor(allocator, &options->host_name);
- if (options_storage->host_name == NULL) {
- goto error;
- }
- options_storage->port = options->port;
- options_storage->bootstrap = aws_client_bootstrap_acquire(options->bootstrap);
- if (options->socket_options != NULL) {
- options_storage->socket_options = *options->socket_options;
- } else {
- options_storage->socket_options.type = AWS_SOCKET_STREAM;
- options_storage->socket_options.connect_timeout_ms = AWS_MQTT5_DEFAULT_SOCKET_CONNECT_TIMEOUT_MS;
- }
- if (options->tls_options != NULL) {
- if (aws_tls_connection_options_copy(&options_storage->tls_options, options->tls_options)) {
- goto error;
- }
- options_storage->tls_options_ptr = &options_storage->tls_options;
- if (!options_storage->tls_options.server_name) {
- struct aws_byte_cursor host_name_cur = aws_byte_cursor_from_string(options_storage->host_name);
- if (aws_tls_connection_options_set_server_name(&options_storage->tls_options, allocator, &host_name_cur)) {
- AWS_LOGF_ERROR(AWS_LS_MQTT5_GENERAL, "Failed to set TLS Connection Options server name");
- goto error;
- }
- }
- }
- if (options->http_proxy_options != NULL) {
- options_storage->http_proxy_config =
- aws_http_proxy_config_new_from_proxy_options(allocator, options->http_proxy_options);
- if (options_storage->http_proxy_config == NULL) {
- goto error;
- }
- aws_http_proxy_options_init_from_config(
- &options_storage->http_proxy_options, options_storage->http_proxy_config);
- }
- options_storage->websocket_handshake_transform = options->websocket_handshake_transform;
- options_storage->websocket_handshake_transform_user_data = options->websocket_handshake_transform_user_data;
- options_storage->publish_received_handler = options->publish_received_handler;
- options_storage->publish_received_handler_user_data = options->publish_received_handler_user_data;
- options_storage->session_behavior = options->session_behavior;
- options_storage->extended_validation_and_flow_control_options =
- options->extended_validation_and_flow_control_options;
- options_storage->offline_queue_behavior = options->offline_queue_behavior;
- options_storage->retry_jitter_mode = options->retry_jitter_mode;
- options_storage->min_reconnect_delay_ms = options->min_reconnect_delay_ms;
- options_storage->max_reconnect_delay_ms = options->max_reconnect_delay_ms;
- options_storage->min_connected_time_to_reset_reconnect_delay_ms =
- options->min_connected_time_to_reset_reconnect_delay_ms;
- options_storage->ping_timeout_ms = options->ping_timeout_ms;
- options_storage->connack_timeout_ms = options->connack_timeout_ms;
- options_storage->ack_timeout_seconds = options->ack_timeout_seconds;
- if (options->topic_aliasing_options != NULL) {
- options_storage->topic_aliasing_options = *options->topic_aliasing_options;
- }
- if (aws_mqtt5_packet_connect_storage_init(&options_storage->connect, allocator, options->connect_options)) {
- goto error;
- }
- options_storage->lifecycle_event_handler = options->lifecycle_event_handler;
- options_storage->lifecycle_event_handler_user_data = options->lifecycle_event_handler_user_data;
- options_storage->client_termination_handler = options->client_termination_handler;
- options_storage->client_termination_handler_user_data = options->client_termination_handler_user_data;
- s_apply_zero_valued_defaults_to_client_options_storage(options_storage);
- return options_storage;
- error:
- aws_mqtt5_client_options_storage_destroy(options_storage);
- return NULL;
- }
- struct aws_mqtt5_operation_disconnect *aws_mqtt5_operation_disconnect_acquire(
- struct aws_mqtt5_operation_disconnect *disconnect_op) {
- if (disconnect_op != NULL) {
- aws_mqtt5_operation_acquire(&disconnect_op->base);
- }
- return disconnect_op;
- }
- struct aws_mqtt5_operation_disconnect *aws_mqtt5_operation_disconnect_release(
- struct aws_mqtt5_operation_disconnect *disconnect_op) {
- if (disconnect_op != NULL) {
- aws_mqtt5_operation_release(&disconnect_op->base);
- }
- return NULL;
- }
|