You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
IoT-For-Beginners/translations/bn/2-farm/lessons/4-migrate-your-plant-to-the.../wio-terminal-connect-hub.md

21 KiB

আপনার IoT ডিভাইসকে ক্লাউডের সাথে সংযুক্ত করুন - Wio Terminal

এই পাঠের এই অংশে, আপনি আপনার Wio Terminal-কে IoT Hub-এর সাথে সংযুক্ত করবেন, টেলিমেট্রি পাঠানোর এবং কমান্ড গ্রহণ করার জন্য।

আপনার ডিভাইসকে IoT Hub-এর সাথে সংযুক্ত করুন

পরবর্তী ধাপটি হলো আপনার ডিভাইসকে IoT Hub-এর সাথে সংযুক্ত করা।

কাজ - IoT Hub-এর সাথে সংযুক্ত করুন

  1. VS Code-এ soil-moisture-sensor প্রকল্পটি খুলুন।

  2. platformio.ini ফাইলটি খুলুন। knolleary/PubSubClient লাইব্রেরি ডিপেনডেন্সি সরিয়ে ফেলুন। এটি পাবলিক MQTT ব্রোকারের সাথে সংযোগ করার জন্য ব্যবহৃত হয়েছিল, এবং IoT Hub-এর সাথে সংযোগ করার জন্য এটি প্রয়োজন নেই।

  3. নিম্নলিখিত লাইব্রেরি ডিপেনডেন্সি যোগ করুন:

    seeed-studio/Seeed Arduino RTC @ 2.0.0
    arduino-libraries/AzureIoTHub @ 1.6.0
    azure/AzureIoTUtility @ 1.6.1
    azure/AzureIoTProtocol_MQTT @ 1.6.0
    azure/AzureIoTProtocol_HTTP @ 1.6.0
    azure/AzureIoTSocket_WiFi @ 1.0.2
    

    Seeed Arduino RTC লাইব্রেরি Wio Terminal-এ একটি রিয়েল-টাইম ক্লকের সাথে ইন্টারঅ্যাক্ট করার কোড প্রদান করে, যা সময় ট্র্যাক করার জন্য ব্যবহৃত হয়। বাকি লাইব্রেরিগুলি আপনার IoT ডিভাইসকে IoT Hub-এর সাথে সংযুক্ত করতে সাহায্য করে।

  4. platformio.ini ফাইলের নিচে নিম্নলিখিত যোগ করুন:

    build_flags =
        -DDONT_USE_UPLOADTOBLOB
    

    এটি একটি কম্পাইলার ফ্ল্যাগ সেট করে যা Arduino IoT Hub কোড কম্পাইল করার সময় প্রয়োজন।

  5. config.h হেডার ফাইলটি খুলুন। সমস্ত MQTT সেটিংস সরিয়ে ফেলুন এবং ডিভাইস সংযোগ স্ট্রিং-এর জন্য নিম্নলিখিত কনস্ট্যান্ট যোগ করুন:

    // IoT Hub settings
    const char *CONNECTION_STRING = "<connection string>";
    

    <connection string>-এর জায়গায় আপনার ডিভাইসের সংযোগ স্ট্রিংটি বসান যা আপনি আগে কপি করেছিলেন।

  6. IoT Hub-এর সাথে সংযোগ একটি সময়-ভিত্তিক টোকেন ব্যবহার করে। এর মানে হলো IoT ডিভাইসকে বর্তমান সময় জানতে হবে। Windows, macOS বা Linux-এর মতো অপারেটিং সিস্টেমের বিপরীতে, মাইক্রোকন্ট্রোলার স্বয়ংক্রিয়ভাবে ইন্টারনেটের মাধ্যমে বর্তমান সময় সিঙ্ক্রোনাইজ করে না। এর মানে হলো আপনাকে একটি NTP সার্ভার থেকে বর্তমান সময় পাওয়ার জন্য কোড যোগ করতে হবে। একবার সময় পাওয়া গেলে, এটি Wio Terminal-এর রিয়েল-টাইম ক্লকে সংরক্ষণ করা যেতে পারে, যা ডিভাইস পাওয়ার হারালে সঠিক সময় পুনরায় অনুরোধ করার অনুমতি দেয়। ntp.h নামে একটি নতুন ফাইল তৈরি করুন এবং নিম্নলিখিত কোড যোগ করুন:

    #pragma once
    
    #include "DateTime.h"
    #include <time.h>
    #include "samd/NTPClientAz.h"
    #include <sys/time.h>
    
    static void initTime()
    {
        WiFiUDP _udp;
        time_t epochTime = (time_t)-1;
        NTPClientAz ntpClient;
    
        ntpClient.begin();
    
        while (true)
        {
            epochTime = ntpClient.getEpochTime("0.pool.ntp.org");
    
            if (epochTime == (time_t)-1)
            {
                Serial.println("Fetching NTP epoch time failed! Waiting 2 seconds to retry.");
                delay(2000);
            }
            else
            {
                Serial.print("Fetched NTP epoch time is: ");
    
                char buff[32];
                sprintf(buff, "%.f", difftime(epochTime, (time_t)0));
                Serial.println(buff);
                break;
            }
        }
    
        ntpClient.end();
    
        struct timeval tv;
        tv.tv_sec = epochTime;
        tv.tv_usec = 0;
    
        settimeofday(&tv, NULL);
    }
    

    এই কোডের বিস্তারিত এই পাঠের আওতার বাইরে। এটি একটি initTime নামক ফাংশন সংজ্ঞায়িত করে যা একটি NTP সার্ভার থেকে বর্তমান সময় পায় এবং Wio Terminal-এর ক্লক সেট করতে এটি ব্যবহার করে।

  7. main.cpp ফাইলটি খুলুন এবং সমস্ত MQTT কোড সরিয়ে ফেলুন, যার মধ্যে রয়েছে PubSubClient.h হেডার ফাইল, PubSubClient ভেরিয়েবল ডিক্লারেশন, reconnectMQTTClient এবং createMQTTClient মেথড, এবং এই ভেরিয়েবল এবং মেথডগুলির কোনো কল। এই ফাইলটি শুধুমাত্র WiFi-তে সংযোগ করার, মাটি আর্দ্রতা পাওয়ার এবং এটি একটি JSON ডকুমেন্টে তৈরি করার কোড ধারণ করবে।

  8. main.cpp ফাইলের শীর্ষে নিম্নলিখিত #include ডিরেক্টিভ যোগ করুন IoT Hub লাইব্রেরি এবং সময় সেট করার জন্য হেডার ফাইল অন্তর্ভুক্ত করতে:

    #include <AzureIoTHub.h>
    #include <AzureIoTProtocol_MQTT.h>
    #include <iothubtransportmqtt.h>
    #include "ntp.h"
    
  9. setup ফাংশনের শেষে নিম্নলিখিত কল যোগ করুন বর্তমান সময় সেট করার জন্য:

    initTime();
    
  10. ফাইলের শীর্ষে, অন্তর্ভুক্ত ডিরেক্টিভগুলির ঠিক নিচে নিম্নলিখিত ভেরিয়েবল ডিক্লারেশন যোগ করুন:

    IOTHUB_DEVICE_CLIENT_LL_HANDLE _device_ll_handle;
    

    এটি একটি IOTHUB_DEVICE_CLIENT_LL_HANDLE ঘোষণা করে, যা IoT Hub-এর সাথে সংযোগের একটি হ্যান্ডেল।

  11. এর নিচে নিম্নলিখিত কোড যোগ করুন:

    static void connectionStatusCallback(IOTHUB_CLIENT_CONNECTION_STATUS result, IOTHUB_CLIENT_CONNECTION_STATUS_REASON reason, void *user_context)
    {
        if (result == IOTHUB_CLIENT_CONNECTION_AUTHENTICATED)
        {
            Serial.println("The device client is connected to iothub");
        }
        else
        {
            Serial.println("The device client has been disconnected");
        }
    }
    

    এটি একটি কলব্যাক ফাংশন ঘোষণা করে যা IoT Hub-এর সাথে সংযোগের স্ট্যাটাস পরিবর্তন হলে, যেমন সংযোগ বা সংযোগ বিচ্ছিন্ন হলে, কল করা হবে। স্ট্যাটাসটি সিরিয়াল পোর্টে পাঠানো হয়।

  12. এর নিচে IoT Hub-এর সাথে সংযোগ করার জন্য একটি ফাংশন যোগ করুন:

    void connectIoTHub()
    {
        IoTHub_Init();
    
        _device_ll_handle = IoTHubDeviceClient_LL_CreateFromConnectionString(CONNECTION_STRING, MQTT_Protocol);
    
        if (_device_ll_handle == NULL)
        {
            Serial.println("Failure creating Iothub device. Hint: Check your connection string.");
            return;
        }
    
        IoTHubDeviceClient_LL_SetConnectionStatusCallback(_device_ll_handle, connectionStatusCallback, NULL);
    }
    

    এই কোডটি IoT Hub লাইব্রেরি কোড ইনিশিয়ালাইজ করে, তারপর config.h হেডার ফাইলের সংযোগ স্ট্রিং ব্যবহার করে একটি সংযোগ তৈরি করে। এই সংযোগটি MQTT-এর উপর ভিত্তি করে। যদি সংযোগ ব্যর্থ হয়, এটি সিরিয়াল পোর্টে পাঠানো হয় - যদি আপনি আউটপুটে এটি দেখেন, সংযোগ স্ট্রিংটি পরীক্ষা করুন। অবশেষে সংযোগ স্ট্যাটাস কলব্যাক সেট আপ করা হয়।

  13. setup ফাংশনে initTime কলের নিচে এই ফাংশনটি কল করুন:

    connectIoTHub();
    
  14. MQTT ক্লায়েন্টের মতো, এই কোডটি একটি একক থ্রেডে চলে, তাই হাব দ্বারা পাঠানো এবং হাবে পাঠানো বার্তাগুলি প্রক্রিয়া করার জন্য সময় প্রয়োজন। loop ফাংশনের শীর্ষে নিম্নলিখিত যোগ করুন এটি করার জন্য:

    IoTHubDeviceClient_LL_DoWork(_device_ll_handle);
    
  15. এই কোডটি বিল্ড এবং আপলোড করুন। আপনি সিরিয়াল মনিটরে সংযোগ দেখতে পাবেন:

    Connecting to WiFi..
    Connected!
    Fetched NTP epoch time is: 1619983687
    Sending telemetry {"soil_moisture":391}
    The device client is connected to iothub
    

    আউটপুটে আপনি NTP সময় ফেচ করা এবং তারপর ডিভাইস ক্লায়েন্ট সংযোগ করতে দেখতে পাবেন। সংযোগ করতে কয়েক সেকেন্ড সময় লাগতে পারে, তাই ডিভাইস সংযোগ করার সময় আপনি আউটপুটে মাটি আর্দ্রতা দেখতে পারেন।

    💁 আপনি NTP-এর UNIX সময়কে একটি আরও পাঠযোগ্য সংস্করণে রূপান্তর করতে পারেন unixtimestamp.com-এর মতো একটি ওয়েবসাইট ব্যবহার করে।

টেলিমেট্রি পাঠান

এখন আপনার ডিভাইস সংযুক্ত হয়েছে, আপনি MQTT ব্রোকারের পরিবর্তে IoT Hub-এ টেলিমেট্রি পাঠাতে পারেন।

কাজ - টেলিমেট্রি পাঠান

  1. setup ফাংশনের উপরে নিম্নলিখিত ফাংশন যোগ করুন:

    void sendTelemetry(const char *telemetry)
    {
        IOTHUB_MESSAGE_HANDLE message_handle = IoTHubMessage_CreateFromString(telemetry);
        IoTHubDeviceClient_LL_SendEventAsync(_device_ll_handle, message_handle, NULL, NULL);
        IoTHubMessage_Destroy(message_handle);
    }
    

    এই কোডটি একটি IoT Hub বার্তা তৈরি করে একটি প্যারামিটার হিসেবে পাস করা স্ট্রিং থেকে, এটি হাবে পাঠায়, তারপর বার্তা অবজেক্টটি পরিষ্কার করে।

  2. loop ফাংশনে এই কোডটি কল করুন, টেলিমেট্রি সিরিয়াল পোর্টে পাঠানোর লাইনের ঠিক পরে:

    sendTelemetry(telemetry.c_str());
    

কমান্ড হ্যান্ডল করুন

আপনার ডিভাইসকে সার্ভার কোড থেকে একটি কমান্ড হ্যান্ডল করতে হবে রিলে নিয়ন্ত্রণ করার জন্য। এটি একটি ডিরেক্ট মেথড অনুরোধ হিসেবে পাঠানো হয়।

কাজ - একটি ডিরেক্ট মেথড অনুরোধ হ্যান্ডল করুন

  1. connectIoTHub ফাংশনের আগে নিম্নলিখিত কোড যোগ করুন:

    int directMethodCallback(const char *method_name, const unsigned char *payload, size_t size, unsigned char **response, size_t *response_size, void *userContextCallback)
    {
        Serial.printf("Direct method received %s\r\n", method_name);
    
        if (strcmp(method_name, "relay_on") == 0)
        {
            digitalWrite(PIN_WIRE_SCL, HIGH);
        }
        else if (strcmp(method_name, "relay_off") == 0)
        {
            digitalWrite(PIN_WIRE_SCL, LOW);
        }
    }
    

    এই কোডটি একটি কলব্যাক মেথড সংজ্ঞায়িত করে যা IoT Hub লাইব্রেরি একটি ডিরেক্ট মেথড অনুরোধ পাওয়ার সময় কল করতে পারে। অনুরোধ করা মেথডটি method_name প্যারামিটারে পাঠানো হয়। এই ফাংশনটি সিরিয়াল পোর্টে কল করা মেথডটি প্রিন্ট করে, তারপর মেথড নামের উপর ভিত্তি করে রিলে চালু বা বন্ধ করে।

    💁 এটি একটি একক ডিরেক্ট মেথড অনুরোধে বাস্তবায়িত হতে পারে, যেখানে রিলের কাঙ্ক্ষিত অবস্থাটি একটি পে-লোডে পাস করা হয় যা মেথড অনুরোধের সাথে পাস করা যায় এবং payload প্যারামিটার থেকে পাওয়া যায়।

  2. directMethodCallback ফাংশনের শেষে নিম্নলিখিত কোড যোগ করুন:

    char resultBuff[16];
    sprintf(resultBuff, "{\"Result\":\"\"}");
    *response_size = strlen(resultBuff);
    *response = (unsigned char *)malloc(*response_size);
    memcpy(*response, resultBuff, *response_size);
    
    return IOTHUB_CLIENT_OK;
    

    ডিরেক্ট মেথড অনুরোধগুলির একটি প্রতিক্রিয়া প্রয়োজন, এবং প্রতিক্রিয়া দুটি অংশে থাকে - একটি টেক্সট হিসেবে প্রতিক্রিয়া এবং একটি রিটার্ন কোড। এই কোডটি নিম্নলিখিত JSON ডকুমেন্ট হিসেবে একটি রেজাল্ট তৈরি করবে:

    {
        "Result": ""
    }
    

    এটি তারপর response প্যারামিটারে কপি করা হয়, এবং এই প্রতিক্রিয়ার আকার response_size প্যারামিটারে সেট করা হয়। এই কোডটি তারপর IOTHUB_CLIENT_OK রিটার্ন করে দেখানোর জন্য যে মেথডটি সঠিকভাবে হ্যান্ডল করা হয়েছে।

  3. কলব্যাকটি সংযুক্ত করতে connectIoTHub ফাংশনের শেষে নিম্নলিখিত যোগ করুন:

    IoTHubClient_LL_SetDeviceMethodCallback(_device_ll_handle, directMethodCallback, NULL);
    
  4. loop ফাংশনটি IoTHubDeviceClient_LL_DoWork ফাংশন কল করবে IoT Hub দ্বারা পাঠানো ইভেন্টগুলি প্রক্রিয়া করার জন্য। এটি delay এর কারণে শুধুমাত্র প্রতি 10 সেকেন্ডে কল করা হয়, যার মানে ডিরেক্ট মেথডগুলি শুধুমাত্র প্রতি 10 সেকেন্ডে প্রক্রিয়া করা হয়। এটি আরও কার্যকর করতে, 10 সেকেন্ডের বিলম্বটি অনেক ছোট বিলম্বে বাস্তবায়িত হতে পারে, প্রতিবার IoTHubDeviceClient_LL_DoWork কল করে। এটি করতে, loop ফাংশনের উপরে নিম্নলিখিত কোড যোগ করুন:

    void work_delay(int delay_time)
    {
        int current = 0;
        do
        {
            IoTHubDeviceClient_LL_DoWork(_device_ll_handle);
            delay(100);
            current += 100;
        } while (current < delay_time);
    }
    

    এই কোডটি বারবার লুপ করবে, IoTHubDeviceClient_LL_DoWork কল করবে এবং প্রতিবার 100ms বিলম্ব করবে। এটি যতবার প্রয়োজন ততবার করবে delay_time প্যারামিটারে দেওয়া সময় বিলম্ব করার জন্য। এর মানে হলো ডিভাইসটি ডিরেক্ট মেথড অনুরোধ প্রক্রিয়া করার জন্য সর্বাধিক 100ms অপেক্ষা করছে।

  5. loop ফাংশনে IoTHubDeviceClient_LL_DoWork কলটি সরিয়ে ফেলুন এবং delay(10000) কলটি নিম্নলিখিত দিয়ে প্রতিস্থাপন করুন এই নতুন ফাংশনটি কল করতে:

    work_delay(10000);
    

💁 আপনি এই কোডটি code/wio-terminal ফোল্ডারে খুঁজে পেতে পারেন।

😀 আপনার মাটি আর্দ্রতা সেন্সর প্রোগ্রামটি IoT Hub-এর সাথে সংযুক্ত হয়েছে!


অস্বীকৃতি:
এই নথিটি AI অনুবাদ পরিষেবা Co-op Translator ব্যবহার করে অনুবাদ করা হয়েছে। আমরা যথাসম্ভব সঠিকতার জন্য চেষ্টা করি, তবে অনুগ্রহ করে মনে রাখবেন যে স্বয়ংক্রিয় অনুবাদে ত্রুটি বা অসঙ্গতি থাকতে পারে। মূল ভাষায় থাকা নথিটিকে প্রামাণিক উৎস হিসেবে বিবেচনা করা উচিত। গুরুত্বপূর্ণ তথ্যের জন্য, পেশাদার মানব অনুবাদ সুপারিশ করা হয়। এই অনুবাদ ব্যবহারের ফলে কোনো ভুল বোঝাবুঝি বা ভুল ব্যাখ্যা হলে আমরা দায়বদ্ধ থাকব না।